Java中两个long类型进行相加运算,值溢出的解决方法

欢迎大家关注我的公众号【老周聊架构】,Java后端主流技术栈的原理、源码分析、架构以及各种互联网高并发、高性能、高可用的解决方案。

博主在项目中遇到了这个需求,开始用的int类型进行运算,字段是存的字节,然后后台进行相应的逻辑运算返回给前台,前台来转化成GB。大家都知道1GB=1024MB,1MB=1024KB,1KB=1024Byte,所以1GB=1024*1024*1024=1073741824Byte。

所以问题来了,1GB的数据我后台就要存1073741824Byte,然后两个int类型进行相加直接溢出。


我们先来回顾下java的8种基本数据类型,Java语言有8种基本数据类型, 分别用于存储整数、浮点数、字符数据和布尔类型数据。

图-1
从上图中可以看出, 基本数据类型主要分为4大类 (整数类型、浮点类型、char、boolean), 整数类型又分为了4小类(byte、short、int、long), 浮点类型也分了2小类(float、double),这些数据类型的区别是怎样的?通过下图展示了这8种数据类型的存储空间及使用场景:

类型名称字节空间使用场景
byte1字节(8位)存储字节数据(较常用)
short2字节(16位)兼容性考虑(很少使用)
int4字节(32位)存储普通整数(常用)
long8字节(64位)存储长整数(常用)
float4字节(32位)存储浮点数(不常用)
double8字节(64位)存储双精度浮点数(常用)
char2字节(16位)存储一个字符(常用)
boolean1字节(8位)存储逻辑变量(true、false)(常用)

ok,回顾完了,我们来看下int类型的数据范围-2的31次方 ~ 2的31次方-1(这个符号不好打出来,反正你懂得哈哈),即-2147483648 ~2147483647,大约正负21个亿多些。

我1GB都十几亿了,也就是说,我int类型只能运算2GB的数据,对于存储文件信息显然是不行的。

然后我想到了long型,long类型的数据范围-2的63次方 ~ 2的63次方-1,即 -9223372036854775808 ~ 9223372036854775807,也就是比int类型多了大约10的10次方,也就是2*10^10GB,可以满足需求,所以我就开干了。。。


我们来看下demo:

package com.test;

import java.math.BigDecimal;

/**
 * @author riemann
 * @date 2019/08/21 20:47
 */
public class TwoLongAddTest {

    public static void main(String[] args) {
        Long a = 2147483648L;
        Long b = 2147483648L;
        BigDecimal add = new BigDecimal(a).add(new BigDecimal(b));
        System.out.println(add.longValue());
    }

}

输出结果:

4294967296

这里两个long类型是不能直接进行相加的运算的,所以这里用BigDecimal 封装下进行相加运算。

直接输出add也可以得到求和的数据,但博主项目中返回给前台是long类型,所以进行了下类型转换add.longValue()


上面我们说了 int 类型的数超过了 -2147483648 ~2147483647这个范围的话,可以用 long 类型来接收。

那如果该数超过了 long 类型呢?

我们上面也说了 long 类型的数据范围是 -9223372036854775808 ~ 9223372036854775807,那比如我的数比正范围的最大数还要多 1 的话即 9223372036854775808

我们来看下 IDEA,会提示该数太大了,超过了 long 类型的范围。

在这里插入图片描述
那你可能会说,那我要这种超过了 long 型的数计算咋办?

不着急,请看下面的代码。

/**
 * @Author: 微信公众号【老周聊架构】
 */
public class BigNumAdd {
    public static void main(String[] args) {
        System.out.println("Long.MAX_VALUE = " + Long.MAX_VALUE);

        String strNum1 = "9223372036854775807";
        String strNum2 = "9223372036854775808";

        bigNumAdd(strNum1, strNum2);
    }

