分享
 
 
 

达内金牌讲师唐亮Java语言细节(中)

王朝java/jsp·作者佚名  2008-05-31
窄屏简体版  字體: |||超大  

class Teacher{4>B

String name;li\

int age;AgDu~g

Teacher(String name,int age){y

this.name=name;T

this.age=age;

}?afP

}©达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛p#j2q0

class Student implements Cloneable{H=x\,

String name;?]

int age;<

Teacher t;//学生1和学生2的引用值都是一样的。<0ptC2

Student(String name,int age,Teacher t){Z

this.name=name;?>daz.

this.age=age;wnC`

this.t=t;-6

}Ls>

public Object clone(){-[HYy/

Student stu=null;s\(.G

try{q

stu=(Student)super.clone();~&9

}catch(CloneNotSupportedException e){a>g

e.printStackTrace();`DI

}]Wwt

stu.t=(Teacher)t.clone();V$

return stu;-Jdj"M

}]a5gf

public static void main(String[] args){M

Teacher t=new Teacher("tangliang",30);LOy9%

Student s1=new Student("zhangsan",18,t);6=<;

Student s2=(Student)s1.clone();

s2.t.name="tony";QpSIF5

s2.t.age=40;u

System.out.println("name="+s1.t.name+","+"age="+s1.t.age);&Yd;

//学生1的老师成为tony,age为40。9

}>

}&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛TFmz-

jY

那应该如何实现深层次的克隆,即修改s2的老师不会影响s1的老师?代码改进如下。{p`

class Teacher implements Cloneable{6"CJU

String name;Q,1Q/

int age;1$

Teacher(String name,int age){Rd

this.name=name;9+>

this.age=age;T}<

}#2@N

public Object clone(){sC

Object obj=null;@

try{L"{

obj=super.clone();7;[WA

}catch(CloneNotSupportedException e){T

e.printStackTrace();H9XP<7

}

return obj;VTf.C

}Acb=

}&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛dtzkF,

&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛"tY

class Student implements Cloneable{`3kBU

String name;y

int age;jn>

Teacher t;zS_~)P

