java 泛型的基本定义和简单使用

1. 泛型集合
2.泛型深入复杂用法
2.1 普通泛型
2.2 < T > 的含义
2.3 < ? > 的含义
2.4 静态泛型
常用的泛型含义:

T - Type(类型)
R - Result(结果)
K - Key(键)
V - Value(值)
E - Element (元素)
N - Number(数字)
? - 不确定类型
【泛型】提供了编译时类型安全检测机制,该机制允许程序员在编译时检测到非法的类型。泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数。
比如我们要写一个排序方法,能够对整型数组、字符串数组甚至其他任何类型的数组进行排序,我们就可以使用 Java 泛型。

1. 泛型集合
概念:参数化类型、类型安全的集合,强制集合元素的类型必须一致。
特点:
1. 编译时即可检查,而非运行时抛出异常;
2. 访问时,不必类型转换(拆箱);
3. 不同泛型之间引用不能相互赋值,泛型不存在多态。

import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
public class TestBasicGeneric {
      public static void main(String[] args) {
            // 集合:元素的类型可以不一致
            // <E> ==  <Student> 强制/约束元素类型必须一致
            List<Student> list = new ArrayList<Student>(); 
            
            list.add(new Student("tom", 20));
            list.add(new Student("jack", 21));
            
            Student s = null;
            for (int i = 0; i < list.size(); i++) {
                  // 省略父类引用接收返回值、省略instanceof类型判断、省略向下转型
                  s = list.get(i); 
                  System.out.println(s.name + " " + s.age); // tom 20  jack 21
            }
             //类型推导:E 可以省略,但创建对象的 <> 不能省略
            List<Integer> ll = new LinkedList<>();
            ll.add(100);
            ll.add(200);
            System.out.println(ll.toString()); // [100, 200]
      }
}
class Student {
      String name;
      int age;
      public Student() {}
      public Student(String name, int age) {
            super();
            this.name = name;
            this.age = age;
      }
}
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
2.泛型深入复杂用法
——需要时间、经验的积累。
概念:约束-规范类型
场景:

2.1 普通泛型
类:创建对象时,为类所定义的泛型,进行参数化赋值
接口:实现接口时,为接口所定义的泛型,进行参数化赋值
public class TestInstanceGeneric {
      public static void main(String[] args) {
            MyClass<Integer> mc1 = new MyClass<Integer>();
            MyClass<String> mc2 = new MyClass<String>();
            mc1.m1(10);
            mc2.m1("Hello");
      }
}
/**
* 案例1:类的实例泛型
*/
class MyClass<E>{ // E代表一种通配任意类型符号,未指明类型前为Object
      public void m1(E e) {} // 泛型可以动态改变类型
      public void m2(Object o) {} // 固定写死不能变
}
/**
* 案例2:接口的实例泛型
*/
//E=Element  T=Type  K=Key  V=Value 等等...
interface MyInterface<T> { // 泛型接口
      public T method(T a);
}
//应用泛型:需要给定具体类型,如果没写默认为Object类型
class MyImplClass implements MyInterface<Dog> {
      @Override
      public Dog method(Dog a) {
            return a;
      }
}
class Dog{}
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
2.2 < T > 的含义
<T> 代表某种通配类型
<T extends Object> 约定类型T为Object的子类
<T extends MyClass> 约定类型T只能为MyClass类的子类
<T extends MyClass & MyInterFace> 约定类型T为MyClass子类同时实现了MyInterFace接口,父类必须写在最前面且可&上多个接口
<T extends MyInterFace> 约定类型T只要实现了MyInterface接口
<T extends MyInterFace> 约定类型T要实现了MyInterface接口(且必须为T泛型的接口)
2.3 < ? > 的含义
<?> 代表任意通配泛型
<? extends FatherClass> 泛型类型?必须是FatherClass的子类
<? extends MyInterface> 泛型类型?必须是MyInterface的实现类
<? super SubClass> 泛型类型?必须是SubClass类或SubClass的父类
<? extends MyInterface<? extends T>> 泛型类型?必须是MyInterface的实现类,且接口又指定了泛型(必须T类型的子类)
<T extends MyInterface<? super T>> 要求T所代表的类型必须实现MyInterface接口,同时,接口泛型必须是T类型或T的父类
import java.util.ArrayList;
import java.util.List;
public class TestStaticGeneric {
      public static void main(String[] args) {
            List<Dog> list1 = new ArrayList<Dog>();
            List<Bus> list2 = new ArrayList<Bus>();
            List<Animal> list3 = new ArrayList<Animal>();
            
            m1(list1); // <? extends Animal> 类型必须是Animal的子类
            // <? extends MyInterface<? extends Vehicle>> 类型必须是:
            // 实现了接口(继承Vehicle的接口)的实现类
            m2(list2); 
            
            m3(list1); // <? super Dog> 既可以使用自身类Dog作为元素类型
            m3(list3); // <? super Dog> 又可以使用父类Animal作为元素类型
      }
      
      /**
       * <?> 代表任意泛型
       * <? extends Animal> 泛型类型?必须是Animal的子类
       * <? extends MyInterface> 泛型类型?必须是MyInterface的实现类
       * <? super Dog> 泛型类型?必须是Dog类或Dog的父类
       * <? extends MyInterface<? extends Vehicle>> 泛型类型?必须是:
       *     MyInterface的实现类,且接口又指定了泛型(类型约束)
       */
      public static void m1(List<? extends Animal> list) {}
      public static void m2(List<? extends MyInterface<? extends  Vehicle>> list2) {}
      public static void m3(List<? super Dog> list) {}
}
interface MyInterface<T>{}
class Animal{}
class Dog extends Animal{}
class Vehicle{}
class Bus extends Vehicle implements MyInterface<Bus>{}
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
2.4 静态泛型
1)定义在方法的返回值类型前面:、、<T extends Comparable>、<T extends Comparable<? super T>>,可使用&多个接口
场景:形参列表、返回值两种场景,不单单可以规范泛型,还可以语义化泛型。
2)定义在方法的形参列表当中:<?>、<? extends Object>、<? super SubClass>,不可使用&
场景:只能应用在形参列表上,规范泛型。

import java.util.ArrayList;
import java.util.List;
public class TestStaticGeneric3 {
      public static void main(String[] args) {
            m( new ArrayList<Integer>() );
            m( new ArrayList<Double>() );
            m( new ArrayList<String>() );
      }
      
      public static <T> void m(List<T> list) { }
      /**
       * 目标:集合中的所有对象,必须具备本类型的两个元素进行比较的方法。
       * <T extends Comparable> 只要List集合中的元素实现了Comparable接口就行
       * <T extends Comparable<T>> 只要List集合中的元素实现了Comparable接口(必须为T泛型的接口)才行
       * 此函数:
       * 当List<T>被传输实参后,要求T所代表的类型必须实现Comparable接口,
       *    同时,接口泛型必须是T类型或T的父类
       */
      public static <T extends Comparable<? super T>> void  sort(List<T> list) { }
}
————————————————
版权声明:本文为CSDN博主「不才Jerry」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/sinat_36184075/article/details/104686649

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值