JDK1.5的新特性
相关阅读
0、前言
"JDK1.5"(开发代号猛虎)的一个重要主题就是通过新增一些特性来简化开发,这些特性包括泛型,for-each 循环,自动装包/拆包,枚举,可变参数, 静态导入 。使用这些特性有助于我们编写更加清晰,精悍,安全的代码。
1.泛型(Generic)
C++通过模板技术可以指定集合的元素类型,而Java在1.5之前一直没有相对应的功能。一个集合可以放任何类型的对象,相应地从集合里面拿对象的时候我们也不得不对他们进行强制得类型转换。猛虎引入了泛型,它允许指定集合里元素的类型,这样你可以得到强类型在编译时刻进行类型检查的好处。
Collection<String> c = new ArrayList(); c.add(new Date());
编译器会给出一个错误:
add(java.lang.String) in java.util.Collection<java.lang.String> cannot be applied to (java.util.Date)
优点:
1、通过引入泛型,我们将获得编译时类型的安全和运行时更小地抛出ClassCastExceptions的可能。如果你试图插入非string类型的值,你将在编译时发现并且修正这类问题。没有泛型,你会发现这样一个bug,当你的客户调用后会告诉你,你所编写的程序抛出ClassCastException异常而崩溃。
2、当你从集合中得到一个元素时你无需进行强制转换。
2.For-Each循环
For-Each循环得加入简化了集合的遍历。假设我们要遍历一个集合对其中的元素进行一些处理。典型的代码为:
void processAll(Collection c){ for(Iterator i=c.iterator(); i.hasNext();){ MyClass myObject = (MyClass)i.next(); myObject.process(); } }
使用For-Each循环,我们可以把代码改写成:
void processAll(Collection<MyClass> c){ for (MyClass myObject : c) myObject.process(); }
优点:简化了集合的遍历,并且避免了强制类型转换。
3.自动装箱/拆箱(Autoboxing/unboxing)
自动装包:基本类型自动转为包装类.(int >> Integer)
自动拆包:包装类自动转为基本类型.(Integer >> int)
在JDK1.5之前,我们总是对集合不能存放基本类型而耿耿于怀,现在自动转换机制解决了我们的问题。
int a = 3; Collection c = new ArrayList(); c.add(a);//自动转换成Integer. Integer b = new Integer(2); c.add(b + 2); //这里Integer先自动转换为int进行加法运算,然后int再次转换为Integer.
优点:自动装包/拆包大大方便了基本类型数据和它们包装类地使用。
4.枚举(Enums)
JDK1.5加入了一个全新类型的“类”-枚举类型。为此JDK1.5引入了一个新关键字enmu. 我们可以这样来定义一个枚举类型。
public enum Color { Red, White, Blue }
然后可以这样来使用Color myColor = Color.Red.
也可以对枚举进行遍历。枚举类型还提供了两个有用的静态方法values()和valueOf(). 我们可以很方便地使用它们,例如
for (Color c : Color.values()) System.out.println(c);
优点:
- 他们提供编译时类型安全。
- 他们都是对象,因此你不需要将他们放入集合中。
- 他们作为一种类的实现,因此你可以添加一些方法。
- 他们为枚举类型提供了合适的命名空间。
一个应用例子,页面下拉框使用枚举(key-value 英文-中文):
1 public enum Status { 2 CREATE("创建") 3 ,WAIT("等待") 4 ,VALID("生效") 5 ,FAIL("失败") 6 ,CHANGE("变更") 7 ,REMOVE("解除"); 8 9 //常用状态 10 public static Status[] listQuery(){ 11 return new Status[]{ 12 WAIT 13 ,VALID 14 ,FAIL 15 ,CHANGE 16 ,REMOVE 17 }; 18 } 19 20 private String value; 21 22 Status( String value ){ 23 this.value = value; 24 } 25 26 public String getValue() { 27 return value; 28 } 29 30 public String toString(){ 31 return this.value; 32 } 33 }
1 <s:select name="dispatch.status" list="@com.ylink.project.ecps.dispatch.entity.Status@listQuery()" listKey="name()" headerKey="" headerValue="" value="dispatch.status.name()" />
5.可变参数(Varargs)
可变参数使程序员可以声明一个接受可变数目参数的方法。注意,可变参数必须是方法声明中的最后一个参数。假设我们要写一个简单的方法打印一些对象,
util.write(obj1);
util.write(obj1,obj2);
util.write(obj1,obj2,obj3);
…
在JDK1.5之前,我们可以用重载来实现,但是这样就需要写很多的重载方法,显得不是很有效。如果使用可变参数的话我们只需要一个方法就行了
public void write(Object... objs) { for (Object obj: objs) System.out.println(obj); }
在引入可变参数以后,Java的反射包也更加方便使用了。对于c.getMethod("test", new Object[0]).invoke(c.newInstance(), new Object[0])),
现在我们可以这样写了c.getMethod("test").invoke(c.newInstance()),这样的代码比原来清楚了很多。
6.静态导入(Static Imports)
要使用用静态成员(方法和变量)我们必须给出提供这个方法的类。使用静态导入可以使被导入类的所有静态变量和静态方法在当前类直接可见,使用这些静态成员无需再给出他们的类名。
可以使用通配符(*)代表导入该类的所有静态方法和静态变量。不允许静态方法和静态变量出现重名的情况。
import static java.lang.Math.max; //导入了Math的max方法 import static java.lang.Integer.*; //导入了Integer的所有静态方法和静态属性 public class StaticImport { public static void main(String[] args){ System.out.println(max(1,2));//通过静态导入使用Math的静态方法 System.out.println(parseInt("123"));//通过静态导入使用Integer的静态方法 System.out.println(MAX_VALUE);//通过静态导入使用Integer的静态属性 } }
优点:使用静态导入,无需再写Math.max(1,2); 。
缺点:不过,过度使用这个特性也会一定程度上降低代码地可读性。