美文网首页Java架构师专题
五分钟学会java中的基础类型封装类

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

作者: 愚公要移山 | 来源:发表于2019-09-19 10:03 被阅读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

    相关文章

      网友评论

        本文标题:五分钟学会java中的基础类型封装类

        本文链接:https://www.haomeiwen.com/subject/qholuctx.html