    private static void bigNumAdd(String strNum1, String strNum2) {
        int len1 = strNum1.length();
        int len2 = strNum2.length();

        int maxLen = Integer.max(len1, len2);
        StringBuilder sb = new StringBuilder();

        // 进位
        int carry = 0;

        for (int i = 0; i < maxLen; i++) {
            int temp = carry;
            carry = 0;

            if (i < len1) {
                temp += Integer.parseInt(String.valueOf(strNum1.charAt(len1 - 1 - i)));
            }
            if (i < len2) {
                temp += Integer.parseInt(String.valueOf(strNum2.charAt(len2 - 1 - i)));
            }
            if (temp >= 10) {
                temp -= 10;
                carry = 1;
            }

            sb.append(temp);
        }

        if (carry > 0) {
            sb.append(carry);
        }

        System.out.println(String.format("%s + %s = %s", strNum1, strNum2, sb.reverse().toString()));
    }
}

控制台输出:

在这里插入图片描述
long 型最大值加与 long 型最大值还大 1 的值之和,成功计算出正确的答案。

好了,希望对你有帮助。

### 回答1: Javalong类型可以和int类型进行计算。long类型可以存储更大范围的整数,占据8个字节(64位),而int类型只能存储较小范围的整数,占据4个字节(32位)。当我们对long类型和int类型进行计算时,Java会自动将int类型转换为long类型,然后进行计算。 例如,我们可以将一个int类型的变量和一个long类型的变量相加,结果将会是一个long类型。代码示例如下: ```java int a = 10; long b = 20; long result = a + b; // 将int类型a转换为long类型,然后与b相加 System.out.println(result); // 输出结果为30 ``` 同样,我们也可以进行其他的数学运算,如减法、乘法和除法。Java会根据需要自动进行类型转换,确保计算结果的准确性。 需要注意的是,当long类型超出int类型的范围时,可能会导致溢出错误。因此,在进行计算时,我们需要确保结果可以被int类型接受,或者使用更为安全的long类型存储计算结果。 综上所述,long类型和int类型可以进行计算,Java会自动进行类型转换以确保计算的准确性。 ### 回答2: Java的数据类型有很多种,其包括整数类型int和浮点数类型float。而"javalong"并不是Java的内置数据类型,我理解为这是指Java的长整型数据类型long。 长整型数据类型long用于存储较大范围的整数,它占据8个字节的存储空间(64位),可以表示范围在-9223372036854775808到9223372036854775807之间的整数。 在Javalong类型和int类型之间可以进行计算。当将long类型和int类型进行计算时,int类型会被自动转换为long类型,然后进行计算。由于long类型的范围比int类型的范围大,所以计算结果可以正常得到。 例如,考虑以下的Java代码: ```java long a = 10000000000L; // 定义一个长整型变量a int b = 100; // 定义一个整型变量b long c = a + b; // 将a和b相加,并将结果赋给长整型变量c System.out.println(c); // 输出c的 ``` 在这个例子,我们定义了一个long类型的变量a,并赋为10000000000L。我们还定义了一个int类型的变量b,并赋为100。然后,我们将a和b相加,并将结果赋给long类型的变量c。最后,我们使用System.out.println()方法打印了变量c的。 运行以上代码,我们会得到输出结果为 "10000000100",即长整型和整型的计算结果。 所以,长整型数据类型long和整型数据类型int之间可以进行计算,并且计算结果可以正常得到。 ### 回答3: JavaJavalong类型可以和int类型进行计算。 Javalong是一种表示整数的数据类型,在Java占用8个字节。而int类型是另一种表示整数的数据类型,在Java占用4个字节。 在Java,当我们用Javalong类型和int类型进行计算时,会根据需要自动进行类型转换。具体来说,如果我们将一个int类型Javalong类型进行运算Java会将int类型自动转换为Javalong类型,然后再进行计算。这是因为Javalong的数据范围更大,可以容纳更大的整数。 例如,我们可以将一个Javalong类型的变量与一个int类型的变量相加Java会自动将int类型转换为Javalong类型,然后进行计算。结果将会是一个Javalong类型。 但需要注意的是,当我们将一个Javalong类型赋给一个int类型的变量时,由于int类型的数据范围较小,可能会导致数据溢出的情况。因此,在进行这类计算时,我们需要谨慎处理,确保不会发生数据溢出的情况。 综上所述,Javalong类型可以和int类型进行计算。在这种计算过程Java会自动进行类型转换,但需要注意数据范围的差异,避免出现数据溢出的情况。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

老周聊架构

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值