Java Puzzlers(6)类库之迷

剩下两章都是讨论java的平台类库的。呵呵,作者对JDK一些写的很差的类库批评的毫不客气,特别是Calendar,Date的处理方式。确实如此,我在处理日期时总是要特别小心,不小心就得出错。

 

1。首先谈谈不变类,在java里,基本类型的外覆类,包括String,BigDecimal,Integer,Double,Short,Byte,Boolean,Float,Long,BigInteger都是所谓的不变类。他们产生的实例都是不可变的。这些类中的任何方法调用都是产生一个新对象返回。最近CSDN上的热贴《华为面试题》的第一题变相考了此点。只不过那道题中并没有调用String的任何方法,而只是把引用的副本指向了另一个字符串。看看题目:

public class Test {
    public static void changeStr(String str){
        str="welcome";
    }
    public static void main(String[] args) {
        String str="1234";
        changeStr(str);
        System.out.println(str);
    }
}

输出为:1234。此题中,方法changeStr中的变量str其实只是传进来的引用str的副本,指向了另一个字符串,作用范围仅仅在方法内"welcome",原对象并没有受到影响,如果我们在方法中调用String的方法,如:

   public static void changeStr(String str){
        str.subString(0,2);

   }结果输出也仍然为:1234.因为subString方法返回的是一个新的对象,与原来的str没有任何关系。具体可以看看这些外覆类的源代码。记住:这些类都是唯读类,不可变类。

 

2。另外,讨论了equals()方法和hashCode()方法的覆写问题。几条原则:

A。无论何时,只要你覆写了equals()方法,你就必须同时覆写hashCode方法

B。覆写此两个方法时,特别注意避免无意识的重载行为,比如下面的写法:

public boolean equals(MyClass object)

你应该机械地对你想要覆写的每一个超类方法都拷贝其声明。

 

3。需要注意的JDK1.4,JDK5的新类:

A。LinkedHashSet。它维护插入的元素的顺序,并不允许重复元素,1.4新引进的。

B。String类的split方法,利用正则表达式分割字符串返回数组。

C。Arrays类的几个新方法,如toString,deepToString。

D。JDK5提供了一整套的位操作方法,整型类型的外覆类(Integer,Long,Short,Byte,Char)现在都支持通用的位操作,包括highhestOneBit,lowestOneBit,bitCount等,具体请见JDK文档

 

4。Date和Calendar类,Date类的大部分方法已经被声明不建议使用,而Calendar的也有一些特性需要注意,如用0来表示一月。(最可恶的特性,为何不跟日常习惯相同),而Date类的getDay()方法返回的是星期的日期,而不是月份的日期,你应该用Calendar.DAY_OF_MONTH代替。反正,写日期程序时,要注意查文档!!

 

5。Math.abs()方法,一定返回参数的绝对值吗??有个特例,请看下面的例子:

 System.out.println(Math.abs(Integer.MIN_VALUE));

此句将打印一个负数。Math.abs方法不能保证一定返回非负数的结果。当传入的参数是Integer.MIN_VALUE或者Long.MIN_VALUE时将直接返回此参数。过去我没注意到这点。

 

6。另外一个教训是关于排序的,简单的比较器:

 Comparator<Integer> cmp = new Comparator<Integer>() {
            public int compare(Integer i1, Integer i2) {
                return i2 - i1;
            }
        };

这样的比较器是不安全的。因为两个整数相减完全有可能溢出!比如

public class Overflow{

      public static void main(String args[]){

         int x=-2000000000;

         int y=2000000000;

        System.out.println(x-y);

     }

}

将输出294967296,而不是一个负数。教训:不要使用基于减法的比较器,除非你能够确定所要比较的数值之间的差永远不会大于Integer.MAX_VALUE。我们可以更改下这个比较器:

 Comparator<Integer> cmp = new Comparator<Integer>() {
            public int compare(Integer i1, Integer i2) {
                return (i2<i1?-1:(i2==i1?0:1));
            }
        };

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值