Java架构师专题

五分钟学会java中的基础类型封装类

2019-09-19  本文已影响0人  愚公要移山

在刚刚学习java的时候,老师不止一次的说过java是一种面向对象的语言,万物皆对象。对于java中的基础数据类型,由于为了符合java中面向对象的特点,同样也有其封装类。这篇文章对其有一个认识。

一、基本认识

其实在jdk1.5之前,在基础数据类型与其封装器之间的转化必须手动进行,但是从jdk1.5之后,由于提供了自动装箱的机制,因此我们不再手动进行了。

装箱:基础类型转封装类型。Integer a = 3

底层实现:Integer a = Integer.valueOf(3);

拆箱:封装类型转基础类型。int b = a;

底层实现:int b = a.intValue();

既然封装类能够封装基础类型,那么能封装的范围是多少呢?

基本类型 封装器 字节数 最大值 最小值
byte Byte 1byte 2^7 - 1 -2^7
short Short 2byte 2^15 - 1 -2^15
char Character 2byte 2^16 - 1 0
int Integer 4byte 2^31 - 1 -2^31
long Long 8byte 2^63 - 1 -2^63
float Float 4byte 3.4e+38 1.4e-45
double Double 8byte 1.8e+308 4.9e-324
boolean Boolean 1byte/4byte/不明确 - -

我们对其进行了一个总结。不过我们应该注意到boolean类型没有给出精确的定义,可能是一个字节也有可能是四个字节,这是为什么呢?java虚拟机规范中规定的是4个字节,但是不同的厂家虚拟机可能不同,所以可能不会按照规范来。

以上这张图想必我们都不陌生,每一种基础类型都有一个唯一的封装类。而且也给出了字节数、最大值最小值等。下面我们就看一下其基本使用:

public class Test{
    public static void main(String[] args) {
        Integer int_a= new Integer(3);
        int int_b=  int_a;//自动完成了拆箱
        
        char char_a = new Character('a');
        char char_b = char_a;//自动完成了拆箱
        
        //其他类似
    }
}

二、基础类型与封装类的区别

1、传递方式不同

基本类型是按值传递,而封装类型是按引用传递的。int是基本类型,直接存放数值;Integer类会产生一个对象引用指向这个对象。

2、存储位置不同

基本类型存储在栈中,封装类的引用存储在栈中,而值是存在堆中。这样看上去好像基础封装类多此一举,而且基本类型的速度也确实会比封装类更快。为什么快呢?这是因为封装类涉及到了对象内存的分配和垃圾回收。但是基本类型直接拿起来就计算了。

三、源码分析

在讲解源码之前我们先给出一个神奇的例子,请看下面的代码:

public class Test{
    public static void main(String[] args) {
        //这种情况会返回true
        Integer a1 = 10;
        Integer a2 = 10;
        System.out.println(a1 == a2); 
        //这种情况会返回false?????
        Integer b1 = 1000;
        Integer b2 = 1000;
        System.out.println(b1 == b2); 
    }
}

第一种情况感觉和第二种情况一模一样呀,为什么第二种是false呢?想要知道原因,我们就必须要深入到源码中找寻答案。

public static Integer valueOf(int i) {
    // i是否在表示范围[-128, 127]中
    if (i >= IntegerCache.low && i <= IntegerCache.high) 
        // 如果在直接取出
        return IntegerCache.cache[i + (-IntegerCache.low)];
    // 如果不在,则创建一个新的
    return new Integer(i);
}

也就是说,在合理表示范围之内就直接拿出来一个旧的数据,如果不在表示范围之内那就创建一个新的。但是对于源码的了解不能仅限于此。我们还是按部就班的分析,下面我们以Integer封装类为例。

1、参数

//最小值::-2147483648
//其实也就是2的10次方,然后取负
@Native public static final int   MIN_VALUE = 0x80000000;
//最大值
//其实也就是2的10次方
@Native public static final int   MAX_VALUE = 0x7fffffff;

2、方法

//方法1:将字符串参数解析为有符号的整数
//第二个参数指定基数
public static int parseInt(String s, int radix) ;

//方法2:类型转换,并给出其中一个例子
public byte byteValue(){
    return (byte) value;
}
public short shortValue()
public int intValue()
public long longValue()
public float floatValue()
public double doubleValue()
    
//方法3:求a和b的和
public static int sum(int a, int b)
//方法4:求最大最小值
public static int max(int a, int b){
    return Math.max(a, b);
}
public static int min(int a, int b){
    return Math.min(a, b);
}

3、缓存

上面支持给出了一些基本的使用方法,不过最核心的还是缓存范围的实现。下面我们看一下:

    private static class IntegerCache {
        static final int low = -128;
        static final int high;
        static final Integer cache[];
        static {
            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);
                    h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
                } catch( NumberFormatException nfe) {
                }
            }
            high = h;
            cache = new Integer[(high - low) + 1];
            int j = low;
            for(int k = 0; k < cache.length; k++)
                cache[k] = new Integer(j++);
            assert IntegerCache.high >= 127;
        }

        private IntegerCache() {}
    }

上面这些代码能看懂最好,看不懂我来描述一下大概意思:这是一个静态内部类,类中定义一个静态cache数组,这个静态数组预先放了自己指定范围内的数据,拆箱的时候首先判断范围然后从缓存数组中去抓取数据。就是这么一个简单的过程。

四、使用场景

上面分析了这么多,最主要的还是如何使用,在什么地方使用。下面我们就总结几个场景:

1、类型之间的转换:

String b = "123"; 
int c = Integer.parseInt(b); 
//如果b中含有非法字符,则会报错
NumberFormatException

2、泛型中使用

List<>为原始类型,不指定元素类型时,会出现不安全的警告:

List is a raw type. References to generic type List<E> should be parameterized

大概意思就是:List的<>中应该指定清楚是那种类型,如Integer、String等。

3、强制类型转换

4、集合中使用

public static void main(String[] args) {
    List<Integer> list = new ArrayList<Integer>(2);
    list.add(1);
    list.add(null);
    for (int value:list){
        System.out.println("value:"+value);
    }
}

这种情况下会出现空指针异常。

OK。今天的文章先到这里,如有问题还请批评指正。


默认标题_方形二维码_2019.08.16.png
上一篇下一篇

猜你喜欢

热点阅读