Java程序员从笨鸟到菜鸟之(十)枚举,泛型详解

转载 2012年03月21日 13:34:25

本文来自:曹胜欢博客专栏。转载请注明出处:http://blog.csdn.net/csh624366188

 

一:首先从枚举开始说起

枚举类型是JDK5.0的新特征。Sun引进了一个全新的关键字enum来定义一个枚举类。下面就是一个典型枚举类型的定义:
public enum Color{
RED
BLUEBLACKYELLOWGREEN
}
   显然,
enum很像特殊的class,实际上enum声明定义的类型就是一个类。 而这些类都是类库中Enum类的子类(java.lang.Enum)。它们继承了这个Enum中的许多有用的方法。我们对代码编译之后发现,编译器将 enum类型单独编译成了一个字节码文件:Color.class
Color
字节码代码
final enum hr.test.Color {
// 
所有的枚举值都是类静态常量
public static final enum hr.test.Color RED;
public static final enum hr.test.Color BLUE;
public static final enum hr.test.Color BLACK;
public static final enum hr.test.Color YELLOW;
public static final enum hr.test.Color GREEN;
private static final synthetic hr.test.Color
[] ENUM$VALUES;

}
下面我们就详细介绍
enum定义的枚举类的特征及其用法。(后面均用Color举例)
1
Color枚举类就是class,而且是一个不可以被继承的final类。其枚举值(REDBLUE..)都是Color类型的类静态常量, 我们可以通过下面的方式来得到Color枚举类的一个实例:
Color c=Color.RED;
注意:这些枚举值都是
public static final的,也就是我们经常所定义的常量方式,因此枚举类中的枚举值最好全部大写。
2
、即然枚举类是class,当然在枚举类型中有构造器,方法和数据域。但是,枚举类的构造器有很大的不同:
1) 构造器只是在构造枚举值的时候被调用。
Java
代码
enum Color{
RED
25500),BLUE00255),BLACK000),YELLOW2552550),GREEN02550;
//
构造枚举值,比如RED25500
private Color
int rvint gvint bv{
this.redValue=rv;
this.greenValue=gv;
this.blueValue=bv;
}
public String toString
(){ //覆盖了父类EnumtoString()
return super.toString
()+“”+redValue+“”+greenValue+“”+blueValue+“”;
}
private int redValue; //
自定义数据域,private为了封装。
private int greenValue;
private int blueValue;
}
  (
2) 构造器只能私有private,绝对不允许有public构造器。 这样可以保证外部代码无法新构造枚举类的实例。这也是完全符合情理的,因为我们知道枚举值是public static final的常量而已。 但枚举类的方法和数据域可以允许外部访问。
Java
代码
public static void main
String args[])
{
// Color colors=new Color
100200300; //wrong
Color color=Color.RED;
System.out.println
color; // 调用了toString()方法
}
3
、所有枚举类都继承了Enum的方法,下面我们详细介绍这些方法。
1) ordinal()方法: 返回枚举值在枚举类种的顺序。这个顺序根据枚举值声明的顺序而定。
Color.RED.ordinal
(); //返回结果:0
Color.BLUE.ordinal
(); //返回结果:1
2) compareTo()方法: Enum实现了java.lang.Comparable接口,因此可以比较象与指定对象的顺序。Enum中的compareTo返回的是两个枚举值的顺 序之差。当然,前提是两个枚举值必须属于同一个枚举类,否则会抛出ClassCastException()异常。(具体可见源代码)
Color.RED.compareTo
Color.BLUE; //返回结果 -1
3) values()方法: 静态方法,返回一个包含全部枚举值的数组。
Color
[] colors=Color.values();
for
Color c:colors{
System.out.print
c+“;
}//
返回结果:REDBLUEBLACK YELLOWGREEN
4) toString()方法: 返回枚举常量的名称。
Color c=Color.RED;
System.out.println
c;//返回结果: RED
5) valueOf()方法: 这个方法和toString方法是相对应的,返回带指定名称的指定枚举类型的枚举常量。
Color.valueOf
“BLUE”; //返回结果: Color.BLUE
6) equals()方法: 比较两个枚举类对象的引用。
Java
代码
//JDK
源代码:
public final boolean equals
Object other) {
return this==other;
}
4
、枚举类可以在switch语句中使用。
Java
代码
Color color=Color.RED;
switch
color{
case RED
: System.out.println“it‘s red”;break;
case BLUE
: System.out.println“it’s blue”;break;
case BLACK
: System.out.println“it‘s blue”;break;
}

二:然后看泛型

