您的位置:首页 > 编程语言 > Java开发

java基础知识总结7(枚举+泛型)

2016-03-29 10:43 477 查看

七、枚举

枚举是一个类类型,是JDK1.5的新特性

枚举的关键字是enum

Java中所有的枚举类都是java.lang.Enum的子类

注意:枚举类中可以包含成员有【字段(常量)、方法(构造方法、普通方法)】

枚举中不能有number;不能赋值————类

只有顺序可以比较———–

语法:

enum 枚举名{

枚举体


}

例:

enum Direction{//枚举体

EAST,SOUTH,WEST,NORTH//后面可以有分号也可以没有


}

调用:枚举名.成员

枚举名 name =枚举名.成员;

Direction dir = Direction.EAST;

Enum的方法使用:

Javap—————查看字节码文件

代码:

enum Direction{

EAST(‘e’),SOUTH(’s’),NORTH(‘n’),WEST(‘w’);

protected char a;


private Direction(char a){

this.a = a;

}

public String toString(){

return a+”“;

}

}

字节码:

———- javap ———-

Compiled from “Direction.java”

final class Direction extends java.lang.Enum {//不能被实例化

public static final Direction EAST;

public static final Direction SOUTH;

public static final Direction NORTH;

public static final Direction WEST;

protected char a;//默认是private,但可以使用4P

public static Direction[] values();//默认方法,用来遍历打印枚举成员

public static Direction valueOf(java.lang.String);

public java.lang.String toString();

static {};//静态初始化块

}

由上面可以知道枚举继承java.lang.Enum类,参数的默认修饰符public static final,在枚举中可以使用4p

public enum A{BLUE,RED,BLACK};//声明并定义一个枚举,初始化为。。。

在一个类中定义一个enum成员,enum默认是静态的————-相当于类变量

枚举的用法

一、常量

枚举出来之前都是通过接口

在JDK1.5 之前,我们定义常量都是: public static fianl,现在可以把相关的常量分组到一个枚举类型里,例:

public enum Color {

RED, GREEN, BLANK, YELLOW

}

二、Switch

JDK1.5之后的switch语句支持Byte,short,int,char,enum类型,使用枚举,能让我们的代码可读性更强,JDK1.7之后开始支持String类型,例:

enum Signal {

GREEN, YELLOW, RED


}

public class TrafficLight {

Signal color = Signal.RED;

public void change() {

switch (color) {

case RED:

color = Signal.GREEN;

break;

case YELLOW:

color = Signal.RED;

break;

case GREEN:

color = Signal.YELLOW;

break;

}

}


}

三、向枚举中添加新方法

要自定义方法,必须在enum实例序列的最后添加一个分号。而且 Java 要求必须先定义 enum实例,例:

public enum Color {

RED("红色", 1), GREEN("绿色", 2), BLANK("白色", 3), YELLO("黄色", 4);

// 成员变量

private String name;

private int index;

// 构造方法

private Color(String name, int index) {

this.name = name;

this.index = index;

}

// 普通方法

public static String getName(int index) {

for (Color c : Color.values()) {

if (c.getIndex() == index) {

return c.name;

}

}

return null;

}

// get set 方法

public String getName() {

return name;

}

public void setName(String name) {

this.name = name;

}

public int getIndex() {

return index;

}

public void setIndex(int index) {

this.index = index;

}


}

四、覆盖枚举的方法

例如覆盖toString()

public enum Color {

RED("红色", 1), GREEN("绿色", 2), BLANK("白色", 3), YELLO("黄色", 4);

// 成员变量

private String name;

private int index;

// 构造方法

private Color(String name, int index) {

this.name = name;

this.index = index;

}

//覆盖方法

@Override

public String toString() {

return this.index+"_"+this.name;

}


}

五、实现接口

所有的枚举都继承自java.lang.Enum类。由于Java 不支持多继承,所以枚举对象不能再继承其他类,例:

