Java 泛型

简介:

1、泛型的由来

  我们先看下面这段代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
        List list = new ArrayList();
list.add(24);  //向集合中添加一个 Integer 类型的数据
list.add("Tom");    //向集合中添加一个 String 类型的数据
 
for(int i = 0 ; i < list.size() ; i++){
    Object obj = list.get(i);  //注意这里每个类型都是 Object
    System.out.println(obj);
}
 
//如果我们遍历的时候就想得到自己想要的数据类型
for(int i = 0 ; i < list.size() ; i++){
    String obj = (String) list.get(i);  //在取 Integer 的时候会报类型转换错误
    System.out.println(obj);
}

  报错信息如下:

  也就是 集合中第二个数据是 Integer,但是我们取出来的时候将其转换为 String 了,所以报错。

  那么这个如何解决呢?

  ①、我们在遍历的时候,根据每个数据的类型判断,然后进行强转。

那么我们说这个集合只有两条数据,我们可以进行判断强转,如果数据有成千上万条呢,我们都通过这样判断强转肯定不可取

  ②、在往集合中加入数据的时候,我们就做好限制,比如这个集合只能添加 String 类型的;下一个集合只能添加 Integer 类型的,那么我们在取数据的时候,由于前面已经限制了该集合的数据类型,那么就很好强转了。

  这第二种解决办法,也就是我们这篇文章讲的 泛型

 

2、什么是泛型?

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

  在Java SE 1.5之前,没有泛型的情况的下,通过对类型Object的引用来实现参数的“任意化”,“任意化”带来的缺点是要做显式的强制类型转换,而这种转换是要求开发者对实际参数类型可以预知的情况下进行的。对于强制类型转换错误的情况,编译器可能不提示错误,在运行的时候才出现异常,这是一个安全隐患。

 

3、泛型的基本用法

  3.1 对于上面的问题我们只需要将上述代码的 List list = new ArrayList()  改为   List<String> list = new ArrayList<String>();

1
2
3
4
5
6
7
8
9
10
        List<String> list = new ArrayList<String>();
//list.add(22);  //向集合中添加一个 Integer 类型的数据时,编译器会报错
list.add("Bob");  //向集合中添加一个 String 类型的数据
list.add("Tom");    //向集合中添加一个 String 类型的数据
 
//如果我们遍历的时候就想得到自己想要的数据类型
for(int i = 0 ; i < list.size() ; i++){
    String obj = list.get(i);  //这里就不需要强转了,前面添加的是什么类型,这里获取的就是什么类型
    System.out.println(obj);
}           

  

  

  3.2 泛型是在编译阶段有效

 

1
2
3
4
5
    List<String> list1 = new ArrayList<String>();
List list2 = new ArrayList();
Class c1 = list1.getClass();
Class c2 = list2.getClass();
System.out.println(c1==c2); //true   

 

  上述代码,由于我们知道反射是在运行时阶段,c1==c2为 true,说明了编译之后的 class 文件中是不包含任意的泛型信息的。如果不信,我们可以看 class 文件的反编译信息

1
2
3
4
5
java.util.List list1 = new ArrayList();
java.util.List list2 = new ArrayList();
Class c1 = list1.getClass();
Class c2 = list2.getClass();
System.out.println(c1.equals(c2));

  我们可以看到 反编译之后的 list1和 list2完全一样。

  结论:Java 泛型只在编译阶段有效,即在编译过程中,程序会正确的检验泛型结果。而编译成功后,class 文件是不包含任何泛型信息的

 

 

  3.3 泛型类和泛型方法

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
public class Box<T> {
    private T box;
    public T getBox(T t){
        this.box = t;
        return t;
    }
    public void getType(){
        System.out.println("T的实际类型为:"+box.getClass().getName());
    }
     
    public static void main(String[] args) {
        Box box = new Box();
        System.out.println(box.getBox(1));
        box.getType();
         
        System.out.println(box.getBox("Tom"));
        box.getType();
    }
 
}

 

  输出结果为:

