Maison  >  Article  >  Java  >  Quelle est la différence entre l'efficacité de la conversion d'un int en chaîne et la conversion d'une chaîne en int en Java ?

Quelle est la différence entre l'efficacité de la conversion d'un int en chaîne et la conversion d'une chaîne en int en Java ?

王林
王林avant
2023-05-12 08:46:161045parcourir

    Comparaison de l'efficacité entre int à string et string à int

    string à int , Deux méthodes

    Interger.parseInt(String)
    Interger.valueOf(String).intValue()

    La deuxième méthode peut consister à voir le code source et à implémenter la première méthode.

    Quelle est la différence entre lefficacité de la conversion dun int en chaîne et la conversion dune chaîne en int en Java ?

    Le commentaire signifie probablement ceci

    /**
          *返回一个包含整数的对象
          *指定的{@ String String}的值。 这个说法是
          *被解释为表示一个有符号的十进制整数
          *就好像这个论据是给予{@link的
          * #parseInt(java.lang.String)}方法。 结果是一个
          表示整数值的整数对象
          *由字符串指定。
         *
          换句话说,这个方法返回一个{@code Integer}
          *对象等于以下值:
         *
          * <blockquote>
          * {@code new Integer(Integer.parseInt(s))}
          * </ blockquote>
         *
          * @param是要解析的字符串。
          * @返回一个保存值的{整数}对象
          *由字符串参数表示。
          * @exception NumberFormatException如果字符串不能被解析
          *作为一个整数。
         */

    La méthode parseInt() est implémentée dans valueOf(). La deuxième fois est beaucoup plus rapide que la première.

     Integer.parseInt(str) : 21
     Integer.valueOf(str).intValue() : 14

    int Il existe généralement trois méthodes pour convertir des chaînes

    • La première : nombre + ""

    • # 🎜🎜#
    • Le deuxième type : string.valueOf()

    • Le troisième type : .toString()

      # 🎜 🎜#
    • Parlons d'abord du premier type, qui est simple et grossier.
    • La deuxième méthode : la couche inférieure utilise toujours la méthode .toString()
    • La troisième méthode C'est le code sur toString()
    • .
    int num = 888888;
     
            //(1)num + ""
            long start = System.currentTimeMillis();//得到开始运行时系统时间
            for(int i=0; i<100000; i++){
                String str = num + "";
            }
            long end = System.currentTimeMillis();//得到结束运行时系统时间
            System.out.println("num + \"\" : " + (end - start));
     
            //(2)String.valueOf(num)
            start = System.currentTimeMillis();
            for(int i=0; i<100000; i++){
                String str = String.valueOf(num);
            }
            end = System.currentTimeMillis();
            System.out.println("String.valueOf(num) : " + (end - start));
     
            //(3)Integer.toString(num)
            start = System.currentTimeMillis();
            for(int i=0; i<100000; i++){
                String str = Integer.toString(num);
            }
            end = System.currentTimeMillis();
            System.out.println("Integer.toString(num) : " + (end - start));

    Le résultat est

    num + "" : 82
    String.valueOf(num) : 32

    Integer.toString( num ): 9

    Après de nombreux tests répétés, toString() est le plus rapide, num+"" est le plus lent et le code source est comme ceci lors de l'utilisation de String.valueOf() .
    public static String valueOf(Object obj) {
        return (obj == null) ? "null" : obj.toString();
    }

    C'est-à-dire que lors de son utilisation, il n'est pas nécessaire de juger si l'objet passé est nul, mais faites particulièrement attention au fait que si l'objet passé est vide, une chaîne nulle sera renvoyée au lieu d'une valeur nulle, c'est quelque chose à garder à l'esprit.

    Problème d'analyse de chaîne en int

    Je crois que de nombreux étudiants ont rencontré un tel problème lors des entretiens, demandant d'encapsuler une fonction pour convertir le type String en type int. Ce problème apparemment simple cache en réalité de nombreux détails, et il n’est pas facile de réellement résumer cette fonction. Ce que l'intervieweur veut examiner, ce n'est pas vraiment la difficulté de l'algorithme lui-même. L'algorithme de cette question n'est en fait pas difficile du tout. Ce que l'intervieweur veut examiner, c'est avec quel soin le programmeur écrit le code et si le problème est complet. en d’autres termes, nous devons faire de notre mieux pour rendre le code robuste. Ci-dessous, nous analysons étape par étape les détails cachés dans ce problème.

    Analysis Wave

    Tout d'abord, nous ne considérons aucune gestion d'exception. En supposant que les données transmises par l'appelant de la fonction sont correctes, il est facile d'écrire. le code suivant :

        public int strToInt(String str) {
            int number = 0;
            for (int i=0; i<str.length(); i++) {
                number *= 10;
                number += (str.charAt(i) - &#39;0&#39;);
            }
            return number;
        }

    Le code ci-dessus traversera chaque caractère de la chaîne, le convertira en l'entier correspondant, puis l'intégrera un par un dans le numéro de données entier.

    Si vous soumettez un tel code à l'intervieweur, le résultat ne sera certainement pas satisfaisant. Parce que vous n'avez pas pris en compte la robustesse du programme. Les fonctions que nous encapsulons sont équivalentes aux interfaces API et sont fournies à tous les développeurs. Il est inévitable que d'autres développeurs ne transmettent pas certains paramètres étranges, et ce code n'a aucune réponse aux paramètres anormaux. . Si un traitement est effectué, une fois le paramètre d'exception transmis, le programme plantera directement. Ensuite, nous améliorerons cette fonction étape par étape et améliorerons sa robustesse.

    1. Car la chaîne entrante est un objet vide ou la chaîne est une chaîne vide

        public int strToInt(String str) throws NumberFormatException{
            if (str == null || str.contentEquals("")) { // 如果传入的字符串为空对象或者传入的字符串为空字符串,则抛出异常
                throw new NumberFormatException("null or empty string"); // 这里直接利用java封装好的异常类,当然我们也可以自己封装异常类,面试官要考察的不是对异常类的封装,而是你要知道要处理异常情况
            }
            int number = 0;
            for (int i=0; i<str.length(); i++) {
                number *= 10;
                number += (str.charAt(i) - &#39;0&#39;);
            }
            return number;
        }
    Tout d'abord, si la chaîne est vide ou si c'est une chaîne vide , si c'est le cas, lancez une exception directement. Ici, nous utilisons la classe d'exception encapsulée en Java NumberFormatException. Bien sûr, nous pouvons également encapsuler la classe d'exception nous-mêmes. Ce que l'intervieweur veut examiner, ce n'est pas l'encapsulation de la classe d'exception, mais celle que vous souhaitez. besoin de savoir quoi faire.

    2. Traitement des bits de signe

    Nous ferions mieux de demander à l'avance à l'intervieweur s'il est possible de transmettre un nombre négatif. Lorsqu'il s'agit d'un nombre positif, c'est. est-ce autorisé ? Bit de signe, si c'est le cas, nous devons traiter le bit de signe. Le premier caractère d'un nombre négatif est "-". Il suffit de déterminer si le premier caractère est "-" pour savoir si le nombre entrant est un. nombre négatif. , si le nombre positif peut avoir un bit de signe, le premier caractère peut être "+", et nous devrons également le traiter en conséquence :
        public int strToInt(String str) throws NumberFormatException{
            if (str == null || str.contentEquals("")) { // 如果传入的字符串为空对象或者传入的字符串为空字符串,则抛出异常
                throw new NumberFormatException("null or empty string"); // 这里直接利用java封装好的异常类,当然我们也可以自己封装异常类,面试官要考察的不是对异常类的封装,而是你要知道要处理异常情况
            }
            boolean negative = false; // negative为true表示是负数,反之为正数
            int pos = 0;
            if (str.charAt(0) == &#39;-&#39;) { // 如果为负数
                negative = true;
                pos++; // 调过第一位符号位
            } else if (str.charAt(0) == &#39;+&#39;) {
                pos++; // 调过第一位符号位
            }
            int number = 0;
            while (pos < str.length()) {
                number *= 10;
                number += (str.charAt(pos) - &#39;0&#39;);
                pos++;
            } 
            return negative ? -number : number; // 如果为负数则返回对应的负数
        }

    3. mauvais caractères

    L'appelant de la fonction peut transmettre une chaîne désordonnée, telle que "abc23123". Dans ce cas, nous devons le gérer en conséquence. Nous devons lancer une exception à l'appelant pour l'informer. lui des caractères transmis. La chaîne est une chaîne illégale :
        public int strToInt(String str) throws NumberFormatException{
            if (str == null || str.contentEquals("")) { // 如果传入的字符串为空对象或者传入的字符串为空字符串,则抛出异常
                throw new NumberFormatException("null or empty string"); // 这里直接利用java封装好的异常类,当然我们也可以自己封装异常类,面试官要考察的不是对异常类的封装,而是你要知道要处理异常情况
            }
            boolean negative = false; // negative为true表示是负数,反之为正数
            int pos = 0;
            if (str.charAt(0) == &#39;-&#39;) { // 如果为负数
                negative = true;
                pos++; // 调过第一位符号位
            } else if (str.charAt(0) == &#39;+&#39;) {
                pos++; // 调过第一位符号位
            }
            int number = 0;
            while (pos < str.length()) {
                if (str.charAt(pos) >= &#39;0&#39; && str.charAt(pos) <= &#39;9&#39;) { // 只有字符在&#39;0&#39;到&#39;9&#39;的范围内,才算正确的字符
                    number *= 10;
                    number += (str.charAt(pos) - &#39;0&#39;);
                    pos++; 
                } else {
                    throw new NumberFormatException("invalid string"); // 当字符是其他字符时,抛出异常告知调用者传入的字符串错误
                }
            } 
            return negative ? -number : number; // 如果为负数则返回对应的负数
        }

    4 Traitement des données entières hors plage

    La chaîne transmise par l'appelant peut être un. chaîne très longue, convertie en L'entier peut dépasser la plage de stockage de l'entier, tel que "12345678674324334". Dans ce cas, nous devons lever une exception pour informer l'appelant que la chaîne entrante dépasse la plage de l'entier : #🎜. 🎜#
        public int strToInt(String str) throws NumberFormatException{
            if (str == null || str.contentEquals("")) { // 如果传入的字符串为空对象或者传入的字符串为空字符串,则抛出异常
                throw new NumberFormatException("null or empty string"); // 这里直接利用java封装好的异常类,当然我们也可以自己封装异常类,面试官要考察的不是对异常类的封装,而是你要知道要处理异常情况
            }
            boolean negative = false; // negative为true表示是负数,反之为正数
            int pos = 0;
            if (str.charAt(0) == &#39;-&#39;) { // 如果为负数
                negative = true;
                pos++; // 调过第一位符号位
            } else if (str.charAt(0) == &#39;+&#39;) {
                pos++; // 调过第一位符号位
            }
            int limit = negative ? (-Integer.MIN_VALUE) : Integer.MAX_VALUE;
            int mult = limit / 10; // 记录最大数/10,让number和这个数比较,如果大于它,则number * 10肯定也就大于最大数
            int number = 0;
            while (pos < str.length()) {
                if (str.charAt(pos) >= &#39;0&#39; && str.charAt(pos) <= &#39;9&#39;) { // 只有字符在&#39;0&#39;到&#39;9&#39;的范围内,才算正确的字符
                    if (number > mult) {// 让number和mult比较,如果大于它,则number * 10肯定也就大于最大数
                        throw new NumberFormatException("input string beyond int size");
                    }
                    number *= 10;
                    int digit = str.charAt(pos) - &#39;0&#39;;
                    if (number > limit - digit) { // 这里不能用number + digit > limit来判断,因为number + digit可能超出整数的存储范围,相加后的数可能是一个负数,但是limit - digit肯定不会超出
                        throw new NumberFormatException("input string beyond int size");
                    } else {
                        number += digit;
                    }
                    pos++;
                } else {
                    throw new NumberFormatException("invalid string"); // 当字符是其他字符时,抛出异常告知调用者传入的字符串错误
                }
            } 
            return negative ? -number : number; // 如果为负数则返回对应的负数
        }
    Above Dans le code, lorsque nous jugeons si un nombre dépassera l'entier maximum, nous comparons d'abord sa valeur (entier maximum/10) au lieu de la multiplier par 10 et de la comparer avec l'entier maximum. car si le nombre * 10 dépasse la plage entière, cela provoquera un débordement de données, et la valeur obtenue peut être un nombre négatif, mais la valeur de (entier maximum/10) ne provoquera pas de débordement de données. C'est aussi un petit détail. Peut-être pensez-vous que cette fonction est parfaite, mais maintenant je veux vous dire que l'écriture ci-dessus est fausse.

    为什么呢?这要从整数的范围说起,整数的取值范围是(-2^31)至(2^31 - 1),从绝对值的角度看,最小负数相比于最大正数大1。所以上面代码中(-Integer.MIN_VALUE)会超出整形的范围,造成数据溢出,也就是说上面的代码对负数最小范围的限制的处理是错误的。那么怎么解决这个问题呢?

    我们换个角度思考,最小负数的绝对值比最大正数的绝对值大1,那(-Integer.MAX_VALUE)的值肯定不会超出整数的范围,我们现在的程序是以正数的方式处理,如果反过来已负数的方式处理,问题不就解决了吗?修改代码如下:

        public int strToInt(String str) throws NumberFormatException{
            if (str == null || str.contentEquals("")) { // 如果传入的字符串为空对象或者传入的字符串为空字符串,则抛出异常
                throw new NumberFormatException("null or empty string"); // 这里直接利用java封装好的异常类,当然我们也可以自己封装异常类,面试官要考察的不是对异常类的封装,而是你要知道要处理异常情况
            }
            boolean negative = false; // negative为true表示是负数,反之为正数
            int pos = 0;
            if (str.charAt(0) == &#39;-&#39;) { // 如果为负数
                negative = true;
                pos++; // 调过第一位符号位
            } else if (str.charAt(0) == &#39;+&#39;) {
                pos++; // 调过第一位符号位
            }
            int limit = negative ? Integer.MIN_VALUE : (-Integer.MAX_VALUE);
            int mult = limit / 10;
            int number = 0;
            while (pos < str.length()) {
                if (str.charAt(pos) >= &#39;0&#39; && str.charAt(pos) <= &#39;9&#39;) { // 只有字符在&#39;0&#39;到&#39;9&#39;的范围内,才算正确的字符
                    if (number < mult) {
                        throw new NumberFormatException("input string beyond int size");
                    }
                    number *= 10;
                    int digit = str.charAt(pos) - &#39;0&#39;;
                    if (number < limit + digit) {
                        throw new NumberFormatException("input string beyond int size");
                    } else {
                        number -= digit;
                    }
                    pos++;
                } else {
                    throw new NumberFormatException("invalid string"); // 当字符是其他字符时,抛出异常告知调用者传入的字符串错误
                }
            } 
            return negative ? number : -number;
        }

    OK,现在我们把能够想到的异常情况处理了。再来考虑一个问题,为什么整形数据的范围是(-2^31)至(2^31 - 1),最小负数的绝对值比最大正数的绝对值要大1呢?

    5、int数据范围的讨论

    我们知道,一个int类型占四个字节,也就是32位,其中第一位是符号位,符号位为0表示正数,为1表示负数,其余31位表示数值。正常来说int类型的数据范围应该是(-2^31-1)到(2^31-1),为什么负数会多一位呢?

    我们首先看一下Java代码中对Integer.MAX_VALUE和Integer.MIN_VALUE的定义:

        /**
         * A constant holding the minimum value an {@code int} can
         * have, -2<sup>31</sup>.
         */
        public static final int   MIN_VALUE = 0x80000000;
     
        /**
         * A constant holding the maximum value an {@code int} can
         * have, 2<sup>31</sup>-1.
         */
        public static final int   MAX_VALUE = 0x7fffffff;
    原码、反码、补码

    我们知道,在计算机中,数据都是以二进制的形式存储的,比如,数字10,其二进制形式就是1010。

    一个字节有8位,每位可以存储一个01字符,byte类型占1个字节,也就是8位,其中,最高位是符号位,用来表示数值是正数还是负数,符号位为0表示正数,符号位为1表示负数。我们先来看一下原码、反码、补码的定义:

    • 原码:符号位加上真值的绝对值, 即用第一位表示符号, 其余位表示值。

    • 反码:正数的反码是其本身;负数的反码是在其原码的基础上, 符号位不变,其余各个位取反。

    • 补码:补码的表示方法是:正数的补码就是其本身;负数的补码是在其原码的基础上, 符号位不变, 其余各位取反, 最后+1。 (即在反码的基础上+1)

    正数的原码、反码、补码都是其本身;负数的反码是在其原码的基础上,符号位不变,其余个位取反,负数的补码是其反码的基础上+1。

    举例说明(下面都以byte类型进行举例):

    数据 原码 反码 补码
    10 00001010 00001010 00001010
    -10 10001010 11110101 11110110

    计算机中,数据都是以补码的形式存储的。为什么要以补码的形式存储呢?有两个原因:

    1、如果数值以补码的形式保存,对一个数进行求补运算,可以得到其相反值

    求补运算:将一个数(包括正数和负数)所有二进制位(包括符号位和数值位)取反,然后在最低位加上1。

    为什么对一个数进行求补运算,可以得到其相反值呢?我们先来分析一下求补运算的定义,现将所有的二进制取反,然后+1,首先一个数和它所有位取反得到的数相加,其结果肯定是11111111,这是因为它们每一位都不一样,然后将结果+1,即11111111 + 1,结果是1 00000000,最高位的1已经溢出,换种方式说,如果以f(n)表示对n进行求补运算,那么对于任意的范围内的数,可以得到:

    n + f(n) = 1 00000000

    f(n) = 1 00000000 - n

    而对于一个正数来说,对其进行求补运算其实得到的就是它的相反数的补码(负数的补码符号位保持不变,其他为全部取反再+1,因为正数和负数的符号位本来就不一样,所以对一个正数进行求补其实得到的就是它的相反数的补码)。

    那么对于一个负数来说呢?对其进行求补运算是否能够得到其对应的正数的补码呢?

    假设n>0,根据上面可知:

    f(n) = 1 00000000 - n

    对f(n)进行求补运算,有:

    f(f(n)) = f(1 00000000 - n) = 1 00000000 - (1 00000000 - n) = n

    其中,1 00000000 - n表示n对应负数的补码,对其进行求补运算得到的就是n,正数的补码就是其原码。

    由上可知:如果数值以补码的形式保存,对一个数进行求补运算,可以得到其相反值,即:f(n) = -n

    2、方便减法运算

    如果数值以补码的方式存储,可以将减法变为加法,省去了减法器,通过上面的推导,如果数据以补码的方式存储,以f(n)表示对n进行求补运算,可以得到:

    f(n) = -n

    那么现在我们需要计算m - n,应该要怎么计算呢?如果以补码的方式存储,那么就有:

    m - n = m + (-n) = m + f(n)

    也就是说,减去一个数只需要加上对其进行求补运算后得到的值即可。

    3、使用补码存储数据,可以对任意的两个数直接进行相加运算,不用考虑符号位

    4、通过补码形式存储,规定10000000对应的负数的最小值,也就是-128。

    由上面可知,如果是byte类型,数据范围应该为[-128,127],最小负数要比最大正数多一位。

    Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

    Déclaration:
    Cet article est reproduit dans:. en cas de violation, veuillez contacter admin@php.cn Supprimer