public interface Behaviour {

void print();

String getInfo();


}

public enum Color implements Behaviour{

RED("红色", 1), GREEN("绿色", 2), BLANK("白色", 3), YELLO("黄色", 4);

// 成员变量

private String name;

private int index;

// 构造方法

private Color(String name, int index) {

this.name = name;

this.index = index;

}


//接口方法

@Override

public String getInfo() {

return this.name;

}

//接口方法

@Override

public void print() {

System.out.println(this.index+":"+this.name);

}


}

六、使用接口组织枚举

public interface Food {

enum Coffee implements Food{

BLACK_COFFEE,DECAF_COFFEE,LATTE,CAPPUCCINO

}

enum Dessert implements Food{

FRUIT, CAKE, GELATO

}


}

七、枚举集合

java.util.EnumSet和java.util.EnumMap是两个枚举集合。EnumSet保证集合中的元素不重复;EnumMap中的key是enum类型,而value则可以是任意类型。

枚举和普通类的区别与联系;

1、枚举与类都可以实现多接口;访问控制符都可以使用(4p),但枚举中默认的是private,类中默认的是package;

2、枚举直接继承java.lang.Enum类,普通类是继承java.lang.Object;其中java.long.Enum类实现了java.long.Serializable和java.long.Comparable两个接口。

3、使用enum定义、非抽象的枚举默认修饰符为public final,因此枚举不能派生子类。

4、枚举的构造器只能使用private访问控制符,如果省略了枚举的访问修饰符其默认为private修饰;因为枚举的字段不能初始化,对象类型的必须调用构造方法,所有有多少个成员构造方法就会运行多少次;

5、枚举的所有实例必须在枚举的第一行显示列出,否则这个枚举永远都不能生产实例,列出这些实例时系统会自动添加public static final修饰,无需程序员显式添加

6、所有的枚举类都提供了一个values方法,该方法可以很方便的遍历所有的枚举值

7、关键字:枚举是enum,类是class

8、枚举是类类型,类是引用类型

instanceof关键字:

public class FX {

public static void main(String[] args) {

boolean b = (“1” instanceof String);//判断某个对象是不是某个类的实例 Result = Object instanceof Class/interface

//Result:布尔类型。

//Object:必选项。任意对象表达式。

//Class:必选项。任意已定义的对象类。

System.out.println(b);

}

}

注解—JDK1.5的新特性

使用位置:类上、方法上、语句上

八、协变与泛型

协变——-可以用在数组、重写,不可以用在枚举中

可变参数:

int… a————-代表一个没有长度限制的数组———-JDK1.5之后有的

输入的必须是数组

Java1.5增加了新特性:可变参数:适用于参数个数不确定,类型确定的情况,java把可变参数当做数组处理。注意:可变参数必须位于最后一项。只支持有一个可变参数。因为参数个数不定,所以当其后边还有相同类型参数时,java无法区分传入的参数属于前一个可变参数还是后边的参数,所以只能让可变参数位于最后一项。

可变参数的特点:

(1)、只能出现在参数列表的最后;

(2)、… 位于变量类型和变量名之间,前后有无空格都可以;

(3)、调用可变参数的方法时,编译器为该可变参数隐含创建一个数组,在方法体中以数组的形式访问可变参数。例:

