我通常使用以下习惯用法来检查String是否可以转换为整数。

public boolean isInteger( String input ) {
    try {
        Integer.parseInt( input );
        return true;
    }
    catch( Exception e ) {
        return false;
    }
}

这只是我,还是看起来有点hackish?什么是更好的方式?


通过早期答案的基准测试.com / users / 28278 / codingwithspike“> CodingWithSpike )看看为什么我改变了立场并接受了 Jonas Klemming的回答这个问题。我认为这个原始代码将被大多数人使用,因为它实现起来更快,更易于维护,但是当提供非整数数据时,它会慢一个数量级。

有帮助吗?

解决方案

如果您不关心潜在的溢出问题,此函数的执行速度比使用 Integer.parseInt()快20-30倍。

public static boolean isInteger(String str) {
    if (str == null) {
        return false;
    }
    int length = str.length();
    if (length == 0) {
        return false;
    }
    int i = 0;
    if (str.charAt(0) == '-') {
        if (length == 1) {
            return false;
        }
        i = 1;
    }
    for (; i < length; i++) {
        char c = str.charAt(i);
        if (c < '0' || c > '9') {
            return false;
        }
    }
    return true;
}

其他提示

你拥有它,但你应该只捕获 NumberFormatException

因为有可能人们仍然访问这里并且在基准测试后会对Regex产生偏见......所以我将提供一个更新版本的基准测试,以及Regex的编译版本。与之前的基准测试相反,这一点显示Regex解决方案实际上具有始终如一的良好性能。

从比尔蜥蜴复制并更新编译版本:

