Java 基础
Java 容器
Java 并发
设计模式
目录

数据类型

# 数据类型

Java 编程语言是静态类型的,这意味着所有变量在使用之前都必须先声明。这涉及声明变量的类型和名称,正如您已经看到的

# 基本类型

  • byte/8
  • char/16
  • short/16
  • int/32
  • float/32
  • long/64
  • double/64
  • boolean/~

boolean 只有两个值:true、false,可以使用 1 bit 来存储,但是具体大小没有明确规定。JVM 会在编译时期将 boolean 类型的数据转换为 int,使用 1 来表示 true,0 表示 false。JVM 支持 boolean 数组,但是是通过读写 byte 数组来实现的。

  • Primitive Data Types (opens new window)
  • The Java® Virtual Machine Specification (opens new window)

# 包装类型

基本类型都有对应的包装类型,基本类型与其对应的包装类型之间的赋值使用自动装箱与拆箱完成。

Integer x = 2;     // 装箱 调用了 Integer.valueOf(2)
int y = x;         // 拆箱 调用了 X.intValue()
1
2
  • Autoboxing and Unboxing (opens new window)

# 在数字文字中使用下划线字符

在 Java SE 7 及更高版本中,任意数量的下划线字符 ( _) 都可以出现在数字文字中的数字之间的任何位置。例如,此功能可让您分隔数字文字中的数字组,从而提高代码的可读性。

例如,如果您的代码包含许多数字,则可以使用下划线字符将三组数字分隔开,类似于使用逗号或空格等标点符号作为分隔符。

下面的示例展示了在数字文字中使用下划线的其他方法:

  • 信用卡号 = 1234_5678_9012_3456L;

  • 社交安全号码 = 999_99_9999L;

  • 浮点数 pi = 3.14_15F;

  • 长十六进制字节 = 0xFF_EC_DE_5E;

  • 长十六进制字 = 0xCAFE_BABE;

  • 长 maxLong = 0x7fff_ffff_ffff_ffffL;

  • 字节nybbles = 0b0010_0101;

  • 长字节=0b11010010_01101001_10010100_10010010;

  • 下划线只能放在数字之间;

  • 以下位置不能放置下划线:

  • 在数字的开头或结尾

  • 浮点数字面值中的小数点附近

  • F或L后缀之前

  • 在需要一串数字的位置

# 缓存池

new Integer(123) 与 Integer.valueOf(123) 的区别在于:

  • new Integer(123) 每次都会新建一个对象;
  • Integer.valueOf(123) 会使用缓存池中的对象,多次调用会取得同一个对象的引用。
Integer x = new Integer(123);
Integer y = new Integer(123);
System.out.println(x == y);    // false
Integer z = Integer.valueOf(123);
Integer k = Integer.valueOf(123);
System.out.println(z == k);   // true
1
2
3
4
5
6

valueOf() 方法的实现比较简单,就是先判断值是否在缓存池中,如果在的话就直接返回缓存池的内容。

public static Integer valueOf(int i) {
    if (i >= IntegerCache.low && i <= IntegerCache.high)
        return IntegerCache.cache[i + (-IntegerCache.low)];
    return new Integer(i);
}
1
2
3
4
5

在 Java 8 中,Integer 缓存池的大小默认为 -128~127。

static final int low = -128;
static final int high;
static final Integer cache[];

static {
    // high value may be configured by property
    int h = 127;
    String integerCacheHighPropValue =
        sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
    if (integerCacheHighPropValue != null) {
        try {
            int i = parseInt(integerCacheHighPropValue);
            i = Math.max(i, 127);
            // Maximum array size is Integer.MAX_VALUE
            h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
        } catch( NumberFormatException nfe) {
            // If the property cannot be parsed into an int, ignore it.
        }
    }
    high = h;

    cache = new Integer[(high - low) + 1];
    int j = low;
    for(int k = 0; k < cache.length; k++)
        cache[k] = new Integer(j++);

    // range [-128, 127] must be interned (JLS7 5.1.7)
    assert IntegerCache.high >= 127;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29

编译器会在自动装箱过程调用 valueOf() 方法,因此多个值相同且值在缓存池范围内的 Integer 实例使用自动装箱来创建,那么就会引用相同的对象。

Integer m = 123;
Integer n = 123;
System.out.println(m == n); // true
1
2
3

基本类型对应的缓冲池如下:

  • boolean values true and false
  • all byte values
  • short values between -128 and 127
  • int values between -128 and 127
  • char in the range \u0000 to \u007F

在使用这些基本类型对应的包装类型时,如果该数值范围在缓冲池范围内,就可以直接使用缓冲池中的对象。

在 jdk 1.8 所有的数值类缓冲池中,Integer 的缓冲池 IntegerCache 很特殊,这个缓冲池的下界是 - 128,上界默认是 127,但是这个上界是可调的,在启动 jvm 的时候,通过 -XX:AutoBoxCacheMax=<size> 来指定这个缓冲池的大小,该选项在 JVM 初始化的时候会设定一个名为 java.lang.IntegerCache.high 系统属性,然后 IntegerCache 初始化的时候就会读取该系统属性来决定上界。

测试例子

package com.code.base.example1;

public class Main {
    public static void main(String[] args) {
        boolean b1 = true;
        byte byte1 = 0b1; //二进制
        short short1 = 3;
        int int1 = 4;
        float float1 = 5.f;
        double double1 = 6.1;

        long idcard = 1234_5678_9012_3456L;

        Integer i1 = 33;
        Integer i2 = 33;
        System.out.println(i1 == i2);// 输出 true
        System.out.println(i1.equals(i2));// 输出 true

        Float i11 = 333f;
        Float i22 = 333f;
        System.out.println(i11 == i22);// 输出 false
        System.out.println(i11.equals(i22));// 输出 true
        Double i3 = 1.2;
        Double i4 = 1.2;
        System.out.println(i3 == i4);// 输出 false
        System.out.println(i3.equals(i4));// 输出 true
    }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28

StackOverflow : Differences between new Integer(123), Integer.valueOf(123) and just 123 (opens new window)