Student(String name,int age,Teacher t){f5'qD

this.name=name;b

this.age=age;?=jc

this.t=t;-(Dt6(

}5

public Object clone(){.A

Student stu=null;8rB!l7

try{2'{<-}

stu=(Student)super.clone();.!58X@

}catch(CloneNotSupportedException e){T

e.printStackTrace();$:

}D^

stu.t=(Teacher)t.clone();R_?yP

return stu;!d=

}8o_b

}&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛 drr

public static void main(String[] args){2

Teacher t=new Teacher("tangliang",30);K@S,3

Student s1=new Student("zhangsan",18,t);VZri7

Student s2=(Student)s1.clone();b#

s2.t.name="tony";JV_

s2.t.age=40;G

System.out.println("name="+s1.t.name+","+"age="+s1.t.age);Q

//学生1的老师不改变。]2==

}&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛M^y4`

&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛/G

3)利用串行化来做深复制3

把对象写到流里的过程是串行化(Serilization)过程,Java程序员又非常形象地称为“冷冻”或者“腌咸菜(picking)”过程;而把对象从流中读出来的并行化(Deserialization)过程则叫做“解冻”或者“回鲜(depicking)”过程。应当指出的是,写在流里的是对象的一个拷贝,而原对象仍然存在于JVM里面,因此“腌成咸菜”的只是对象的一个拷贝,Java咸菜还可以回鲜。nt

在Java语言里深复制一个对象,经常可以先使对象实现Serializable接口,然后把对象(实际上只是对象的一个拷贝)写到一个流里(腌成咸菜),再从流里读出来(把咸菜回鲜),便可以重建对象。^

如下为深复制源代码。ofO

public Object deepClone(){ p*][t

//将对象写到流里#

ByteArrayOutoutStream bo=new ByteArrayOutputStream();LW

ObjectOutputStream oo=new ObjectOutputStream(bo);#l,

oo.writeObject(this);Hd[0?2

//从流里读出来?

ByteArrayInputStream bi=new ByteArrayInputStream(bo.toByteArray());d%

ObjectInputStream oi=new ObjectInputStream(bi);Vd

return(oi.readObject());+5}2M/

}&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛Q

&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛yU8

这样做的前提是对象以及对象内部所有引用到的对象都是可串行化的,否则,就需要仔细考察那些不可串行化的对象可否设成transient,从而将之排除在复制过程之外。上例代码改进如下。B%IU!e

&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛0D

class Teacher implements Serializable{A

String name;F_3

int age;~;qX"C

Teacher(String name,int age){DO

this.name=name;v?*E

this.age=age;Q$CjG=

}uSBTg

}&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛<#

class Student implements SerializableG

{&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛H=O

String name;//常量对象。A

int age;#"<%

Teacher t;//学生1和学生2的引用值都是一样的。ukLf

Student(String name,int age,Teacher t){<58

this.name=name;dIkmB

this.age=age;5d?b'

this.p=p;9.XA:

}xQ

public Object deepClone() throws IOException,dDOIF!

OptionalDataException,ClassNotFoundExceptionZ_Z-

{&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛CFMPe

//将对象写到流里'Yy

ByteArrayOutoutStream bo=new ByteArrayOutputStream();g7E05

ObjectOutputStream oo=new ObjectOutputStream(bo);1W

oo.writeObject(this);Yj+Qe

//从流里读出来gkC0

ByteArrayInputStream bi=new ByteArrayInputStream(bo.toByteArray());I%yw

ObjectInputStream oi=new ObjectInputStream(bi);9

return(oi.readObject());\

}&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛__

&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛y&0

}&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛O

public static void main(String[] args){"

Teacher t=new Teacher("tangliang",30);3g>u

Student s1=new Student("zhangsan",18,t);v^Aow

Student s2=(Student)s1.deepClone();/5#t

s2.t.name="tony";/

s2.t.age=40;=9-?

System.out.println("name="+s1.t.name+","+"age="+s1.t.age);daiY@

//学生1的老师不改变。{BR#

}

5,String类和对象池

我们知道得到String对象有两种办法:v}

String str1="hello";n

String str2=new String("hello");xE

这两种创建String对象的方法有什么差异吗?当然有差异,差异就在于第一种方法在对象池中拿对象,第二种方法直接生成新的对象。在JDK5.0里面,Java虚拟机在启动的时候会实例化9个对象池,这9个对象池分别用来存储8种基本类型的包装类对象和String对象。当我们在程序中直接用双引号括起来一个字符串时,JVM就到String的对象池里面去找看是否有一个值相同的对象,假如有,就拿现成的对象,假如没有就在对象池里面创建一个对象,并返回。所以我们发现下面的代码输出true:"GF^`C

String str1="hello";t`

String str2="hello";H

System.out.println(str1==str2);\5]+%

这说明str1和str2指向同一个对象,因为它们都是在对象池中拿到的,而下面的代码输出为false:0)BvDc

String str3="hello"7xu{

String str4=new String("hello");YH

System.out.println(str3==str4);8

因为在任何情况下,只要你去new一个String对象那都是创建了新的对象。JcW*?

与此类似的,在JDK5.0里面8种基本类型的包装类也有这样的差异:>

Integer i1=5;//在对象池中拿^

Integer i2 =5;//所以i1==i2)T!

Integer i3=new Integer(5);//重新创建新对象,所以i2!=i3R{

&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛jxpB

对象池的存在是为了避免频繁的创建和销毁对象而影响系统性能,那我们自己写的类是否也可以使用对象池呢?当然可以,考察以下代码:*:s;

class Student{d"'D

private String name;v]D>

private int age;>@e

private static HashSet pool=new HashSet();//对象池pP~i1

ew#F2a

public Student(String name,int age){jf e

this.name=name;Lppu

this.age=age;X^4"B

}zdO*:`

&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛4yBp

//使用对象池来得到对象的方法r

public static Student newInstance(String name,int age){A7b

//循环遍历对象池%B

for(Student stu:pool){uY4_

if(stu.name.equals(name) && stu.age==age){Ib!yH

return stu;6;Rk(9

}K

}Q1%g[o

//假如找不到值相同的Student对象,则创建一个Student对象T

//并把它加到对象池中然后返回该对象。0ac~

Student stu=new Student(name,age);.;=S

pool.add(stu);$pO*h

return stu;JdF

}Vio

}&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛

&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛H-

public class Test{-G>_

public static void main(String[] args){yy

Student stu1=Student.newInstance("tangliang",30);//对象池中拿+

Student stu2=Student.newInstance("tangliang",30);//所以stu1==stu2J p5

Student stu3=new Student("tangliang",30);//重新创建,所以stu1!=stu3B'@>

System.out.println(stu1==stu2);i,:r4

System.out.println(stu1==stu3);)dg_

}eZA

}&copy;

6,2.0-1.1==0.9吗?Vz&]G0

考察下面的代码:3z"I

double a=2.0,b=1.1,c=0.9;a

if(a-b==c){$s

System.out.println("YES!");i8

}else{)i

System.out.println("NO!");`cQI

}&copy;达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛S

以上代码输出的结果是多少呢?你认为是“YES!”吗?那么,很遗憾的告诉你,不对,Java语言再一次cheat了你,以上代码会输出“NO!”。为什么会这样呢?其实这是由实型数据的存储方式决定的。我们知道实型数据在内存空间中是近似存储的,所以2.0-1.1的结果不是0.9,而是0.88888888889。所以在做实型数据是否相等的判定时要非常的谨慎。一般来说,我们不建议在代码中直接判定两个实型数据是否相等,假如一定要比较是否相等的话我们也采用以下方式来判定:Sb_

if(Math.abs(a-b)<1e-5){{h$?z

//相等?yh<

}else{/Z

//不相等1]]z-

}kx`jG

上面的代码判定a与b之差的绝对值是否小于一个足够小的数字,假如是,则认为a与b相等,否则,不相等。

7,判定奇数

以下的方法判定某个整数是否是奇数,考察是否正确:'L#a

public boolean isOdd(int n){VJ1*

return (n%2==1);Q6Dz

}n

很多人认为上面的代码没问题,但实际上这段代码隐藏着一个非常大的BUG,当n的值是正整数时,以上的代码能够得到正确结果,但当n的值是负整数时,以上方法不能做出正确判定。例如,当n=-3时,以上方法返回false。因为根据Java语言规范的定义,Java语言里的求余运算符(%)得到的结果与运算符左边的值符号相同,所以,-3%2的结果是-1,而不是1。那么上面的方法正确的写法应该是:_

public boolean isOdd(int n){Y$;*\c

return (n%2!=0);h5

}x

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有