糊里糊涂地用了Java快三年多了,一直没有很系统地读过一本Java的经典书籍。借到一本<<Core Java ,8th>>,记下来自己在里面领悟到的细节点滴。
1. 抽奖程序算法。
很简单的一个应用,由于抽奖的结果不能出现重复。以前会每次来查看Math.Random()的结果是不是在已经抽出的列表中,如果有的话重新抽一次。看到书中巧妙的实现,很灵活。记录如下:
//抽奖的样本总量数组 int[] numers=new int[allSeek]; for(int i=0;i<numers.length;i++){ numers[i]=i+1; } //抽奖得到的结果存储数组 int[] result=new int[pickerSeek]; for(int i=0;i<result.length;i++){ //随机抽取 int r=(int)(Math.random()*allSeek); //存储抽奖结果 result[i]=numers[r]; //由于已经抽出一个,下次抽奖的问题将减一,将当前被抽出的位置的样本用样本问题中的最后一个样本值替换 numers[r]=numers[allSeek-1]; //抽奖样本问题减一 allSeek--; }
2. 封装中访问器的实现
特别注意不应该编写返回引用可变对象的访问器方法。这样的话会破坏类的封装性。如下代码
class Person{ private Date birthday; public Date getBirthday(){ return this.birthday; } }
初步来看的话,是没有问题的,但如果使用以下代码来访问:
Person p1=new Person(); Date birthday=p1.getBirthday(); birthday.setTime(birthday.getTime()-1);
此时,在我们去访问p1.getBirthday()时就会发现当前的birthday发生了变化。这是因为birthday与p1.birthday引用的是同一个可变对象。所以,我们如果需要返回一个可变对象的引用 ,就应用首先对它进行克隆(clone),如下代码:
class Person{ private Date birthday; public Date getBirthday(){ return (Date)this.birthday.clone(); } }
3. 基于类的访问权限
我们知道,方法可以访问所调用对象的私有数据。一个方法可以访问所属类的所有对象的私有数据,这点有点拗口,看下面的代码。
class Person{ private String name; public boolean equals(Person otherPerson){ return this.name.equals(otherPerson.name); } }
代码没什么问题,我们在使用中的一种典型的使用形式如下:
if(zhangsan.equals(boss)){ //zhangsan's name is the same as boss }
我们类方法中的代码在访问zhangsan的私有域name时没有问题。但是,它还访问了boss的私有域,有点奇怪。但这是合法的,它的原因是boss是Person类的对象,而Person类的方法可以访问Person类的任何一个对象 的私有域。
4. Java程序设计语言总是采用值调用。
这也就是说,方法得到的是所有参数值 的一个拷贝,特别是,方法不能修改传递给它的任何参数变量的内容。如下:
对于第三点,我们用下面的代码来说明:
public class Person{ private String name; public Person(String name){ this.name=name; } public String getName(){ return this.name; } public static void swap ( Person p1, Person p2){ Person temp=p1; p1=p2; p2=temp; } public static void main(String[] args){ Person p1=new Person("zhangsan"); Person p2=new Person("lisi"); Person.swap(p1,p2); System.out.println(p1.getName()+","+p2.getName()); } }
上面代码的执行结果:
zhangsan,lisi
这是因为,在static方法中的参数p1与p2被初始化为两个对象引用的拷贝,这个方法交换的是两个拷贝。在方法执行完毕后,参数变量p1与p2都被垃圾回收器给回收了。原来的变量p1与p2仍然引用 的是方法调用之前引用的对象。
5. final的用法
通常,我们将final关键字应用于局部变量,实例变量及静态变量。在所有这些情况下,它们的全部定义都是:在创建这个变量之后,只能够为其赋值一次。此后,再也不能修改它的值了。
注意两点:
int counter=0; Date[] dates=new Date[100]; for(int i=0;i<dates.length;i++){ dates[i]=new Date(){ public int compareTo(Date other){ counter++; //Error return super.compareTo(other); } }; } Array.sort(dates); System.out.println(counter+" comparisons.");
由于我们很清楚地知道counter需要更新,所以不能将counter声明为final。由于Integer对象是不可变的,所以也不能用Integer代替它。所以,我们应该使用一个长度为1的数组,代码如下:
final int[] counter=new int[1]; Date[] dates=new Date[100]; for(int i=0;i<dates.length;i++){ dates[i]=new Date(){ public int compareTo(Date other){ counter[0]++; return super.compareTo(other); } }; } Array.sort(dates); System.out.println(counter[0]+" comparisons.");
在此,数组变量仍然被声明为final,但是这仅仅表示 不可以让它引用 另外一个数组。数组中的数据元素可以自由地更改。