1
2
3
4
1
T的实际类型为:java.lang.Integer<br>
Tom
T的实际类型为:java.lang.String

  

  3.4 泛型通配符

  在泛型中,我们可以用 ? 来代替任意类型

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
public List wildCard(List<?> list){
     
    return list;
}
 
public static void main(String[] args) {
    GenericTest gt = new GenericTest();
    //构造一个 Interger 类型的集合
    List<Integer> integer = new ArrayList<Integer>();
    integer.add(1);
    System.out.println(gt.wildCard(integer));
    //构造一个 String 类型的集合
    List<String> str = new ArrayList<String>();
    gt.wildCard(str);
    //构造一个 Object 类型的集合
    List<Object> obj = new ArrayList<Object>();
    obj.add(1);
    obj.add("a");
    System.out.println(gt.wildCard(obj));
    //构造一个 任意类型的 集合,这和 List<Object> 存放数据没啥区别
    List list = new ArrayList();
    gt.wildCard(list);
     
}

 

  

 

  3.5 泛型的上限和下限

  ①、上限: 语法(? extends className),即只能为 className 或 className 的子类

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
//通配符的下限,只能是 Number 或 Number的子类
    public List wildCard(List<? extends Number> list){
         
        return list;
    }
 
    public static void main(String[] args) {
        GenericTest gt = new GenericTest();
        //构造一个 Interger 类型的集合
        List<Integer> integer = new ArrayList<Integer>();
        integer.add(1);
        System.out.println(gt.wildCard(integer));
        //构造一个 String 类型的集合
        List<String> str = new ArrayList<String>();
        //gt.wildCard(str);   //编译报错
        //构造一个 Object 类型的集合
        List<Object> obj = new ArrayList<Object>();
        obj.add(1);
        obj.add("a");
        //System.out.println(gt.wildCard(obj)); //编译报错
         
    }

  ①、下限: 语法(? super className),即只能为 className 或 className 的父类

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
//通配符的上限,只能是 Number 或 Number的父类
    public List wildCard(List<? super Number> list){
         
        return list;
    }
 
    public static void main(String[] args) {
        GenericTest gt = new GenericTest();
        //构造一个 Interger 类型的集合
        List<Integer> integer = new ArrayList<Integer>();
        integer.add(1);
        //System.out.println(gt.wildCard(integer));  //编译报错
        //构造一个 String 类型的集合
        List<String> str = new ArrayList<String>();
        //gt.wildCard(str);   //编译报错
        //构造一个 Object 类型的集合
        List<Object> obj = new ArrayList<Object>();
        obj.add(1);
        obj.add("a");
        System.out.println(gt.wildCard(obj));
    }

  

4、泛型的注意事项

  4.1、不能用基本类型来定义泛型,如 int、float

1
List<int> list = new ArrayList<int>(); //不能用 int 这样的基本类型定义泛型

  关于这一点很好想明白,因为 集合中只能存放引用类型的数据,即使你存入基本类型的,Java还是会通过自动拆箱和自动装箱机制将其转换为引用类型

  

  4.2、如果使用 ? 接收泛型对象时,则不能设置被泛型指定的内容

1
2
List<?> list = new ArrayList<>();
        list.add("aa");  //错误,无法设置

  

  4.3、泛型方法的定义与其所在的类是否是泛型类是没有任何关系的,所在的类可以是泛型类,也可以不是泛型类

 

  4.4、泛型类没有继承关系,即String 为 Object 类型的子类,则 List<String> 是 List<Object> 的子类这句话是错误的

  原因:假设上面那句话是正确的,那么由于泛型的产生机制就是放什么类型的数据进去,取出来的就是什么类型,而不用进行类型转换,这里把 String 类型的数据放入Object 类型的泛型集合中,那么取出来的应该就是 String      类型的数据,而实际上取出来的是 Object 类型的数据,这与泛型的产生机制相违背,故不成立!





      本文转自zsdnr  51CTO博客,原文链接:http://blog.51cto.com/12942149/1929675,如需转载请自行联系原作者




