Java核心技术

Java包装器与装箱,拆箱

0.包装器

  在Java中,所有基本类型都有一个与之对应的类。像int类型与Integer类相对应,double类型与Double类相对应。这些类被称为包装器(wrapper),或者叫对象包装器。Java有8种基本类型,有9个包装器,分别为:Intger、Long、Short、Byte、Double、Float、Character、Boolean以及Void。前6个类都派生于一个公共的超类Number。

  包装器是不可变的。也就是说构造好了包装器,就不能更改包装在其中的值。此外,包装器类是final类,无法定义它们的子类。

 

1.装箱

  装箱(boxing),是从Java SE 5.0开始出现的新特性。装箱自动将基本数据类型转换为对应包装器对象。

没有装箱,如果要生成一个数值为10的Integer对象,可以这么做:

Integer a=new Integer(10); 

有了装箱,就可以简化为这样:

Integer a=10;//即Integer a=Integer.valueOf(10);

 通过反编译class文件后,我们知道装箱实际上是通过valueOf()方法实现的,该方法返回一个Integer对象。

 

2.拆箱

  拆箱(unboxed),与装箱是正好相反的操作。自动将包装器对象转换为对应的基本数据类型。

Integer i=new Integer(5);
int b=i;//自动将Integer类对象变成int类数据类型,再赋给int类变量b

如果没有拆箱,就变为下面的代码

Integer i=new Integer(5);
int b=i.intValue(); //intValue方法以int的形式返回Integer对象的值

通过反编译,我们知道拆箱是通过xxxValue()方法实现的,该方法返回一个xxx类型的值。 

 

3.装箱和拆箱是编译器认可的,而不是虚拟机。

编译器在生成类的字节码的同时,插入必要的方法调用;虚拟机只是执行这些字节码。

 

4.==与equals()方法

由于装箱拆箱的存在,常常会给人一种错觉,让人认为基本数据类型和它们对应的包装器对象是一样的。

“==“操作符用于比较它左右的操作对象是否相同。  

  1).当==符号比较基本数据类型时,比较的是它们的值。

  2).当==符号计较对象时,比较的是它们的是否指向同一个区域(即是否有相同的引用)。

  3).当==操作符的两边,一个操作数是基本数据类型,另一个是对象时,则会将对象进行拆箱,从而变成两个基本数据类型进行值的比较。

所以避免出错和造成不必要的混乱,在比较两个包装器对象时,尽量不要使用==,而是使用equals()方法。

来看使用"=="进行比较的几个例子:

例一:

Integer a = new Integer(100);  
Integer b = 100;  
System.out.println(a == b); 

答案是false。

例二:

Integer a = 100;  
Integer b = 100;  
System.out.println(a == b);  

答案是true。

例三:

Integer a = 156;  
Integer b = 156;  
System.out.println(a == b); 

答案是false。  

例四:

Integer a = Integer.valueOf(100);  
Integer b = 100;  
System.out.println(a == b);

答案是true。

 

为什么会是这样子的结果呢?我们先从例四开始分析。

之前说过,装箱实际上是因为编译器调用了valueOf()方法。所以例四实际就是以下代码:

Integer a = Integer.valueOf(100);  
Integer b = Integer.valueOf(100); 
System.out.println(a == b);

我们再来看一下Integer类中,valueOf()的具体实现:

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

通过查看IntegerCache类的具体实现,我们发现low就是-128,而high就是127。所以valueOf()方法就很明确了,当参数i为在区间[-128,127]上的int数值时,就返回缓存中的Integer对象(也就是说,从-128到127这258个数已经一次性被初始化好了,而不是创建新的);i不在这个区间内,就返回一个新创建的Integer对象。

所以在例四中,两次调用valueOf(100),返回的是同一个对象,这个对象是已经存在的。a和b指向同一个对象,所以a==b为true。

在例三中,装箱调用valueOf(),但参数为156,不在-128和127之间,所以新建了两个Integer对象,a和b指向不同对象,所以输出为false。

例一,例二也因此可以理解了。

 

但要注意,Integer、Short、Byte、Character、Long这几个类的valueOf方法的实现是类似的。

     Double、Float的valueOf方法的实现是类似的。

Double类中,valueOf()的原代码如下

public static Double valueOf(double d) {
        return new Double(d);
    }

所以下面代码的输出为两个false也不难理解。

复制代码

Double a = 100.0;
Double b = 100.0;
Double c = 200.0;
Double d = 200.0;
System.out.println(a==b);
System.out.println(c==d);

复制代码

 

最后再来看一个比较复杂例子:

例五:

复制代码

Integer a = 1;
Integer b = 2;
Integer c = 3;
Integer d = 3;
Integer e = 200;
Integer f = 200;
Long g = 3L;
Long h = 2L;
         
System.out.println(c==d);
System.out.println(e==f);
System.out.println(c==(a+b));
System.out.println(c.equals(a+b));
System.out.println(g==(a+b));
System.out.println(g.equals(a+b));
System.out.println(g.equals(a+h));

复制代码

答案分别为 true false true true true true false true。

前两个不需要解释了,前面的例子里已经有了。

 第三个输出中,a+b会进行拆箱(你无法将相加两个基本类型对象),先两个int型数值然后再相加,因此==号右边变成了一个int型的数值。而前面的"3)"提到过,这里还会进行一次拆箱,将c也拆箱成int值,从而变成int值的比较,所以为true。