泛型(Generic type 或者generics)是对 Java 语言的类型系统的一种扩展,以支持创建可以按类型进行参数化的类。可以把类型参数看作是使用参数化类型时指定的类型的一个占位符,就像方法的形式参数是运行时传递的值的占位符一样。 

1.泛型的好处:

1类型安全。泛型的主要目标是提高 Java 程序的类型安全。通过知道使用泛型定义的变量的类型限制,编译器可以在一个高得多的程度上验证类型假设。没有泛型,这些假设就只存在于程序员的头脑中(或者如果幸运的话,还存在于代码注释中)。 

2·消除强制类型转换。泛型的一个附带好处是,消除源代码中的许多强制类型转换。这使得代码更加可读,并且减少了出错机会。 尽管减少强制类型转换可以降低使用泛型类的代码的罗嗦程度,但是声明泛型变量会带来相应的罗嗦

3· 潜在的性能收益。泛型为较大的优化带来可能。在泛型的初始实现中,编译器将强制类型转换(没有泛型的话,程序员会指定这些强制类型转换)插入生成的字节码中。但是更多类型信息可用于编译器这一事实,为未来版本的JVM 的优化带来可能。 

2.类型参数:
在定义泛型类或声明泛型类的变量时,使用尖括号来指定形式类型参数。形式类型参数与实际类型参数之间的关系类似于形式方法参数与实际方法参数之间的关系,只是类型参数表示类型,而不是表示值。 
泛型类中的类型参数几乎可以用于任何可以使用类名的地方。例如,下面是
java.util.Map接口的定义的摘录: 
public interface Map<K, V> { 
public void put(K key, V value); 
public V get(K key); 

3.泛型不是协变的 
关于泛型的混淆,一个常见的来源就是假设它们像数组一样是协变的。其实它们不是协变的。
List<Object>不是List<String>的父类型。 
如果 
扩展 B,那么 的数组也是 的数组,并且完全可以在需要B[]的地方使用A[]: 
Integer[] intArray = new Integer[10]; 
Number[] numberArray = intArray; 
上面的代码是有效的,因为一个
Integer是一个Number,因而一个Integer数组是一个Number数组。但是对于泛型来说则不然。下面的代码是无效
List<Integer> intList = new ArrayList<Integer>(); 
List<Number> numberList = intList; // invalid 

4.泛型中的类型通配符 

假设您具有该方法: 
void printList(List l) { 
for (Object o : l) 
System.out.println(o); 

上面的代码在 
JDK 5.0 上编译通过,但是如果试图用List<Integer>调用它,则会得到警告。出现警告是因为,您将泛型 (List<Integer>)传递给一个只承诺将它当作List(所谓的原始类型)的方法,这将破坏使用泛型的类型安全。 
如果试图编写像下面这样的方法,那么将会怎么样? 
void printList(List<Object> l) { 
for (Object o : l) 
System.out.println(o); 

它仍然不会通过编译,因为一个
List<Integer>不是一个List<Object>(正如前一屏泛型不是协变的 中所学的)。这才真正烦人——现在您的泛型版本还没有普通的非泛型版本有用! 解决方案是使用类型通配符: 
void printList(List<?> l) { 
for (Object o : l) 
System.out.println(o); 


上面代码中的问号是一个类型通配符。它读作
问号List<?>是任何泛型List的父类型,所以您完全可以将 List<Object>List<Integer>或 List<List<List<Flutzpah>>>传递给printList() 

5.泛型方法 
(在类型参数 一节中)您已经看到,通过在类的定义中添加一个形式类型参数列表,可以将类泛型化。方法也可以被泛型化,不管它们定义在其中的类是不是泛型化的。 
泛型类在多个方法签名间实施类型约束。在
List<V>中,类型参数V出现在get()add()contains()等方法的签名中。 当创建一个Map<K, V>类型的变量时,您就在方法之间宣称一个类型约束。您传递给add()的值将与get()返回的值的类型相同。 
类似地,之所以声明泛型方法,一般是因为您想要在该方法的多个参数之间宣称一个类型约束。例如,下面代码中的
ifThenElse()方法,根据它的第一个参数的布尔值,它将返回第二个或第三个参数: 

public <T> T ifThenElse(boolean b, T first, T second) { 
return b ? first : second; 

为什么您选择使用泛型方法,而不是将类型T添加到类定义呢?(至少)有两种情况应该这样做: 
当泛型方法是静态的时,这种情况下不能使用类类型参数。 
当 上的类型约束对于方法真正是局部的时,这意味着没有在相同类的另一个 方法签名中使用相同 类型 的约束。通过使得泛型方法的类型参数对于方法是局部的,可以简化封闭类型的签名。 
有限制类型 
在前一屏泛型方法 的例子中,类型参数
V是无约束的或无限制的类型。有时在还没有完全指定类型参数时,需要对类型参数指定附加的约束。 
考虑例子
Matrix类,它使用类型参数V,该参数由Number类来限制: 
public class Matrix<V extends Number> { ... } 
编译器允许您创建
Matrix<Integer>Matrix<Float>类型的变量,但是如果您试图定义 Matrix<String>类型的变量,则会出现错误。类型参数V被判断为由Number限制。在没有类型限制时,假设类型参数由 Object限制。这就是为什么前一屏泛型方法 中的例子,允许List.get()List<?>上调用时返回Object,即使编译器不知道类型参数V的类型。

《Java程序员由笨鸟到菜鸟》电子版书正式发布,欢迎大家下载

在众多朋友的支持和鼓励下,《Java程序员由菜鸟到笨鸟》电子版终于和大家见面了。本电子书涵盖了从java基础到javaweb开放框架的大部分内容。在编写的过程中,难免会出现一些错误,希望大家能多多提些...
  • csh624366188
  • csh624366188
  • 2012年09月20日 11:01
  • 51632

Java程序员从笨鸟到菜鸟之(十)枚举,泛型详解

 转载自:曹胜欢http://blog.csdn.net/csh624366188/article/details/7314991 一:首先从枚举开始说起 枚举类型是JDK5.0的新特征。S...
  • freeideas
  • freeideas
  • 2015年03月12日 17:21
  • 157

程序员怎么在短时间内从菜鸟到高级架构师

程序员怎么在短时间内从菜鸟到高级架构师,(向贴吧大神致敬,个人觉得里面我看过的书籍挺好的,献给新手)说道面试及笔试题,一般不外乎Java语言基础、Java语言高级、UML和OO和模式、数据库、测试、数...
  • u014242422
  • u014242422
  • 2016年04月24日 23:10
  • 2076

Java程序员从笨鸟到菜鸟之(六十四)细谈Hibernate(十五)HQL与QBC查询方式详解

首先来看一下,hibernate提供的几种检索方式: 1.导航对象图检索方式 :根据已经加载的对象,导航到其他对象。例如,对于已经加载的Customer对象,调用它的getOrders().iter...
  • csh624366188
  • csh624366188
  • 2012年05月31日 13:42
  • 8975

Java程序员从笨鸟到菜鸟之(五十二)细谈Hibernate(三)Hibernate常用API详解及源码分析

新接触一个框架的目的就是想利用这个框架来为我们做一些工作,或者是让他来简化我们的工作,利用这个框架无非就是要利用这个框架所给我们提供的API去操作我们的数据,所以利用一个框架的好坏很大一部分取决于你对...
  • csh624366188
  • csh624366188
  • 2012年05月17日 09:36
  • 17724

java程序员从笨鸟到菜鸟之(六)继承详解

一. 代码块 概念:  使用{} 括起来的代码,称为代码块 分类: 根据它位置和声明的不同,我们可以将代码块分为局部代码块,构造代码块,静态代码块,同步代码块(多线程涉及) a. 局部代码...
  • wzj_110
  • wzj_110
  • 2017年10月16日 14:05
  • 56

java基础-泛型 枚举

一.泛型泛型提供了编译时类型安全检测机制,该机制允许程序员在编译时检测到非法的类型。1.泛型方法在调用时可以接收不同类型的参数。根据传递给泛型方法的参数类型,编译器适当地处理每一个方法调用。 下面是...
  • u012426959
  • u012426959
  • 2017年12月08日 09:11
  • 131

Java程序员从菜鸟到笨鸟

Java程序员从菜鸟到笨鸟      Java作为当前的主流开发语言,拥有越来越多的学习者,本书主要面向Java初学者,对SSH框架做了深入详细的探讨...更多...
  • lihang421
  • lihang421
  • 2013年04月17日 10:15
  • 471

JAVA程序员从笨鸟到菜鸟

http://blog.csdn.net/csh624366188/article/list/2?viewmode=contents
  • michael493439861
  • michael493439861
  • 2012年10月27日 14:22
  • 227

Java菜鸟到大牛学习路线培训教程

这是一套Java菜鸟到大牛学习路线培训教程,由工作了10年的资深Java架构师整理。主要分5个阶段:Java程序员->Java初级软件工程师->Java中级软件工程师->Java高级软件工程师->Ja...
  • qq_38472891
  • qq_38472891
  • 2017年05月23日 12:41
  • 1068
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Java程序员从笨鸟到菜鸟之(十)枚举,泛型详解
举报原因:
原因补充:

(最多只允许输入30个字)