相关文章
|
2月前
|
安全 Java 编译器
揭秘JAVA深渊:那些让你头大的最晦涩知识点,从泛型迷思到并发陷阱,你敢挑战吗?
【8月更文挑战第22天】Java中的难点常隐藏在其高级特性中,如泛型与类型擦除、并发编程中的内存可见性及指令重排,以及反射与动态代理等。这些特性虽强大却也晦涩,要求开发者深入理解JVM运作机制及计算机底层细节。例如,泛型在编译时检查类型以增强安全性,但在运行时因类型擦除而丢失类型信息,可能导致类型安全问题。并发编程中,内存可见性和指令重排对同步机制提出更高要求,不当处理会导致数据不一致。反射与动态代理虽提供运行时行为定制能力,但也增加了复杂度和性能开销。掌握这些知识需深厚的技术底蕴和实践经验。
54 2
|
16天前
|
Java 编译器 容器
Java——包装类和泛型
包装类是Java中一种特殊类,用于将基本数据类型(如 `int`、`double`、`char` 等)封装成对象。这样做可以利用对象的特性和方法。Java 提供了八种基本数据类型的包装类:`Integer` (`int`)、`Double` (`double`)、`Byte` (`byte`)、`Short` (`short`)、`Long` (`long`)、`Float` (`float`)、`Character` (`char`) 和 `Boolean` (`boolean`)。包装类可以通过 `valueOf()` 方法或自动装箱/拆箱机制创建。
24 9
Java——包装类和泛型
|
19天前
|
安全 Java API
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
String常量池、String、StringBuffer、Stringbuilder有什么区别、List与Set的区别、ArrayList和LinkedList的区别、HashMap底层原理、ConcurrentHashMap、HashMap和Hashtable的区别、泛型擦除、ABA问题、IO多路复用、BIO、NIO、O、异常处理机制、反射
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
|
16天前
|
存储 安全 搜索推荐
Java中的泛型
【9月更文挑战第15天】在 Java 中,泛型是一种编译时类型检查机制,通过使用类型参数提升代码的安全性和重用性。其主要作用包括类型安全,避免运行时类型转换错误,以及代码重用,允许编写通用逻辑。泛型通过尖括号 `&lt;&gt;` 定义类型参数,并支持上界和下界限定,以及无界和有界通配符。使用泛型需注意类型擦除、无法创建泛型数组及基本数据类型的限制。泛型显著提高了代码的安全性和灵活性。
|
2月前
|
安全 Java Go
Java&Go泛型对比
总的来说,Java和Go在泛型的实现和使用上各有特点,Java的泛型更注重于类型安全和兼容性,而Go的泛型在保持类型安全的同时,提供了更灵活的类型参数和类型集的概念,同时避免了运行时的性能开销。开发者在使用时可以根据自己的需求和语言特性来选择使用哪种语言的泛型特性。
37 7
|
2月前
|
存储 算法 Java
14 Java集合(集合框架+泛型+ArrayList类+LinkedList类+Vector类+HashSet类等)
14 Java集合(集合框架+泛型+ArrayList类+LinkedList类+Vector类+HashSet类等)
41 2
14 Java集合(集合框架+泛型+ArrayList类+LinkedList类+Vector类+HashSet类等)
|
2月前
|
存储 安全 Java
如何理解java的泛型这个概念
理解java的泛型这个概念
|
2月前
|
存储 缓存 Java
|
2月前
|
安全 Java
【Java 第六篇章】泛型
Java泛型是自J2 SE 1.5起的新特性,允许类型参数化,提高代码复用性与安全性。通过定义泛型类、接口或方法,可在编译时检查类型安全,避免运行时类型转换异常。泛型使用尖括号`&lt;&gt;`定义,如`class MyClass&lt;T&gt;`。泛型方法的格式为`public &lt;T&gt; void methodName()`。通配符如`?`用于不确定的具体类型。示例代码展示了泛型类、接口及方法的基本用法。
11 0
|
2月前
|
Java
【Java基础面试四十五】、 介绍一下泛型擦除
这篇文章解释了Java泛型的概念,它解决了集合类型安全问题,允许在创建集合时指定元素类型,避免了类型转换的复杂性和潜在的异常。
下一篇
无影云桌面