黑马程序员:Java培训、Android培训、iOS培训、.Net培训

JAVA范型-基础

一、泛型的概念

1、实现了参数化类型

2、用于编写可应用于多种类型的代码,即所编写的代码可应用于许多许多的类型。

3、范型容器、范型接口、范型方法都是经典的用法。

二、泛型与多态

1、多态是一种泛化机制。在使用类型说明的地方,使用多态确实具备一定的灵活性。但,多态是受限制的:只能接受基类或其子类(拘泥于单继承体系,使程序受限太多)。而且,在实际编写代码时,只能使用已存在的基类或接口。同时,一旦指明了接口,程序就会要求你的代码必须使用特定接口(方法)。

2、泛型:使我们能够编写比多态更具灵活性、更加通用的代码,它使代码能够适用于“某种不具体类型”,而不是一个具体接口或类。

3、使用泛型机制编写的代码要比那些杂乱地使用Object变量,然后进行强制类型转换具有更好的安全性和可读性。

4、泛型另一个优点:能够在编译时检测错误,而不是在运行时。例如:

JDK1.5前 : public interface Comparable{public int comparaTo(Object 0)}

//编译通过、运行错误

Comparalbe c = new Date();   //Date实现了Comparalbe

c.compareTo(“red”);

JDK1.5:public interface Comparable<T>{public int comparaTo(T 0)}

//编译错误,因为传递给compareTo()的参数必须为Date类型

Comparalbe<Date> c = new Date();   //Date实现了Comparalbe

c.compareTo(“red”);

三、泛型的限制

1、基本类型不能使用作为类型参数。例如:

Comparalbe<int>是错误的,应该是Comparalbe<Integer>。

2、不能使用泛型类型创建实例。例如:

E object = new E();

3、不能使用泛型类型创建数组。例如:

E[] elements = new E[10];

(可以创建一个Object数组,然后将它的类型转换为E[]来规避这个限制)

(E[] elements = (E[]) new Object[10];)

(有编译警告)

4、不允许使用范型类创建泛型数组。例如:

ArrayList<String>[] list = new ArrayList<String>[10];

(可以使用:ArrayList<String>[] list = (ArrayList<String>[])new ArrayList [10];)

(有编译警告)

5、在静态环境下(静态域、静态方法、静态块)不允许类的参数是泛型类型。

public class test<E>{    //类的参数是泛型

private static E o;   //静态域

public static void m(E 0){} //静态方法

static{ }  //静态块

}

6、异常类不能是泛型的。例如:

public class MyException<T> extends Exception{

}

四、泛型方法(参数化方法)

1、泛型方法可以在泛型类中,也以在非泛型类中。即,是否拥有泛型方法,与其是否是泛型类没有关系。

2、泛型方法使得该方法能够独立于类而产生变化。如果使用泛型方法可以取代将整个类泛型化,那么就应该只使用泛型方法。

3、泛型方法定义:只需要将泛型参数列表置于返回类型之前。例如:

private <T> void m(T t){}

public static <T> T m(){return t;} //T t

4、泛型方法调用:与普通方法的调用一样。

五、泛型接口和类

1、泛型接口和类的定义:用尖括号括住类型参数,放在接口(类名)后面。例如:

public class Holder<T>{}

public interface Inter<T>{}

2、读作:具有T类型参数的类。

3、泛型接口和类的经典用法:泛型容器。

六、边界(泛型参数类型限定):<T extends BoundingTypeList>

1、T:为界限列表BoundingTypeList的的子类或子接口

2、BoundingTypeList:界限列表至多只能有一个类,或多个接口,并且类必须放在界限列表的第一位。界限列表使用&连接,例如,

BoundingTypeList:class & interfaceA & interfaceB

注:其中,class称为第一边界,interfaceA为第二边界,以此类推。如果没有界限列表,Object为第一边界。

3、泛型类型限定,也称为泛型类型的边界。它使得我们可以在用于泛型的参数类型上设置限制条件,规定了泛型可以应用的类型(这些泛型类型应该满足限制条件)。另一个更重要的效果就是可以按照自己的的边界类型来调用方法,而无界泛型参数调用的方法只是那些可用Object调用的方法(在下一节详解)。

七、泛型翻译:擦除

1、擦除的神秘之处