private final Pattern pattern = Pattern.compile("^-?\\d+
ByException - integer data: 45
ByException - non-integer data: 465

ByRegex - integer data: 272
ByRegex - non-integer data: 131

ByCompiledRegex - integer data: 45
ByCompiledRegex - non-integer data: 26

ByJonas - integer data: 8
ByJonas - non-integer data: 2
quot;); public void runTests() { String big_int = "1234567890"; String non_int = "1234XY7890"; long startTime = System.currentTimeMillis(); for(int i = 0; i < 100000; i++) IsInt_ByException(big_int); long endTime = System.currentTimeMillis(); System.out.print("ByException - integer data: "); System.out.println(endTime - startTime); startTime = System.currentTimeMillis(); for(int i = 0; i < 100000; i++) IsInt_ByException(non_int); endTime = System.currentTimeMillis(); System.out.print("ByException - non-integer data: "); System.out.println(endTime - startTime); startTime = System.currentTimeMillis(); for(int i = 0; i < 100000; i++) IsInt_ByRegex(big_int); endTime = System.currentTimeMillis(); System.out.print("\nByRegex - integer data: "); System.out.println(endTime - startTime); startTime = System.currentTimeMillis(); for(int i = 0; i < 100000; i++) IsInt_ByRegex(non_int); endTime = System.currentTimeMillis(); System.out.print("ByRegex - non-integer data: "); System.out.println(endTime - startTime); startTime = System.currentTimeMillis(); for (int i = 0; i < 100000; i++) IsInt_ByCompiledRegex(big_int); endTime = System.currentTimeMillis(); System.out.print("\nByCompiledRegex - integer data: "); System.out.println(endTime - startTime); startTime = System.currentTimeMillis(); for (int i = 0; i < 100000; i++) IsInt_ByCompiledRegex(non_int); endTime = System.currentTimeMillis(); System.out.print("ByCompiledRegex - non-integer data: "); System.out.println(endTime - startTime); startTime = System.currentTimeMillis(); for(int i = 0; i < 100000; i++) IsInt_ByJonas(big_int); endTime = System.currentTimeMillis(); System.out.print("\nByJonas - integer data: "); System.out.println(endTime - startTime); startTime = System.currentTimeMillis(); for(int i = 0; i < 100000; i++) IsInt_ByJonas(non_int); endTime = System.currentTimeMillis(); System.out.print("ByJonas - non-integer data: "); System.out.println(endTime - startTime); } private boolean IsInt_ByException(String str) { try { Integer.parseInt(str); return true; } catch(NumberFormatException nfe) { return false; } } private boolean IsInt_ByRegex(String str) { return str.matches("^-?\\d+<*>quot;); } private boolean IsInt_ByCompiledRegex(String str) { return pattern.matcher(str).find(); } public boolean IsInt_ByJonas(String str) { if (str == null) { return false; } int length = str.length(); if (length == 0) { return false; } int i = 0; if (str.charAt(0) == '-') { if (length == 1) { return false; } i = 1; } for (; i < length; i++) { char c = str.charAt(i); if (c <= '/' || c >= ':') { return false; } } return true; }

结果:

<*>

做了快速基准测试。除非您开始弹出多个方法并且JVM必须做很多工作才能使执行堆栈到位,否则异常实际上并不是那种费用。当采用相同的方法时,他们的表现并不差。

 public void RunTests()
 {
     String str = "1234567890";

     long startTime = System.currentTimeMillis();
     for(int i = 0; i < 100000; i++)
         IsInt_ByException(str);
     long endTime = System.currentTimeMillis();
     System.out.print("ByException: ");
     System.out.println(endTime - startTime);

     startTime = System.currentTimeMillis();
     for(int i = 0; i < 100000; i++)
         IsInt_ByRegex(str);
     endTime = System.currentTimeMillis();
     System.out.print("ByRegex: ");
     System.out.println(endTime - startTime);

     startTime = System.currentTimeMillis();
     for(int i = 0; i < 100000; i++)
         IsInt_ByJonas(str);
     endTime = System.currentTimeMillis();
     System.out.print("ByJonas: ");
     System.out.println(endTime - startTime);
 }

 private boolean IsInt_ByException(String str)
 {
     try
     {
         Integer.parseInt(str);
         return true;
     }
     catch(NumberFormatException nfe)
     {
         return false;
     }
 }

 private boolean IsInt_ByRegex(String str)
 {
     return str.matches("^-?\\d+<*>quot;);
 }

 public boolean IsInt_ByJonas(String str)
 {
     if (str == null) {
             return false;
     }
     int length = str.length();
     if (length == 0) {
             return false;
     }
     int i = 0;
     if (str.charAt(0) == '-') {
             if (length == 1) {
                     return false;
             }
             i = 1;
     }
     for (; i < length; i++) {
             char c = str.charAt(i);
             if (c <= '/' || c >= ':') {
                     return false;
             }
     }
     return true;
 }

输出:

  

ByException:31

     

ByRegex:453(注意:每次重新编译模式)

     

ByJonas:16

我同意Jonas K的解决方案也是最强大的。看起来他赢了:)

org.apache.commons.lang.StringUtils.isNumeric 

尽管Java的标准库确实错过了这样的实用函数

我认为Apache Commons是“必须拥有的”对于每个Java程序员

太糟糕了,它还没有移植到Java5

部分取决于你的意思“可以转换为整数”。

如果您的意思是“可以在Java中转换为int”,那么乔纳斯的答案是一个好的开始,但还没有完成这项工作。例如,它将通过99999999999999999999999999999。我会在方法结束时从你自己的问题中添加正常的try / catch调用。

逐字符检查将有效地拒绝“根本不是整数”。案例,留下“它是一个整数,但Java无法处理它”由较慢的异常路由捕获的情况。您也可以手动执行此操作,但它会更复杂很多

只有一条关于regexp的评论。这里提供的每个例子都是错的!如果你想使用正则表达式,不要忘记编译模式需要花费很多时间。这样:

str.matches("^-?\\d+
Pattern.matches("-?\\d+", input);
quot;)

还有这个:

import java.util.regex.Pattern;

/**
 * @author Rastislav Komara
 */
public class NaturalNumberChecker {
    public static final Pattern PATTERN = Pattern.compile("^\\d+<*>quot;);

    boolean isNaturalNumber(CharSequence input) {
        return input != null && PATTERN.matcher(input).matches();
    }
}

导致在每个方法调用中编译模式。要正确使用它,请按照:

<*>

我从rally25rs回复中复制了代码,并为非整数数据添加了一些测试。无可否认,结果有利于Jonas Klemming发布的方法。当你有整数数据时,我最初发布的Exception方法的结果非常好,但是当你没有整数数据时它们是最差的,而RegEx解决方案的结果(我敢打赌很多人使用) 一直坏。有关已编译的正则表达式示例,请参阅 Felipe的答案,该示例要快得多。

public void runTests()
{
    String big_int = "1234567890";
    String non_int = "1234XY7890";

    long startTime = System.currentTimeMillis();
    for(int i = 0; i < 100000; i++)
        IsInt_ByException(big_int);
    long endTime = System.currentTimeMillis();
    System.out.print("ByException - integer data: ");
    System.out.println(endTime - startTime);

    startTime = System.currentTimeMillis();
    for(int i = 0; i < 100000; i++)
        IsInt_ByException(non_int);
    endTime = System.currentTimeMillis();
    System.out.print("ByException - non-integer data: ");
    System.out.println(endTime - startTime);

    startTime = System.currentTimeMillis();
    for(int i = 0; i < 100000; i++)
        IsInt_ByRegex(big_int);
    endTime = System.currentTimeMillis();
    System.out.print("\nByRegex - integer data: ");
    System.out.println(endTime - startTime);

    startTime = System.currentTimeMillis();
    for(int i = 0; i < 100000; i++)
        IsInt_ByRegex(non_int);
    endTime = System.currentTimeMillis();
    System.out.print("ByRegex - non-integer data: ");
    System.out.println(endTime - startTime);

    startTime = System.currentTimeMillis();
    for(int i = 0; i < 100000; i++)
        IsInt_ByJonas(big_int);
    endTime = System.currentTimeMillis();
    System.out.print("\nByJonas - integer data: ");
    System.out.println(endTime - startTime);

    startTime = System.currentTimeMillis();
    for(int i = 0; i < 100000; i++)
        IsInt_ByJonas(non_int);
    endTime = System.currentTimeMillis();
    System.out.print("ByJonas - non-integer data: ");
    System.out.println(endTime - startTime);
}

private boolean IsInt_ByException(String str)
{
    try
    {
        Integer.parseInt(str);
        return true;
    }
    catch(NumberFormatException nfe)
    {
        return false;
    }
}

private boolean IsInt_ByRegex(String str)
{
    return str.matches("^-?\\d+
ByException - integer data: 47
ByException - non-integer data: 547

ByRegex - integer data: 390
ByRegex - non-integer data: 313

ByJonas - integer data: 0
ByJonas - non-integer data: 16
quot;); } public boolean IsInt_ByJonas(String str) { if (str == null) { return false; } int length = str.length(); if (length == 0) { return false; } int i = 0; if (str.charAt(0) == '-') { if (length == 1) { return false; } i = 1; } for (; i < length; i++) { char c = str.charAt(i); if (c <= '/' || c >= ':') { return false; } } return true; }

结果:

<*>

有番石榴版:

import com.google.common.primitives.Ints;

Integer intValue = Ints.tryParse(stringValue);

如果无法解析字符串,它将返回null而不是抛出异常。

这个更短,但更短并不一定更好(并且它不会捕获超出范围的整数值,如danatel的评论中指出的那样

input.matches("^-?\\d+<*>quot;);

就个人而言,由于实现是通过辅助方法进行的,并且正确性超过了长度,我只会使用类似于你所拥有的东西(减去基本 Exception 类而不是 NumberFormatException) )。

您可以使用字符串类的matches方法。 [0-9]表示它可以是的所有值,+表示它必须至少有一个字符长,而*表示它可以是零个或多个字符长。

boolean isNumeric = yourString.matches("[0-9]+"); // 1 or more characters long, numbers only
boolean isNumeric = yourString.matches("[0-9]*"); // 0 or more characters long, numbers only

这是Jonas Klemming的Java 8变体回答:

public static boolean isInteger(String str) {
    return str != null && str.length() > 0 &&
         IntStream.range(0, str.length()).allMatch(i -> i == 0 && (str.charAt(i) == '-' || str.charAt(i) == '+')
                  || Character.isDigit(str.charAt(i)));
}

测试代码:

public static void main(String[] args) throws NoSuchAlgorithmException, UnsupportedEncodingException {
    Arrays.asList("1231231", "-1232312312", "+12313123131", "qwqe123123211", "2", "0000000001111", "", "123-", "++123",
            "123-23", null, "+-123").forEach(s -> {
        System.out.printf("%15s %s%n", s, isInteger(s));
    });
}

测试代码的结果:

        1231231 true
    -1232312312 true
   +12313123131 true
  qwqe123123211 false
              2 true
  0000000001111 true
                false
           123- false
          ++123 false
         123-23 false
           null false
          +-123 false

如果您的String数组包含纯整数和字符串,则下面的代码应该有效。你只需要看第一个角色。 例如[&QUOT; 4英寸,&QUOT; 44&QUOT;,&QUOT; ABC&QUOT;,&QUOT; 77&QUOT;,&QUOT;键&QUOT;]

if (Character.isDigit(string.charAt(0))) {
    //Do something with int
}

您还可以使用扫描仪 class,并使用 hasNextInt() - 这也允许你测试其他类型,如花车等。

怎么样:

return Pattern.matches("-?\\d+", input);

您只需检查 NumberFormatException : -

 String value="123";
 try  
 {  
    int s=Integer.parseInt(any_int_val);
    // do something when integer values comes 
 }  
 catch(NumberFormatException nfe)  
 {  
          // do something when string values comes 
 }  

您可以尝试使用apache utils

NumberUtils.isNumber( myText)

请参阅此处的javadoc

您可能还需要考虑帐户中的用例:

如果大多数时候您希望数字有效,那么捕获异常只会在尝试转换无效数字时导致性能开销。而调用一些 isInteger()方法然后使用转换Integer.parseInt()总是导致有效数字的性能开销 - 字符串是解析两次,一次检查,一次转换。

这是 Jonas '代码的修改,用于检查字符串是否在范围内以转换为整数。

public static boolean isInteger(String str) {
    if (str == null) {
        return false;
    }
    int length = str.length();
    int i = 0;

    // set the length and value for highest positive int or lowest negative int
    int maxlength = 10;
    String maxnum = String.valueOf(Integer.MAX_VALUE);
    if (str.charAt(0) == '-') { 
        maxlength = 11;
        i = 1;
        maxnum = String.valueOf(Integer.MIN_VALUE);
    }  

    // verify digit length does not exceed int range
    if (length > maxlength) { 
        return false; 
    }

    // verify that all characters are numbers
    if (maxlength == 11 && length == 1) {
        return false;
    }
    for (int num = i; num < length; num++) {
        char c = str.charAt(num);
        if (c < '0' || c > '9') {
            return false;
        }
    }

    // verify that number value is within int range
    if (length == maxlength) {
        for (; i < length; i++) {
            if (str.charAt(i) < maxnum.charAt(i)) {
                return true;
            }
            else if (str.charAt(i) > maxnum.charAt(i)) {
                return false;
            }
        }
    }
    return true;
}

如果您使用的是Android API,则可以使用:

TextUtils.isDigitsOnly(str);

另一种选择:

private boolean isNumber(String s) {
    boolean isNumber = true;
    for (char c : s.toCharArray()) {
        isNumber = isNumber && Character.isDigit(c);
    }
    return isNumber;
}

如果要检查字符串是否表示适合int类型的整数,我对jonas的答案做了一点修改,以便表示整数大于Integer.MAX_VALUE或小于Integer.MIN_VALUE的字符串,现在将返回false。例如:“3147483647”将返回false,因为3147483647大于2147483647,同样,“-2147483649”也将返回false,因为-2147483649小于-2147483648。

public static boolean isInt(String s) {
  if(s == null) {
    return false;
  }
  s = s.trim(); //Don't get tricked by whitespaces.
  int len = s.length();
  if(len == 0) {
    return false;
  }
  //The bottom limit of an int is -2147483648 which is 11 chars long.
  //[note that the upper limit (2147483647) is only 10 chars long]
  //Thus any string with more than 11 chars, even if represents a valid integer, 
  //it won't fit in an int.
  if(len > 11) {
    return false;
  }
  char c = s.charAt(0);
  int i = 0;
  //I don't mind the plus sign, so "+13" will return true.
  if(c == '-' || c == '+') {
    //A single "+" or "-" is not a valid integer.
    if(len == 1) {
      return false;
    }
    i = 1;
  }
  //Check if all chars are digits
  for(; i < len; i++) {
    c = s.charAt(i);
    if(c < '0' || c > '9') {
      return false;
    }
  }
  //If we reached this point then we know for sure that the string has at
  //most 11 chars and that they're all digits (the first one might be a '+'
  // or '-' thought).
  //Now we just need to check, for 10 and 11 chars long strings, if the numbers
  //represented by the them don't surpass the limits.
  c = s.charAt(0);
  char l;
  String limit;
  if(len == 10 && c != '-' && c != '+') {
    limit = "2147483647";
    //Now we are going to compare each char of the string with the char in
    //the limit string that has the same index, so if the string is "ABC" and
    //the limit string is "DEF" then we are gonna compare A to D, B to E and so on.
    //c is the current string's char and l is the corresponding limit's char
    //Note that the loop only continues if c == l. Now imagine that our string
    //is "2150000000", 2 == 2 (next), 1 == 1 (next), 5 > 4 as you can see,
    //because 5 > 4 we can guarantee that the string will represent a bigger integer.
    //Similarly, if our string was "2139999999", when we find out that 3 < 4,
    //we can also guarantee that the integer represented will fit in an int.
    for(i = 0; i < len; i++) {
      c = s.charAt(i);
      l = limit.charAt(i);
      if(c > l) {
        return false;
      }
      if(c < l) {
        return true;
      }
    }
  }
  c = s.charAt(0);
  if(len == 11) {
    //If the first char is neither '+' nor '-' then 11 digits represent a 
    //bigger integer than 2147483647 (10 digits).
    if(c != '+' && c != '-') {
      return false;
    }
    limit = (c == '-') ? "-2147483648" : "+2147483647";
    //Here we're applying the same logic that we applied in the previous case
    //ignoring the first char.
    for(i = 1; i < len; i++) {
      c = s.charAt(i);
      l = limit.charAt(i);
      if(c > l) {
        return false;
      }
      if(c < l) {
        return true;
      }
    }
  }
  //The string passed all tests, so it must represent a number that fits
  //in an int...
  return true;
}
is_number = true;
try {
  Integer.parseInt(mystr)
} catch (NumberFormatException  e) {
  is_number = false;
}

你做了什么,但你可能不应该总是那样检查。投掷例外应保留用于“例外”情况(可能适用于您的情况),并且在性能方面非常昂贵。

Number number;
try {
    number = NumberFormat.getInstance().parse("123");
} catch (ParseException e) {
    //not a number - do recovery.
    e.printStackTrace();
}
//use number

这只适用于正整数。

public static boolean isInt(String str) {
    if (str != null && str.length() != 0) {
        for (int i = 0; i < str.length(); i++) {
            if (!Character.isDigit(str.charAt(i))) return false;
        }
    }
    return true;        
}

这对我有用。只需识别String是基元还是数字。

private boolean isPrimitive(String value){
        boolean status=true;
        if(value.length()<1)
            return false;
        for(int i = 0;i<value.length();i++){
            char c=value.charAt(i);
            if(Character.isDigit(c) || c=='.'){

            }else{
                status=false;
                break;
            }
        }
        return status;
    }

要检查所有int字符,您只需使用双重否定字符。

if(!searchString.matches(&quot; [^ 0-9] + $&quot;))...

[^ 0-9] + $检查是否有任何非整数字符,因此如果测试失败则测试失败。只是不那样,你就成功了。

发现这可能有帮助:

public static boolean isInteger(String self) {
    try {
        Integer.valueOf(self.trim());
        return true;
    } catch (NumberFormatException nfe) {
        return false;
    }
}

我相信进入异常的风险为零,因为正如您在下面所见,您总是可以安全地将 int 解析为 String ,而不是相反。

所以:

  1. 如果字符串中的每个字符插槽至少匹配,则检查 字符 {&quot; 0“,”1“,”2“,”3“,”4“,”5“,”6“,”7“,” 8英寸,&QUOT; 9&QUOT;} 即可。

    if(aString.substring(j, j+1).equals(String.valueOf(i)))
    
  2. 总和您在上述广告位中遇到的所有时间 字符。

    digits++;
    
  3. 最后你检查你遇到的时间是否为整数 字符等于给定字符串的长度。

    if(digits == aString.length())
    
  4. 在实践中我们有:

        String aString = "1234224245";
        int digits = 0;//count how many digits you encountered
        for(int j=0;j<aString.length();j++){
            for(int i=0;i<=9;i++){
                if(aString.substring(j, j+1).equals(String.valueOf(i)))
                        digits++;
            }
        }
        if(digits == aString.length()){
            System.out.println("It's an integer!!");
            }
        else{
            System.out.println("It's not an integer!!");
        }
    
        String anotherString = "1234f22a4245";
        int anotherDigits = 0;//count how many digits you encountered
        for(int j=0;j<anotherString.length();j++){
            for(int i=0;i<=9;i++){
                if(anotherString.substring(j, j+1).equals(String.valueOf(i)))
                        anotherDigits++;
            }
        }
        if(anotherDigits == anotherString.length()){
            System.out.println("It's an integer!!");
            }
        else{
            System.out.println("It's not an integer!!");
        }
    

    结果是:

      

    这是一个整数!!

         

    这不是一个整数!!

    同样,您可以验证 String float 还是 double ,但在这种情况下,您只能遇到字符串中的一个。(点)当然要检查 digits ==(aString.length() - 1)

      

    同样,这里遇到解决异常的风险为零,但是如果你计划解析一个已知包含数字的字符串(假设 int 数据类型),你必须先检查一下它适合数据类型。否则你必须施展它。

    我希望我帮助

许可以下: CC-BY-SA归因
不隶属于 StackOverflow
scroll top