WEB开发网
开发学院WEB开发Jsp 达内金牌讲师唐亮Java语言细节(中) 阅读

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

 2008-01-05 20:38:25 来源:WEB开发网   
核心提示: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、学
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
}>
}©达内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=
}©达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛  dtzkF,
©达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛  "tYclass 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
}©达内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==
}©达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛  M^y4`
©达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛  /G3)利用串行化来做深复制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);V d
return(oi.readObject());+5}2M/
}©达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛  Q
©达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛  yU8
这样做的前提是对象以及对象内部所有引用到的对象都是可串行化的,否则,就需要仔细考察那些不可串行化的对象可否设成transient,从而将之排除在复制过程之外。上例代码改进如下。B%IU!e
©达内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
}©达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛  <#
class Student implements SerializableG
{©达内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-
{©达内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());\
}©达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛  __
©达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛  y&0
}©达内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{
©达内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;Lp pu
this.age=age; X^4"B
}zdO*:`
©达内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
}Vi o
}©达内IT技术论坛—中国人学Java、学C++、学C#/.Net、学软件、学IT的地方 -- 达内科技论坛  ©达内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

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
}©达内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

Tags:达内 金牌 讲师

编辑录入:爽爽 [复制链接] [打 印]
赞助商链接