Java 基本数据类型和封装类型

本文详细介绍了Java中Integer对象的常量池技术,包括其工作原理、创建方式以及为何使用常量池。解释了Integer对象的缓存范围[-128,127],并探讨了常量池大小的选择。同时,文章讲解了自动装箱和拆箱的概念及其实现,展示了如何在代码中应用这些概念,并提供了相关示例。
摘要由CSDN通过智能技术生成

八种基本数据类型的封装数据类型如下表:

基本数据类型封装数据类型类型描述字节数范围
byteByte整型1[-128, 127]
shortShort整型2[- 32768, 32767] or [-2^15, 2^15-1]
intInteger整型4[-2^31, 2^31-1]
longLong长整型8[-2^63, 2^63 -1]
floatFloat单精度浮点型4-
doubleDouble双精度浮点型8-
charCharacter字符型2[0, 65535]
booleanBoolean布尔类型>1-

Byte、Short、Integer、Long、Character、Boolean 实现了常量池技术。前四种常量值范围为[-128, 127],Character 范围为[0, 127],Boolean 为 true 或 false

Float、Double 没有实现常量池技术。
在这里插入图片描述

解读 Integer 常量池

简介

Integer 是基本数据类型 int 的封装类型,使用位移的方式实现大部分操作。
提供了 int 转 String,String 转 int,以及处理 int 时的一些常量和方法。

常量池(对象缓存)

先看代码

        int a = 127;
        int b = 128;
        int c = -128;
        int d = -129;

        Integer e = 127;
        Integer f = 127;
        Integer g = new Integer(127);

        Integer h = 128;
        Integer i = 128;
        Integer j = new Integer(128);

        Integer k = -128;
        Integer l = -128;
        Integer m = new Integer(-128);

        Integer n = -129;
        Integer o = -129;
        Integer p = new Integer(-129);

        System.out.println(a == e);            // true
        System.out.println(a == g);            // true
        System.out.println(e == f);            // true
        System.out.println(e == g);            // false
        System.out.println(e.equals(a));       // true
        System.out.println(g.equals(a));       // true
        System.out.println(e.equals(f));       // true
        System.out.println(e.equals(g));       // true

解读:

  1. new Integer(123),是创建一个新的 Integer 对象,不会从常量池中获取值;
  2. Integer i = 123,相当于调用 valueOf() 方法,该值会从常量池中获取;

双等号 ==

  1. 用于比较基本数据类型的时候比较的是数值是否相等
  2. 用于比较两个对象时比较的是两个对象的内存地址

Integer 对象的两种创建方式

  1. 直接赋值: 例如:Integer a = 1;
  2. 构造函数: 例如:Integer a = new Integer(1);
    直接赋值方式:通过 IntegerCache 对象缓存进行复制,缓存数值的范围为 [-128, 127],java.lang.Integer.IntegerCache.high 可以设置缓存的最大值,-XX:AutoBoxCacheMax=<size>
    构造函数方式:通过 Integer 的构造函数创建的 Integer 对象,不会从缓存中赋值,而是创建一个新的对象。

缓存对象代码:

private static class IntegerCache {
        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;
        }

        private IntegerCache() {}
    }

为什么要使用常量池技术 ?

将常用的数值缓存到JVM常量池中在一定程度上可以提高速率。

常量池为什么选择这个范围,可不可以更大或更小 ?

常量池的大小没有一个最优的值,要根据项目的业务需求而定,可以通过 -XX:AutoBoxCacheMax=<size> 参数进行调整,个人习惯将其调整为 20000

小结

  1. Byte、Short、Integer、Long 都具有对象缓存。
  2. 对象缓存是为了提高性能;
  3. 缓存以支持 JLS 要求的 -128(含) 到 127(含)之间的值的自动装箱的对象标识语义。

自动装箱和拆箱

自动装箱是 Java 编译器在原始数据类型与其所对应的对象包装类型之间进行的自动转换。例如,将 int 转换为 Integer,将 double 转换为 Double 等等叫做自动装箱。如果转换以相反的方式进行,则称为拆箱。

这是自动装箱的最简单实例:

Character ch = 'a';

考虑以下代码:

List<Integer> li = new ArrayList<>();
for (int i = 1; i < 50; i += 2) {
	li.add(i);
}

尽管你将 int 值作为原始数据类型而不是 Integer 对象添加到 li 中,但代码会编译。因为 li 是一个 Integer 对象列表,而不是一个 int 值列表,你可能想知道为什么 Java 编译器不会发出编译时错误。编译器不会产生错误,因为它从 i 创建了一个 Integer 对象并将该对象添加到 li。因此,编译器在运行时将之前的代码转换为以下代码:

List<Integer> li = new ArrayList<>();
for (int i = 1; i < 50; i += 2) {
	li.add(Integer.valueOf(i));
}

将原始数据类型值(例如 int)转换为相应包装类(Integer)的对象称为自动装箱。当原始数据类型的值是:

  • 作为参数传递给需要相应包装类的对象的方法;
  • 分配给相应包装类的变量;

考虑以下方法:

public static int sumEven(List<Integer> li) {
	int sum = 0;
	for (Integer i : li) {
		if (i % 2 ==0){
			sum += i;
		}
		return sum;
	}
}

因为取余(%)和一元加号(+=)运算符不适用于 Integer 对象,你可能想知道为什么 Java 编译器编译该方法而不发出任何错误。编译器不会发生错误,因为它会在运行时调用 intValue 方法将 Integer 转换为 int:

public static int sumEven(List<Integer> li) {
	int sum = 0;
	for (Integer i : li) {
		if (i.intValue() % 2 == 0) {
			sum += i.intValue();
		}
		return sum;
	}
}

将包装类型(Integer)的对象转换为其对应的原始(int)值称为拆箱。当包装类的对象是:

  • 作为参数传递给需要相应原始类型值的方法;
  • 分配给相应原始类型的变量;

拆箱示例展示了这是如何工作的:

import java.util.ArrayList;
import java.util.List;
public class Unboxing {
	public static void main(String[] args) {
		Integer i = new Integer(-8);
		// 1. 通过方法调用拆箱
		int absVal = absoluteValue(i);
		System.out.println("absolute value of " + i + " = " + absVal);
		// 通过调用方法自动装箱
		List<Double> ld = new ArrayList<>();
		ld.add(3.1416);
		// 通过分配拆箱
		double pi = ld.get(0);
		System.out.println("pi = " + pi);
	}
	public static int absoluteValue(int i) {
		return (i < 0) ? -i : i;
	}
}

该程序打印一下内容:

absolute value of -8 = 8
pi = 3.1416

自动装箱和拆箱使开发人员能够编写更简洁的代码,使其更易于阅读。下表列出了 Java 编译器用于自动装箱和拆箱的基本类型及其相应的包装类型:

Primitive typeWrapper class
booleanBoolean
byteByte
charCharacter
floatFloat
intInteger
longLong
shortShort
doubleDouble
  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值