java泛型是J2 SE1.5中引入的一个新特性,其本质是参数化类型,也就是说所操作的数据类型被指定为一个参数(type parameter)这种参数类型可以用在类、接口和方法的创建中,分别称为泛型类、泛型接口、泛型方法。
提出背景
Java集合(Collection)中元素的类型是多种多样的。例如,有些集合中的元素是Byte类型的,而有些则可能是String类型的,等等。Java允许程序员构建一个元素类型为Object的Collection,其中的元素可以是任何类型在Java SE 1.5之前,没有泛型(Generics)的情况下,通过对类型Object的引用来实现参数的“任意化”,“任意化”带来的缺点是要作显式的强制类型转换,而这种转换是要求开发者对实际参数类型可以在预知的情况下进行的。对于强制类型转换错误的情况,编译器可能不提示错误,在运行的时候才出现异常,这是一个安全隐患。因此,为了解决这一问题,J2SE 1.5引入泛型也是自然而然的了。作用
第一是泛化。可以用T代表任意类型Java语言中引入泛型是一个较大的功能增强不仅语言、类型系统和编译器有了较大的变化,以支持泛型,而且类库也进行了大翻修,所以许多重要的类,比如集合框架,都已经成为泛型化的了,这带来了很多好处。第二是类型安全。泛型的一个主要目标就是提高ava程序的类型安全,使用泛型可以使编译器知道变量的类型限制,进而可以在更高程度上验证类型假设。如果不用泛型,则必须使用强制类型转换,而强制类型转换不安全,在运行期可能发生ClassCast Exception异常,如果使用泛型,则会在编译期就能发现该错误。
第三是消除强制类型转换。泛型可以消除源代码中的许多强制类型转换,这样可以使代码更加可读,并减少出错的机会。
第四是向后兼容。支持泛型的Java编译器(例如JDK1.5中的Javac)可以用来编译经过泛型扩充的Java程序(Generics Java程序),但是现有的没有使用泛型扩充的Java程序仍然可以用这些编译器来编译。
使用注意事项
使用ava泛型应该注意如下几点:①在定义一个泛型类时,在“<>”之间定义形式类型参数,例如:“class TestGen<K,V>”,其中“K”,“V”不代表值,而是表示类型。
②实例化泛型对象时,一定要在类名后面指定类型参数的值(类型),一共要有两次书写。
③泛型中<Kextends ObjecD,extends>并不代表继承,它是类型范围限制。
④使用泛型时,泛型类型必须为引用数据类型,不能为基本数据类型,Java中的普通方法,构造方法,静态方法中都可以使用泛型,方法使用泛型之前必须先对泛型进行声明,可以使用任意字母,一般都要大写。
⑤不可以用一个本地类型(如int float)来替换泛型。
⑥运行时类型检查,不同类型的泛型类是等价的(Pair与Pair是属于同一个类型Pair),这一点要特别注意,即如果obj instance of Pai == true的话,并不代表objget First()的返回值是一个String类型。
⑦泛型类不可以继承Exception类,即泛型类不可以作为异常被抛出。
⑧不可以定义泛型数组。
⑨不可以用泛型构造对象,即:first = new T();是错误的。
⑩在static方法中不可以使用泛型,泛型变量也不可以用static关键字来修饰
⑪不要在泛型类中定义equals(Tx)这类方法,因为Object类中也有equals方法,当泛型类被擦除后,这两个方法会冲突。
⑫根据同一个泛型类衍生出来的多个类之间没有任何关系,不可以互相赋值。
⑬若某个泛型类还有同名的非泛型类,不要混合使用,坚持使用泛型类。
上面照抄自百度百科
Java泛型的创建和使用Demo
复制Javapackage com.test.collection;
/**
* @Desc -泛型的创建和使用
* @Author luolei
* @Date 2020/03/12 16:22
*/
public class FanxingDemo {
public static void main(String[] args) {
MyClass<String, Integer> list = new MyClass<String, Integer>("luolei", 21);
System.out.println(list.toString()); /*输出 MyClass{parm1=luolei, parm2=21}*/
MyClass list2 = new MyClass("luolei", 21);
System.out.println(list2.toString()); /*输出 MyClass{parm1=luolei, parm2=21}*/
}
}
/**
* @Desc:
* 类名后面添加"<T>",表示该类的内部代码中,所有的T类型都为外部需要时再指定的泛型。
* 如果泛型不止一个,则可以用逗号隔开,比如两个泛型可用"<T, R>"
* @Date: 2020/3/12 16:38
*/
class MyClass<T, R> {
private T parm1; /*数据类型为泛型T*/
private R parm2; /*数据类型为泛型R*/
public MyClass(T parm1, R parm2) {
this.parm1 = parm1;
this.parm2 = parm2;
}
@Override
public String toString() {
return "MyClass{" +
"parm1=" + parm1 +
", parm2=" + parm2 +
'}';
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42