patternjavaMinor
TopCoder problem "Substitute" - SRM 160 (Division II Level One)
Viewed 0 times
problem160divisionlevelsubstitutetopcodersrmone
Problem
I have solved the problem "Substitute" at Topcoder.
A simple, easy to remember system for encoding integer amounts can be very useful. For example, dealers at flea markets put the information about an item on a card that they let potential buyers see. They find it advantageous to encode the amount they originally paid for the item on the card.
A good system is to use a substitution code, in which each digit is encoded by a letter. An easy to remember 10-letter word or phrase, the key, is chosen. Every '1' in the value is replaced by the first letter of the key, every '2' is replaced by the second letter of the key, and so on. Every '0' is replaced by the last letter of the key. Letters that do not appear in the key can be inserted anywhere without affecting the value represented by the code.. This helps to make the resulting code much harder to break (without knowing the key).
Create a class Substitute that contains the method getValue that is given the Strings key and code as input and that returns the decoded value.
I started studying Big O notation and I think the complexity of my program is \$O(n^2)\$. How can I make my program more efficient and clean? Can I make it \$O(n)\$?
A simple, easy to remember system for encoding integer amounts can be very useful. For example, dealers at flea markets put the information about an item on a card that they let potential buyers see. They find it advantageous to encode the amount they originally paid for the item on the card.
A good system is to use a substitution code, in which each digit is encoded by a letter. An easy to remember 10-letter word or phrase, the key, is chosen. Every '1' in the value is replaced by the first letter of the key, every '2' is replaced by the second letter of the key, and so on. Every '0' is replaced by the last letter of the key. Letters that do not appear in the key can be inserted anywhere without affecting the value represented by the code.. This helps to make the resulting code much harder to break (without knowing the key).
Create a class Substitute that contains the method getValue that is given the Strings key and code as input and that returns the decoded value.
I started studying Big O notation and I think the complexity of my program is \$O(n^2)\$. How can I make my program more efficient and clean? Can I make it \$O(n)\$?
public class Substitute {
public int getValue(String key, String code) {
String s = "";
for (int i = 0; i = 10) {
x = 0;
}
s = s + x;
}
}
}
return Integer.parseInt(s);
}
}Solution
Analysis
You've picked an interesting problem about which to ask about Big-O complexity. The problem is, you're being sloppy about what you call "n". This function has two inputs: a
The outer
The complexity, therefore, is O(n ∙ m (m + n) + n). If we consider m to be constant, that simplifies to O(n2).
However, the "silliest" part of the code is
Perspective
It's important to keep a sense of perspective, though. Big-O complexity is useful for getting a feel for how well an algorithm scales to handle large input. For example, it can help you estimate whether an algorithm that analyzes a million DNA bases will finish in seconds or hours.
On the other hand, Big-O analysis is nearly useless for predicting the performance of "small" problems like this one. The challenge states that
Suggested solution
Building a string and then parsing it as an integer is inefficient. Why not just compute the integer as you go?
You've picked an interesting problem about which to ask about Big-O complexity. The problem is, you're being sloppy about what you call "n". This function has two inputs: a
key of length 10 and a code of variable length. Let's say that m is the length of key (which is 10) and n is the length of code.The outer
i loop executes n times. For each of those, outer loops, the inner j loop executes m times. For each inner loop, key.indexOf() — an O(m) operation — is called; then we might do s = s + x — an O(n) operation. Finally, Integer.parseInt(s) is an operation that I would expect to be O(n).The complexity, therefore, is O(n ∙ m (m + n) + n). If we consider m to be constant, that simplifies to O(n2).
However, the "silliest" part of the code is
s = s + x. That O(n) operation could be O(1) if you had used a StringBuilder instead of repeated string concatenation. With a StringBuilder, your function would become O(n).Perspective
It's important to keep a sense of perspective, though. Big-O complexity is useful for getting a feel for how well an algorithm scales to handle large input. For example, it can help you estimate whether an algorithm that analyzes a million DNA bases will finish in seconds or hours.
On the other hand, Big-O analysis is nearly useless for predicting the performance of "small" problems like this one. The challenge states that
n is at most 9 — and in any case, a result larger than 9 digits would overflow an int. It is very possible that an algorithm that Big-O analysis predicts will scale well could actually perform worse than a simpler algorithm that in theory should scale poorly. For example, the time it takes to initialize an "efficient" data structure could exceed the time it takes to do a naïve String.indexOf() search.Suggested solution
Building a string and then parsing it as an integer is inefficient. Why not just compute the integer as you go?
public int getValue(String key, String code) {
int value = 0;
for (int i = 0; i 0) {
value = 10 * value + pos;
}
}
return value;
}Code Snippets
public int getValue(String key, String code) {
int value = 0;
for (int i = 0; i < code.length(); i++) {
int pos = 1 + key.indexOf(code.charAt(i));
if (pos == 10) {
value *= 10;
} else if (pos > 0) {
value = 10 * value + pos;
}
}
return value;
}Context
StackExchange Code Review Q#131699, answer score: 6
Revisions (0)
No revisions yet.