1)ArrayList<String>和ArrayList<Integer>

//main()

Class c1 = new ArrayList<String>().getClass();

Class c2 = new ArrayList<Integer>().getClass();

System.out.println(c1 == c2);

//output

ture

显然,我们都认为ArrayList<String>和ArrayList<Integer>是不同的类型。然而,程序运行的结果显示:ArrayList<String>和ArrayList<Integer>是同一类型。这是因为:在泛型代码内部,无法获得任何有关泛型参数类型的信息(这也是泛型的使用受限的原因之一,例如,不能创建泛型数组)。

2)擦除带来的问题:在使用泛型时,任何具体类型信息都被擦除,而我们只知道在使用一个原生对象(Object)(如果没有使用边界的话),因此我们只能调用的方法只是那些可用Object调用的方法。

class HasF{ public void f(){System.out.println(“HasF”);}}

class Temp<T>{

T obj;

public Temp(T aObj){obj = aObj;}

public void meth(){obj.f();} //compile error

}

//main()

new Temp<HasF>(new HasF()).meth();

这是因为编译器无法将meth()必须能够在obj上调用f()这一需求映射到HasF拥有f()这一事实。为了解决这一问题,我们就需要到泛型边界(泛型类型参数将擦除到它的第一边界,而没有边界则擦除到原生Object)。

class Temp<T extends HasF>{    //边界HasF,可调用所有HasF暴露的方法

T obj;

public Temp(T aObj){obj = aObj;}

public void meth(){obj.f();} //

}

2、泛型类

无论何时定义一个泛型类型,相应的原始类型(raw type)都会被自动提供。原始类型的名字就是删去了类型参数的泛型类型的名字。类型参数T会被擦除(erased),并用第一边界(无限定类型的变量用Object)替换。例如:

//泛型                               //原始类型(类型参数擦除)

public class Pair<T>{                   public class Pair{

private T frist;                         private Object frist;

private T second;                       private Object second;

public T getFrist(){ retrun frist;}           public Object getFrist(){ retrun frist;}

public void setSecond(T aSecond){…..}     public void

setSecond(Object aSecond){…..}

}                                   }

3、泛型方法

1)也会进行类型擦除,泛型方法的类型参数T会被擦除(erased),并用第一边

界(无限定类型的变量用Object)替换。例如:

public <T extends Comparable> T m(){…….}  //泛型方法

public Comparable m(){…….}              //擦除后的方法

2)泛型方法擦除带来的问题:与多态冲突

//擦除前

class DateInterval extends Pair<Date>{

public void setSecond(Date aSecond){…..}

}

//擦除后

class DateInterval extends Pair{

public void setSecond(Date aSecond){…..}

}

//与此同时,DateInterval从Pair中继承方法

public void setSecond(Object aSecond){…..}

//显然,DateInterval有两个setSecond()方法,若执行下面语句

DateInterval interval = new DateInterval();

Pair<Date> pair = interval;

pair.setSecond(aDate);

// 对setSecond()的调用是多态的,pair引用了DateInterva对象,我们希望调用

// DateInterval. setSecond()。问题在于类型擦除与多态发生了冲突。解决办法:

//编译器在DateInterval中生成一个桥方法。

public void setSecond(Object aSecond){setSecond((Date)second)}

3)泛型表达式:返回类型被擦除,编译器插入强制类型转换

Pair<Employee> buddies = …..;

Employee buddy = buddies.getFrist();

//对getFrist()的调用是对原始方法getFrist()的调用,然后将返回的Object强制转//换为Employee

