Java中对于数据类型,可以分别两大数据类型:基本数据类型和引用数据类型。

基本数据类型组成

Java中内置了8种基本数据类型,其中包括6种数字类型(4个整数型,2个浮点型)、1种字符型、1种布尔型。再说数据类型之前,先说一下计算中最小的信息存储单位:bit,二进制数的一位包含的信息或2个选项中特别指定1个的需要信息量称为一比特,是表示信息的最小单位,只有两种状态:0和1。

byte型:

  • byte 数据类型是8位(bit)、有符号的,以二进制补码表示的整数;
  • 最小值是 -128(-2^7);
  • 最大值是 127(2^7-1);
  • 默认值是 0;
  • byte 类型用在大型数组中节约空间,主要代替整数,因为 byte 变量占用的空间只有 int 类型的四分之一;
  • 例子:byte a = 100,byte b = -50。

short:

  • short 数据类型是 16 位、有符号的以二进制补码表示的整数
  • 最小值是 -32768(-2^15);
  • 最大值是 32767(2^15 - 1);
  • Short 数据类型也可以像 byte 那样节省空间。一个short变量是int型变量所占空间的二分之一;
  • 默认值是 0;
  • 例子:short s = 1000,short r = -20000。

int:

  • int 数据类型是32位、有符号的以二进制补码表示的整数;
  • 最小值是 -2,147,483,648(-2^31);
  • 最大值是 2,147,483,647(2^31 - 1);
  • java中整数类型的默认值。
  • 默认值是 0 ;
  • 例子:int a = 100000, int b = -200000。

long:

  • long 数据类型是 64 位、有符号的以二进制补码表示的整数;
  • 最小值是 -9,223,372,036,854,775,808(-2^63);
  • 最大值是 9,223,372,036,854,775,807(2^63 -1);
  • 这种类型主要使用在需要比较大整数的系统上;
  • 默认值是 0L;
  • 例子: long a = 100000L,Long b = -200000L。 "L"理论上不分大小写,但是若写成"l"容易与数字"1"混淆,不容易分辩。所以最好大写。

float:

  • float 数据类型是单精度、32位、符合IEEE 754标准的浮点数;
  • float 在储存大型浮点数组的时候可节省内存空间;
  • 默认值是 0.0f;
  • 浮点数不能用来表示精确的值,如货币;
  • 例子:float f1 = 234.5f。

double:

  • double 数据类型是双精度、64 位、符合IEEE 754标准的浮点数;
  • java中浮点数的默认类型为double类型;
  • double类型同样不能表示精确的值,如货币;
  • 默认值是 0.0d;
  • 例子:double d1 = 123.4。

boolean:

  • boolean数据类型表示一位的信息;
  • 只有两个取值:true 和 false;
  • 这种类型只作为一种标志来记录 true/false 情况;
  • 默认值是 false;
  • 例子:boolean flag= true。

char:

  • char类型是一个单一的 16 位 Unicode 字符;
  • 最小值是 \u0000(即为0);
  • 最大值是 \uffff(即为65,535);
  • char 数据类型可以储存任何字符;
  • 例子:char c= 'A';。

Java中的数据类型自动转换机制

上边我们说到了Java中的基本数据类型,说到底,这些基本数据类型还是为了进行数值的存储和 徐计算。接下来我们在说一下各个数据类型之间进行计算会发生什么。我们按照数据类型所占存储空间大小由小到大排序为:

byte   <  short   <   int(char)    <    long   <   float   <  double。

这里不包括boolean类型,虽然boolean占一个bit,但是一般只用来表示真假值,不存在进行与其它几个数据类型之间进行计算。这里我为什么把char也放进来呢,与boolean不同,char类型是可以运算的因为char在ASCII等字符编码表中,有对应的数值。所以,我们在用char进行计算时直接当做ASCII表对应的整数来对待。

1.隐式转换

从小到大,可以隐式转换,数据类型将自动提升。就是将占空间小的赋值给大的,不会报错,数据类型会自动转成声明的数据类型。如下代码,编译器没有报错。

@Test
    public void test(){
        byte a = 10;
        short b = a;
        int c = b;
        long d = c;
        float e = d;
        double f = e;
    }

简单来说我们可以这么理解,我把一杯1升容量杯子里的水倒入一个2升的杯子里,顺便倒,因为1升杯子装满了也比2升小,大概这个意思,所以将小数据类型赋值给大数据类型可以实现隐氏的自动转换。那么上边例子里,对于byte a = 10我有点疑问,java整数数据类型的默认值是int啊,那么数值10应该是int型的,我们说小到大实现隐式转换,这不是大到小吗,为什么没有编译报错呢。我们再来看下下边代码。

@Test
    public void test(){
        byte a = 10;
        byte b = 127;
        byte c = 128; // 编译报错
    }

这里涉及到一个叫字面值(literal)的问题,字面值就是表面上的值,例如整型字面值在源代码中就是诸如 5、0、 -200这样的。如果整型子面子后面加上L或者l,则这个字面值就是long类型,比如:1000L代表一个long类型的值。如果不加L或者l,则为int类型。基本类型当中的byte short int long都可以通过不加L的整型字面值(我们就称作int字面值吧)来创建,例如 byte b = 100; short s = 5;对于long类型,如果大小超出int所能表示的范围(32 bits),则必须使用L结尾来表示。整型字面值可以有不同的表示方式:16进制【0X or 0x】、10进制【nothing】、八进制【0】2进制【0B or 0b】等,二进制字面值是JDK 7以后才有的功能。在赋值操作中,int字面值可以赋给byte short int long,Java语言会自动处理好这个过程。如果方法调用时不一样,调用test(0)的时候,它能匹配的方法是test(int),当然不能匹配test(byte)方法,至于为什么Java没有像支持赋值操作那样支持方法调用,不得而知。注意区别包装器与原始类型的自动转换(anto-boxing,auto-unboxing)。byte d = 'A';也是合法的,字符字面值可以自动转换成16位的整数。

@Test
    public void test(){
        byte a = 10;
        byte b = 127;
        byte c = 128; // 编译报错
        byte d = 10L; // 编译报错
    }

2.显示转换

显示转换是相对与隐式转换的,显示转换则是从大到小,将大数据类型赋值给小数据类型时需要显示的进行强制类型转换,才可以编译通过。如下代码:

@Test
    public void test(){
        long a = 127L;
        int b = (int)a;
        short c = (short) b;
        byte d = (byte) c;
    }

当我们将一个2升的杯子的水向一个1升的杯子倒时可能就会有部分水溢出来,因为超过了1升杯子的最大容量,同理,我们将大数据类型强制转换成小数据类型时可能会存在精度损失。所以我们必须要进行这个操作的话,就得进行强制转换,这样编译器才认可。另外,这里有个小疑问:上边d打印出来是什么呢?答案肯定是127。那么我将a赋值为128L呢,d打印出来是什么呢?答案是-128;128为int整数32位,前24为全部为0 后8位是1000 0000,(byte)128 转型为byte后首位是1,java认为是负数的补码标识,补码转原码,符号位不变,数值位取反加1,第9位为1,后8为位1000 0000,仍然是-128。

3.计算时类型的自动转换

当基本数据类型进行+、-、<<等计算操作时,会自动进行类型的提升,其规则为:

  • 所有的byte型、short型、char型的值将被提升到int型;   
  • 如果一个操作数是long型,计算结果就是long型;
  • 如果一个操作数是float型,计算结果就是float型;
  • 如果一个操作数是double型,计算结果就是double型。

带着上边的规则,我们看下下边代码中注释着问号的代码会不会编译成功:

@Test
    public void test(){
        byte a = 10;
        a = a + 1; // ?
        a += a; // ?
        a += 1; // ?
        a =+ a; // ?
        a =+ 1; // ?

        int x = 10;
        long y = 10;
        int z = x + 10; // ?
        z = y + 10; // ?
        z = x + y; // ?
    }

心里想好答案,或记在小本本上,我开始公布答案了o(* ̄︶ ̄*)o:

@Test
    public void test(){
        byte a = 10;
        a = a + 1; // 失败
        a += a; // 成功
        a += 1; // 成功
        a =+ a; // 失败
        a =+ 1; // 成功

        int x = 10;
        long y = 10;
        int z = x + 10; // 成功
        z = y + 10; // 失败
        z = x + y; // 失败
    }

这里在判断结果类型时,我通过java中的自动拆装箱写了一个判断方法来进行验证,可以验证运算结果的数据类型到底是什么。示例代码如下:

@Test
    public void test(){
       byte a = 127;
       a += a;
       judge(a);// 输出 byte

       int x = 10;
       long y = 100L;
       judge(x + y);// 输出 long
    }

    /**
     * @desc 判断传入参数的数据类型
     * @author dataozi
     * @date 2020/5/31 17:24
     */
    private void judge(Object o){
        if(o instanceof Integer){
            System.out.println("int");
        }else if(o instanceof Double){
            System.out.println("double");
        } else if(o instanceof Long){
            System.out.println("long");
        } else if(o instanceof Short){
            System.out.println("short");
        } else if(o instanceof Byte){
            System.out.println("byte");
        } else if(o instanceof Float){
            System.out.println("float");
        } else {
            System.out.println("我裂开了");
        }
    }

a = a + 1;实际上是byte = int + int,这里需要进行强转a = (byte)(a + 1),不强转的话编译报错。

a += a;转换一下是a = a + a,实际还是byte = int + int,这里为什么没有报错呢,是+=会自动进行强转,实际上最终变成了byte = (byte)(int + int)。

a += 1;和a += a同理。

a =+ a;这是我自己想的迷惑题,这不是个运算操作,实际就是 a = +a;这里编译器会报错。编译器不会进行计算,它虽然知道a是byte性,但是它不知道+a会不会超过byte的范围,这里+a会被自动升为int,实际变成了byte = +int,所以需要强转a = (byte)+a;

a =+ 1;这时实际是a= +1;因为是直接赋值没有计算操作,编译器知道+1在byte的范围之内,所以自动进行了类型转换。换成a =+ 129的话就编译报错。

int z = x +10;没什么好说的,int = int + int。

z = y + 10;y为long型,所以结果会升为long型,int = long + int,z是int型,需要强转z = (int)(y + 10)。

z = x + y;int = int + long,结果会升为long型,同样需要强转z = (int)(x + y)。

总结

在使用基本数据类型存储或计算时,一定要了解类型的范围和运算规则,否则带来因为类型使用不当带来的精度损失,可能是很严重的。

下篇总结什么是进制,进制之间的换算