LeetCode – String to Integer (atoi) (Java)
mplement atoi to convert a string to an integer.
Hint: Carefully consider all possible input cases. If you want a challenge, please do not see below and ask yourself what are the possible input cases.
http://www.cnblogs.com/grandyang/p/4125537.html
http://blog.csdn.net/linhuanmars/article/details/21145129
这道题还是对于Integer的处理,在Reverse Integer这道题中我有提到,这种题的考察重点并不在于问题本身,而是要注意corner case的处理,整数一般有两点,一个是正负符号问题,另一个是整数越界问题。思路比较简单,就是先去掉多余的空格字符,然后读符号(注意正负号都有可能,也有可能没有符号),接下来按顺序读数字,结束条件有三种情况:(1)异常字符出现(按照C语言的标准是把异常字符起的后面全部截去,保留前面的部分作为结果);(2)数字越界(返回最接近的整数);(3)字符串结束。
http://www.cnblogs.com/springfor/p/3896499.html
http://jane4532.blogspot.com/2013/09/string-to-integerleetcode.html
http://www.cnblogs.com/springfor/p/3896499.html
Cache/pre-compute Integer.MAX_VALUE/10, Integer.MAX_VALUE%10
2 if (str == null || str.length() < 1)
3 return 0;
4
5 // trim white spaces at beginning and end
6 str = str.trim();
7
8 char flag = '+';
9
10 // check negative or positive
11 int i = 0;
12 if (str.charAt(0) == '-') {
13 flag = '-';
14 i++;
15 } else if (str.charAt(0) == '+') {
16 i++;
17 }
18 // use double to store result
19 int result = 0;
20
21 // calculate value
22 while (str.length() > i && str.charAt(i) >= '0' && str.charAt(i) <= '9') {
23 if(Integer.MAX_VALUE/10 < result || (Integer.MAX_VALUE/10 == result && Integer.MAX_VALUE%10 < (str.charAt(i) - '0'))) //\\
24 return flag == '-' ? Integer.MIN_VALUE : Integer.MAX_VALUE;
26 result = result * 10 + (str.charAt(i) - '0');
27 i++;
28 }
29
30 if (flag == '-')
31 result = -result;
32
33 return result;
34 }
https://leetcode.com/discuss/8886/my-simple-solution
http://n00tc0d3r.blogspot.com/2013/06/string-to-integer-atoi.html
If we are allowed to use Java regex (FYI, here is a tutorial of Java Regex), then we can remove leading whitespace and trailing invalid characters using replaceAll method of String class.
mplement atoi to convert a string to an integer.
Hint: Carefully consider all possible input cases. If you want a challenge, please do not see below and ask yourself what are the possible input cases.
1. null or empty string 2. white spaces 3. +/- sign 4. calculate real value 5. handle min & max
The function first discards as many whitespace characters as necessary until the first non-whitespace character is found. Then, starting from this character, takes an optional initial plus or minus sign followed by as many numerical digits as possible, and interprets them as a numerical value.
The string can contain additional characters after those that form the integral number, which are ignored and have no effect on the behavior of this function.
If the first sequence of non-whitespace characters in str is not a valid integral number, or if no such sequence exists because either str is empty or it contains only whitespace characters, no conversion is performed.
If no valid conversion could be performed, a zero value is returned. If the correct value is out of the range of representable values, INT_MAX (2147483647) or INT_MIN (-2147483648) is returned.
字符串转为整数是很常用的一个函数,由于输入的是字符串,所以需要考虑的情况有很多种。我之前有一篇文章是关于验证一个字符串是否为数字的,参见 Valid Number。在那篇文章中,详细的讨论了各种情况,包括符号,自然数,小数点的出现位置,判断他们是否是数字。个人以为这道题也应该有这么多种情况。但是这题只需要考虑数字和符号的情况:
1. 若字符串开头是空格,则跳过所有空格,到第一个非空格字符,如果没有,则返回0.
2. 若第一个非空格字符是符号+/-,则标记sign的真假,这道题还有个局限性,那就是在c++里面,+-1和-+1都是认可的,都是-1,而在此题里,则会返回0.
3. 若下一个字符不是数字,则返回0. 完全不考虑小数点和自然数的情况,不过这样也好,起码省事了不少。
4. 如果下一个字符是数字,则转为整形存下来,若接下来再有非数字出现,则返回目前的结果。
5. 还需要考虑边界问题,如果超过了整形数的范围,则用边界值替代当前值。
public int myAtoi(String str) { if (str.isEmpty()) return 0; int sign = 1, base = 0, i = 0, n = str.length(); while (i < n && str.charAt(i) == ' ') ++i; if (i < n && (str.charAt(i) == '+' || str.charAt(i) == '-')) { sign = (str.charAt(i++) == '+') ? 1 : -1; } while (i < n && str.charAt(i) >= '0' && str.charAt(i) <= '9') { if (base > Integer.MAX_VALUE / 10 || (base == Integer.MAX_VALUE / 10 && str.charAt(i) - '0' > 7)) { return (sign == 1) ? Integer.MAX_VALUE : Integer.MIN_VALUE; } base = 10 * base + (str.charAt(i++) - '0'); } return base * sign; }http://www.jiuzhang.com/solutions/string-to-integer-atoi/
public int myAtoi(String str) { // Start typing your Java solution below // DO NOT write main() function if(str == null) { return 0; } str = str.trim(); if (str.length() == 0) { return 0; } int sign = 1; int index = 0; if (str.charAt(index) == '+') { index++; } else if (str.charAt(index) == '-') { sign = -1; index++; } long num = 0; for (; index < str.length(); index++) { if (str.charAt(index) < '0' || str.charAt(index) > '9') break; num = num * 10 + (str.charAt(index) - '0'); if (num > Integer.MAX_VALUE ) { break; } } if (num * sign >= Integer.MAX_VALUE) { return Integer.MAX_VALUE; } if (num * sign <= Integer.MIN_VALUE) { return Integer.MIN_VALUE; } return (int)num * sign; }http://www.programcreek.com/2012/12/leetcode-string-to-integer-atoi/
public int atoi(String str) { if (str == null || str.length() < 1) return 0; // trim white spaces str = str.trim(); char flag = '+'; // check negative or positive int i = 0; if (str.charAt(0) == '-') { flag = '-'; i++; } else if (str.charAt(0) == '+') { i++; } // use double to store result double result = 0; //\\ use long // calculate value while (str.length() > i && str.charAt(i) >= '0' && str.charAt(i) <= '9') {//\\ result = result * 10 + (str.charAt(i) - '0'); i++; } if (flag == '-') result = -result; // handle max and min if (result > Integer.MAX_VALUE) return Integer.MAX_VALUE; if (result < Integer.MIN_VALUE) return Integer.MIN_VALUE; return (int) result; } |
这道题还是对于Integer的处理,在Reverse Integer这道题中我有提到,这种题的考察重点并不在于问题本身,而是要注意corner case的处理,整数一般有两点,一个是正负符号问题,另一个是整数越界问题。思路比较简单,就是先去掉多余的空格字符,然后读符号(注意正负号都有可能,也有可能没有符号),接下来按顺序读数字,结束条件有三种情况:(1)异常字符出现(按照C语言的标准是把异常字符起的后面全部截去,保留前面的部分作为结果);(2)数字越界(返回最接近的整数);(3)字符串结束。
public int atoi(String str) {
if(str==null)
{
return 0;
}
str = str.trim();
if(str.length()==0)
return 0;
boolean isNeg = false;
int i = 0;
if(str.charAt(0)=='-' || str.charAt(0)=='+')
{
i++;
if(str.charAt(0)=='-')
isNeg = true;
}
int res = 0;
while(i<str.length())
{
if(str.charAt(i)<'0'||str.charAt(i)>'9')
break;
int digit = (int)(str.charAt(i)-'0');
if(isNeg && res>-((Integer.MIN_VALUE+digit)/10))//\\
return Integer.MIN_VALUE;
else if(!isNeg && res>(Integer.MAX_VALUE-digit)/10)//\\
return Integer.MAX_VALUE;
res = res*10+digit;
i++;
}
return isNeg?-res:res;
}
public static int myAtoi(String str) {
if (str == null || str.length() == 0)
return 0;//
str = str.trim();
char firstChar = str.charAt(0);
int sign = 1, start = 0, len = str.length();
long sum = 0;
if (firstChar == '+') {
sign = 1;
start++;
} else if (firstChar == '-') {
sign = -1;
start++;
}
for (int i = start; i < len; i++) {
if (!Character.isDigit(str.charAt(i)))
return (int) sum * sign;
sum = sum * 10 + str.charAt(i) - '0';
if (sign == 1 && sum > Integer.MAX_VALUE)
return Integer.MAX_VALUE;
if (sign == -1 && (-1) * sum < Integer.MIN_VALUE)
return Integer.MIN_VALUE;
}
return (int) sum * sign;
}
因为遇见过了atol,string to long 这样的问题,就不能用这种比当前数据类型长的方法解决。
另一种方法是更加普遍和巧妙的,就是用这样的判断:
1. Integer.MAX_VALUE/10 < result; //当前转换结果比Integer中最大值/10还大(因为这个判断放在while循环最开始,之后还要对result进行*10+当前遍历元素的操作,所以如果还乘10的result已经比Integer.MAX_VALUE/10还大,可想而知,乘了10更大)
2. Integer.MAX_VALUE/10 == result && Integer.MAX_VALUE%10 <(str.charAt(i) - '0') //另外的情况就是,当前result恰跟 Integer.MAX_VALUE/10相等,那么就判断当前遍历的元素值跟最大值的最后一位的大小关系即可
1 public int atoi(String str) {
2 if (str == null || str.length() < 1)
3 return 0;
4
5 // trim white spaces at beginning and end
6 str = str.trim();
7
8 char flag = '+';
9
10 // check negative or positive
11 int i = 0;
12 if (str.charAt(0) == '-') {
13 flag = '-';
14 i++;
15 } else if (str.charAt(0) == '+') {
16 i++;
17 }
18 // use double to store result
19 int result = 0;
20
21 // calculate value
22 while (str.length() > i && str.charAt(i) >= '0' && str.charAt(i) <= '9') {
23 if(Integer.MAX_VALUE/10 < result || (Integer.MAX_VALUE/10 == result && Integer.MAX_VALUE%10 < (str.charAt(i) - '0'))) //\\
24 return flag == '-' ? Integer.MIN_VALUE : Integer.MAX_VALUE;
25
26 result = result * 10 + (str.charAt(i) - '0');
27 i++;
28 }
29
30 if (flag == '-')
31 result = -result;
32
33 return result;
34 }
2 if (str == null || str.length() < 1)
3 return 0;
4
5 // trim white spaces at beginning and end
6 str = str.trim();
7
8 char flag = '+';
9
10 // check negative or positive
11 int i = 0;
12 if (str.charAt(0) == '-') {
13 flag = '-';
14 i++;
15 } else if (str.charAt(0) == '+') {
16 i++;
17 }
18 // use double to store result
19 int result = 0;
20
21 // calculate value
22 while (str.length() > i && str.charAt(i) >= '0' && str.charAt(i) <= '9') {
23 if(Integer.MAX_VALUE/10 < result || (Integer.MAX_VALUE/10 == result && Integer.MAX_VALUE%10 < (str.charAt(i) - '0'))) //\\
24 return flag == '-' ? Integer.MIN_VALUE : Integer.MAX_VALUE;
25
26 result = result * 10 + (str.charAt(i) - '0');
27 i++;
28 }
29
30 if (flag == '-')
31 result = -result;
32
33 return result;
34 }
http://jane4532.blogspot.com/2013/09/string-to-integerleetcode.html
http://www.cnblogs.com/springfor/p/3896499.html
Cache/pre-compute Integer.MAX_VALUE/10, Integer.MAX_VALUE%10
另一种方法是更加普遍和巧妙的,就是用这样的判断:
1. Integer.MAX_VALUE/10 < result; //当前转换结果比Integer中最大值/10还大(因为这个判断放在while循环最开始,之后还要对result进行*10+当前遍历元素的操作,所以如果还乘10的result已经比Integer.MAX_VALUE/10还大,可想而知,乘了10更大)
2. Integer.MAX_VALUE/10 == result && Integer.MAX_VALUE%10 <(str.charAt(i) - '0') //另外的情况就是,当前result恰跟 Integer.MAX_VALUE/10相等,那么就判断当前遍历的元素值跟最大值的最后一位的大小关系即可
1 public int atoi(String str) {2 if (str == null || str.length() < 1)
3 return 0;
4
5 // trim white spaces at beginning and end
6 str = str.trim();
7
8 char flag = '+';
9
10 // check negative or positive
11 int i = 0;
12 if (str.charAt(0) == '-') {
13 flag = '-';
14 i++;
15 } else if (str.charAt(0) == '+') {
16 i++;
17 }
18 // use double to store result
19 int result = 0;
20
21 // calculate value
22 while (str.length() > i && str.charAt(i) >= '0' && str.charAt(i) <= '9') {
23 if(Integer.MAX_VALUE/10 < result || (Integer.MAX_VALUE/10 == result && Integer.MAX_VALUE%10 < (str.charAt(i) - '0'))) //\\
24 return flag == '-' ? Integer.MIN_VALUE : Integer.MAX_VALUE;
26 result = result * 10 + (str.charAt(i) - '0');
27 i++;
28 }
29
30 if (flag == '-')
31 result = -result;
32
33 return result;
34 }
II:
1 public int atoi(String str) {
2 if (str == null || str.length() < 1)
3 return 0;
4
5 // trim white spaces
6 str = str.trim();
7
8 char flag = '+';
9
10 // check negative or positive
11 int i = 0;
12 if (str.charAt(0) == '-') {
13 flag = '-';
14 i++;
15 } else if (str.charAt(0) == '+') {
16 i++;
17 }
18 // use double to store result
19 double result = 0;
20
21 // calculate value
22 while (str.length() > i && str.charAt(i) >= '0' && str.charAt(i) <= '9') {
23 result = result * 10 + (str.charAt(i) - '0');
24 i++;
25 }
26
27 if (flag == '-')
28 result = -result;
29
30 // handle max and min
31 if (result > Integer.MAX_VALUE)
32 return Integer.MAX_VALUE;
33
34 if (result < Integer.MIN_VALUE)
35 return Integer.MIN_VALUE;
36
37 return (int) result;
38 }
2 if (str == null || str.length() < 1)
3 return 0;
4
5 // trim white spaces
6 str = str.trim();
7
8 char flag = '+';
9
10 // check negative or positive
11 int i = 0;
12 if (str.charAt(0) == '-') {
13 flag = '-';
14 i++;
15 } else if (str.charAt(0) == '+') {
16 i++;
17 }
18 // use double to store result
19 double result = 0;
20
21 // calculate value
22 while (str.length() > i && str.charAt(i) >= '0' && str.charAt(i) <= '9') {
23 result = result * 10 + (str.charAt(i) - '0');
24 i++;
25 }
26
27 if (flag == '-')
28 result = -result;
29
30 // handle max and min
31 if (result > Integer.MAX_VALUE)
32 return Integer.MAX_VALUE;
33
34 if (result < Integer.MIN_VALUE)
35 return Integer.MIN_VALUE;
36
37 return (int) result;
38 }
https://leetcode.com/discuss/8886/my-simple-solution
I think we only need to handle four cases:
- discards all leading whitespaces
- sign of the number
- overflow
- invalid input
http://n00tc0d3r.blogspot.com/2013/06/string-to-integer-atoi.html
public int atoi(String str) {
// use regex to validate the input
String pattern = "(^\\s*)([+-]?[0-9]+)(.*$)";
if (!str.matches(pattern)) return 0;
// trim the string to [+-]?[0-9]+
String numstr = str.replaceAll(pattern, "$2");
// convert string to integer
int sign = (numstr.charAt(0) == '-') ? -1 : 1;
long res = 0;
for (int i = (numstr.charAt(0) == '+' || numstr.charAt(0) == '-') ? 1 : 0; i<numstr.length(); ++i) {
res = res * 10 + (numstr.charAt(i) - '0');
if (sign > 0 && res > Integer.MAX_VALUE) return Integer.MAX_VALUE;
else if (sign < 0 && res * (-1) < Integer.MIN_VALUE) return Integer.MIN_VALUE;
}
return (int)res*sign;
}
If we are not allowed to use replaceAll method, we need to iterate the input string character by character.
public int atoi(String str) {
long res = 0; int index = 0;
// trim leading whitespace
while (index < str.length() && str.charAt(index) == ' ') ++index;
if (index == str.length()) return (int)res;
// get sign
int sign = 1;
if (str.charAt(index) == '-') {
++index;
sign = -1;
} else if (str.charAt(index) == '+') {
++index;
}
// convert digits
while (index < str.length() && Character.isDigit(str.charAt(index))) {
res = res * 10 + (str.charAt(index++) - '0');
if (sign > 0 && res > Integer.MAX_VALUE) return Integer.MAX_VALUE;
else if (sign < 0 && res * (-1) < Integer.MIN_VALUE) return Integer.MIN_VALUE;
}
return (int)res*sign;
}
Read full article from LeetCode – String to Integer (atoi) (Java)