黑马----JAVA泛型基础的更多相关文章

  1. 一个小栗子聊聊JAVA泛型基础

    背景 周五本该是愉快的,可是今天花了一个早上查问题,为什么要花一个早上?我把原因总结为两点: 日志信息严重丢失,茫茫代码毫无头绪. 对泛型的认识不够,导致代码出现了BUG. 第一个原因可以通过以后编码 ...

  2. Java:泛型基础

    泛型 引入泛型 传统编写的限制: 在Java中一般的类和方法,只能使用具体的类型,要么是基本数据类型,要么是自定义类型.如果要编写可以应用于多种类型的代码,这种刻板的限制就会束缚很多! 解决这种限制的 ...

  3. java泛型基础

    泛型是Java SE 1.5的新特性, 泛型的本质是参数化类型, 也就是说所操作的数据类型被指定为一个参数. 这种参数类型可以用在类.接口和方法的创建中, 分别称为泛型类.泛型接口.泛型方法.  Ja ...

  4. java 泛型基础问题汇总

    泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数.这种参数类型可以用在类.接口和方法的创建中,分别称为泛型类.泛型接口.泛型方法. Java语言引 ...

  5. java泛型基础、子类泛型不能转换成父类泛型

    参考http://how2j.cn/k/generic/generic-generic/373.html 1.使用泛型的好处:泛型的用法是在容器后面添加<Type>Type可以是类,抽象类 ...

  6. java泛型基础、子类泛型不能转换成父类泛型--未完待续

    参考http://how2j.cn/k/generic/generic-generic/373.html 1.使用泛型的好处:泛型的用法是在容器后面添加<Type>Type可以是类,抽象类 ...

  7. 使用java泛型设计通用方法

    泛型是Java SE 1.5的新特性, 泛型的本质是参数化类型, 也就是说所操作的数据类型被指定为一个参数. 因此我们可以利用泛型和反射来设计一些通用方法. 现在有2张表, 一张user表和一张stu ...

  8. Java基础教程:泛型基础

    Java基础教程:泛型基础 引入泛型 传统编写的限制: 在Java中一般的类和方法,只能使用具体的类型,要么是基本数据类型,要么是自定义类型.如果要编写可以应用于多种类型的代码,这种刻板的限制就会束缚 ...

  9. Java基础学习总结(83)——Java泛型总结

    1. 什么是泛型? 泛型(Generic type 或者 generics)是对 Java 语言的类型系统的一种扩展,以支持创建可以按类型进行参数化的类.可以把类型参数看作是使用参数化类型时指定的类型 ...

随机推荐

  1. uva 10817(状压dp)

    题意:就是有个学校要招老师.要让没门课至少有两个老师可以上.每个样样例先输入三个数字课程数量s,已经在任的老师数量,和应聘的老师数量.已经在任的一定要聘请. 思路是参考了刘汝佳书上的,关键如何状压. ...

  2. ios基础篇(二十七)—— Json解析

    一.什么是Json JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式.它基于ECMAScript的一个子集. JSON采用完全独立于语言的文本格式,但是也使 ...

  3. thinkjs中自定义sql语句

    一直以为在使用thinkjs时,只能是它自带的sql语句查询,当遇到类似于这样的sql语句时,却不知道这该怎样来写程序,殊不知原来thinkjs可以执行自定义sql语句 SELECT * from a ...

  4. C# in VS

    1. DllImport是System.Runtime.InteropServices命名空间下与与非托管相关的一个属性类,负责导出从非托管的dll中导出函数信息,导出的函数在声明时必须有extern ...

  5. classLIST元素增删改查方法

    window.onload=function () { var oDiv=document.getElementsByTagName('div')[0]; var oInP=document.getE ...

  6. EF6 CodeFirst+Repository+Ninject+MVC4+EasyUI实践(四)

    前言 这一篇,我们终于到了讲解Entity Framework CodeFirst 的时刻了,首先创建实体对象模型,然后会通过配置Fluent API的方式来对实体对象模型进行完整的数据库映射操作. ...

  7. Install Atom editor in ubuntu 14.04

    Step 1: Add repository sudo add-apt-repository ppa:webupd8team/atom Step 2: Update the repository su ...

  8. WinPipe后门程序代码示例(仅限技术交流)

    具体怎么编译,生成执行程序,不懂得先学习C++程序代码编译和集成开发环境. 多的不说了,只有两个代码文件,一个头文件,一个源文件.不多说了,直接上干货. (恶意使用,或者商用,后果自负,与本人无关.) ...

  9. [转载]Bison-Flex 笔记

    FLEX 什么是FLEX?它是一个自动化工具,可以按照定义好的规则自动生成一个C函数yylex(),也成为扫描器(Scanner).这个C函数把文本串作为输入,按照定义好的规则分析文本串中的字符,找到 ...

  10. javascript选择排序

    function selectionSort(arr){ var index,value; for(var i = 0;i < arr.length;i ++){ index = i; //先记 ...