第四个输出中,因为有加号,所以和前面一样,a和b分别拆箱,然后相加,得到int型结果3。但和前面不同是的,equals()的参数是对象,所以3又自动装箱成了包装器对象。前面说过,包装器对象之间的比较最好用equals()方法,这样就会避免不必要的错误。

第五个输出中,因为是数值的比较,虽然一个是long型的3,一个是int型的3,但它们还是相同的。

最后两个也是大同小异,注意,int数值+long数值,根据基本数据类型的隐式转换原则,得到的结果是long型数值,最后会装箱成Long类对象。包装器对象用equals()方法进行比较,先比较对象的类相同是否相同,若不同,则为false;若属于同一类,且包装在其中的值相同,则为true。所以最后一个输出为true,倒数第二个输出为false。

在软件设计里到处都是模式,框架。有次朋友问什么是模式?我也在学习中,就我的学习经验,给出以下小结。(注意:个人观点,仅供参考,欢迎指正。)??1.什么是模式???模式,即pattern。其实就是解决某一类问题的方法论。你把解决某类问题的方法总结归纳到理论高度,那就是模式。??Alexander给出的经典定义是:每个模式都描述了一个在我们的环境中不断出现的问题,然后描述了该问题的解决方案的核心。通过这种方式,你可以无数次地使用那些已有的解决方案,无需在重复相同的工作。??模式有不同的领域,建筑领域有建筑模式,软件设计领域也有设计模式。当一个领域逐渐成熟的时候,自然会出现很多模式。??什么是框架???框架,即framework。其实就是某种应用的半成品,就是一组组件,供你选用完成你自己的系统。简单说就是使用别人搭好的舞台,你来做表演。而且,框架一般是成熟的,不断升级的软件。??2.为什么要用模式???因为模式是一种指导,在一个良好的指导下,有助于你完成任务,有助于你作出一个优良的设计方案,达到事半功倍的效果。而且会得到解决问题的最佳办法。??为什么要用框架???因为软件系统发展到今天已经很复杂了,特别是服务器端软件,设计到的知识,内容,问题太多。在某些方面使用别人成熟的框架,就相当于让别人帮你完成一些基础工作,你只需要集中精力完成系统的业务逻辑设计。而且框架一般是成熟,稳健的,他可以处理系统很多细节问题,比如,事物处理,安全性,数据流控制等问题。还有框架一般都经过很多人使用,所以结构很好,所以扩展性也很好,而且它是不断升级的,你可以直接享受别人升级代码带来的好处。??框架一般处在低层应用平台(如J2EE)和高层业务逻辑之间的中间层。??软件为什么要分层???为了实现“高内聚、低耦合”。把问题划分开来各个解决,易于控制,易于延展,易于分配资源…总之好处很多啦:)。??3.以下所述主要是JAVA,J2EE方面的模式和框架:??常见的设计模式有什么???首先,你要了解的是GOF的《设计模式--可复用面向对象软件的基础》一书(这个可以说是程序员必备的了),注意:GOF不是一个人,而是指四个人。它的原意是Gangs Of Four,就是“四人帮”,就是指此书的四个作者:Erich Gamma,Richard Helm,Ralph Johnson,John Vlissides。这本书讲了23种主要的模式,包括:抽象工厂、适配器、外观模式等。??还有其他的很多模式,估计有100多种。??软件设计模式太多,就我的理解简单说一下最常见的MVC模式。??MVC模式是1996年由Buschmann提出的:??模型(Model):就是封装数据和所有基于对这些数据的操作。??视图(View):就是封装的是对数据显示,即用户界面。??控制器(Control):就是封装外界作用于模型的操作和对数据流向的控制等。??另外:??RUP(Rational Unified Process)软件统一过程,XP(Extreme Programming)极端编程,这些通常被叫做“过程方法”,是一种软件项目实施过程的方法论,它是针对软件项目的实施过程提出的方法策略。也是另一个角度的模式。??4.常见的JAVA框架有什么???WAF:??全称:WEB APPLICATION FRAMEWORK??主要应用方面:EJB层,(WEB层也有,但是比较弱)。??主要应用技术:EJB等??出处:http://java.sun.com/blueprints/code/index.html??简述:这是SUN在展示J2EE平台时所用的例子PetStore(宠物商店系统)里面的框架。是SUN蓝皮书例子程序中提出的应用框架。它实现了 MVC和其他良好的设计模式。SUN的网站上有技术资料,最好下载PetStore来研究,WEBLOGIC里自带此系统,源码在beaweblogic700samplesserversrcpetstore。这是学习了解J2EE的首选框架。??免费。??Struts:??主要应用方面:WEB层。??主要应用技术:JSP,TagLib,JavaBean,XML等??出处:http://jakarta.apache.org/struts/index.html??简述:这是APACHE的开源项目,目前应用很广泛。基于MVC模式,结构很好,基于JSP。Jbuilder8里已经集成了STRUTS1.02的制作。??免费。??简述WAF+STRUTS结合的例子:WEB层用STRUTS,EJB层用WAF:??JSP(TagLib)——>ActionForm——>Action ——> Event——>EJBAction——>EJB ——>DAO——>Database JSP(TagLib)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值