public class Varable {

public static void main(String [] args){

System.out.println(add(2,3));

System.out.println(add(2,3,5));

}

public static int add(int x,int …args){

int sum=x;

for(int i=0;i

协变的作用范围

1、数组:如果 A 是 B 的超类,则 A[] 也是 B[] 的超类,所有需要 A[] 的地方都可以用 B[] 代替,例:

public class Test2 {

public static void main(String[] args){

Integer[] ints=new Integer[1];

ints[0]=99;

show(ints);

}

static void show(Number[] ns){

System.out.println(Arrays.toString(ns));

}


}

2、重写

public class Test2 {

public static void main(String[] args){

P p=new C();

System.out.println(p.get());

}


}

class P{

public Number get(){

return new Integer(0);

}


}

class C extends P{

public Integer get(){

return new Integer(1);

}


}

3、泛型不能协变:如果 A 是 B 的超类,则 List 和 List 无关,需要 List 的地方不可以用 List[b] 代替,例:[/b]

public class Test2 {

public static void main(String[] args) {

List<Integer> ints = new ArrayList<Integer>();

//incompatible type--类型不兼容

show(ints);

}

static void show(List<Number> ns) {

System.out.println(ns);

}


}

泛型(Generic)的本质是类型参数化

————-是JDK1.5提供的新技术,它的本质是类型参数化,类似C++中的模板,它根据应用主要分为泛型方法,泛型接口,泛型类,协变不适用于泛型

泛型————-泛型类class Demo{},泛型接口interface Inter{},泛型方法public T show(T a) {}

泛型的好处是在编译的时候检查类型安全,并且所有的强制转换都是自动和隐式的,提高代码的重用率

例1:

public class GenericDemo{

public static void main(String[] args) {

C c1 = new C(new A("aaa"));

C c2 = new C(new B(3.4));//包装成double---自动装箱

Object obj = c1.getObj();

A obj = (A)(c1.getObj());//强制转换

System.out.println(obj);

}


}

class A{

private String x;

public A(String x){

this.x=x;

}

public String getX(){

return x;

}


}

class B{

private Double d;

public B(Double d){

this.d=d ;

}

public Double getD(){

return d;

}


}

class C{

private Object obj;

public C(Object obj){

this.obj=obj ;

}

public Object  getObj(){

return obj;

}


}

泛型的使用

一、泛型类:

public class Demo1 {

public static void main(String[] args) {

Generic<String> c = new Generic<String>("asad");

System.out.println(c.getA());

//Generic<int> c1 = new Generic<int>(12);必须要是包装类

}


}

class Generic{//泛型类

private T a;

public Generic(T a){

this.a = a;

}

public T getA(){

return a;

}


}

二、泛型方法

public static <T> T getX(T t){//泛型方法----------<  >--表声明
return t;

}


三、泛型接口

public class GenericDemo3{

public static void main(String[] args) {

D < int[][]> d = new D< int[][]>();

d.show(new int[][]{{1,4},{2,3}});

}

}

interface IA{

void show(T t);

}

class D implements IA{

public void show(T t){

System.out.println(t.getClass().getName());

}

}

有界类型:

上界:extends 默认上界为Object

Class G < T extends Number,K extends Object,H super Integer>{}

下界:super———-只有通配符有下限

extends关键字声明了类型的上界,表示参数化的类型可能是所指定的类型,或者是此类型的子类

super关键字声明了类型的下界,表示参数化的类型可能是所指定的类型,或者是此类型的父类型,直至Object

规则和限制

1、泛型的类型参数只能是类类型(包括自定义类),不能是简单类型。

2、同一种泛型可以对应多个版本(因为参数类型是不确定的),不同版本的泛型类实例是不兼容的。

3、泛型的类型参数可以有多个。< T,K>

4、泛型的参数类型可以使用extends关键字,例如< T extends superclass>。习惯上称为“有界类型”。

5、泛型的参数类型还可以是通配符类型。例如Class< ? > classType = Class.forName(“java.lang.String”)

6、如果只指定了< ? >,而没有extends,则默认是允许Object及其下的任何Java类了。也就是任意类。

7、通配符泛型不但可以向下限制,如< ? extends Collection >,还可以向上限制,如< ? super Double >,表示类型只能接受Double及其上层父类类型,如Number、Object类型的实例。

8、泛型类定义可以有多个泛型参数,中间用逗号隔开,还可以定义泛型接口,泛型方法。这些都与泛型类中泛型的使用规则类似。

一个普通static方法,无法访问泛型类的类型参数,所以,若普通static方法需要使用泛型参数,必须使其成为泛型方法。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: