Hi,小伙伴你好~欢迎进入泛型第二节内容的学习,在学习之前友情提醒一下:学习泛型需要小伙伴们具备一定的javaSE基础,如果之前小伙伴们没有接触过java,大家可以移步到千锋北京java好程序员的javaSE课程进行学习。
没看过第一节内容的朋友可以点击博主首页查看第一关内容哦~
1. 为什么引入泛型/引入泛型的必要性
来,我们深入分析下上述代码运行时出现错误的主要原因:
ArrayList的本质是一个Object数组Object[] elementArr,这种设计虽然体现出来了泛型的思想(泛型:泛
指任意类型),但是有以下问题:
由于早期我们开发者经常操作集合对象,所以频繁的出现运行期间异常问题,这种问题困扰了很多人,SUN
官方的设计师们也下定决心解决这个问题,这些大佬们在想: 如果集合能和数组一样,在定义时就指定好类
型,这样就不会出现运行期间异常问题了,也就规避了很多安全隐患。
所以SUN官方提出了泛型技术来解决操作集合中存在的这些问题。
2. 泛型的引入时机
SUN官方在推出jdk1.5版本时,就对这样的问题提出了解决方案: 泛型。
泛型的由来:通过Object转型问题引入 ,早期的Object类型可以接收任意的对象类型,泛型其实指的就是任
意的对象类型,但是在实际的使用中,会有类型转换的问题。也就存在这隐患,所以Java提供了泛型来解决这
个安全问题。
所以后期SUN官方的这些大佬们,对泛型的设计核心思想就两条:
来我们看一下,SUN官方jdk1.5之后,对List集合的改造:
/**
* @param <E> the type of elements in this list: 定义List时,E可以指代任意对象的类型
*/
public interface List<E> extends Collection<E> {
----------
}
下面我们看一下List的使用:
public class GenericsDemo {
public static void main(String[] args) {
//1.创建一个List对象:指定泛型为String
List<String> list = new ArrayList<String>();
//2.向List中添加数据:必须添加String类型的数据
list.add("corn");
list.add("java");
//3.遍历集合
for (int i = 0; i <list.size() ; i++) {
//4.把集合中的每个元素转成String类型
String ele = (String) list.get(i);
//5.打印-测试结果
System.out.println("元素的值:"+ele);
}
}
}
那么如果我们使用 List
可以看到下图:不允许添加除String类型以外的,其它类型的数据,这样后期操作就不会有问题了。
小伙伴,我们来简单的总结一下:
虽然泛型在定义时可以表示任意对象的类型,但是我们在使用是,必须明确泛型指代的具体类型,那么
就这么一点小小的改动带来的确实本质性的变化: 一劳永逸,体现了泛型的通用性,规避了很多安全问题。
3. 泛型能做的哪些事
经过我们刚才的"一顿分析与操作",小伙伴应该基本清楚了泛型能够解决那些问题了。
jdk1.5之后加入泛型,主要是为了解决类型转换的安全隐患,具体体现如下:
那么小伙伴们,我们通过下面代码来细细品味一番:
public class GenericsDemo2 {
public static void main(String[] args) {
//1.创建一个泛型为String的集合: 解决泛型对象实例化之后,可以随意添加任何类型的对象的问题。
List<String> list = new ArrayList<String>();
//2.向List中添加String类型的数据:解决获取泛型元素前,需要提前确定元素的类型的问题。。
list.add("corn");
list.add("java");
//list.add(66);//报编译错误:不能添加int型的数据
//3.遍历集合
for (int i = 0; i <list.size() ; i++) {
//4.把集合中的每个元素:解决获取元素时,需要进行显式类型转换的问题。
String ele = list.get(i);// 不需要类型强转:解决容易出现类型转换出错的问题。
//5.打印-测试结果
System.out.println("元素的值:"+ele);
}
}
}
其实看完这些代码后,想必小伙伴心中都有了一个明确的答案:
之前我们没有使用泛型操作集合,可以添加任意类型的数据,在后期运行代码时,进行类型转换就会出问题。
如果我们使用了泛型,那么在添加数据时,如果添加的数据类型不对,编译就会出问题,更不用说后期运行了。
所以我们用一句话总结泛型:
泛型主要是将运行期间的异常问题,转移到编译期间来体现,避免了类型强制转换的问题。
闯关练习
需求:创建一个指定泛型为Integer的Set集合,添加数字1到100,取出里面的偶数。
答案:
public class GenericsDemo3 {
public static void main(String[] args) {
//1.创建一个泛型为Integer的集合
Set<Integer> numbers = new HashSet<Integer>();
//2.向set集合中添加数字:1-100
for (int i = 1; i <=100 ; i++) {
numbers.add(i);
}
//3.遍历set集合,获取里面的偶数并打印
for (Integer elementData : numbers) {
//4.条件判断:
if(elementData%2==0){
//5.打印测试:
System.out.println("偶数是:"+elementData);
}
}
}
}
走到这里,相信小伙伴们已经知道泛型的基本使用了,那么泛型之美到底体现在什么地方呢? 我们一起揭开这
位“美人”的神秘面纱。
泛型之美具体体现如下三个方面:
1. 编译期间类型检查。
如下代码:
Set<Integer> set = new HashSet<Integer>();//指定set集合的泛型为Integer
set.add(100);//添加数字
set.add("java");//报编译期间异常:集合的泛型为Integer,不能添加String的字符串
来,小伙们我们来分析下这段代码:
1.我们创建了一个带Integer泛型的Set集合对象,指定Set集合只能添加Integer类型的数据
2.如果添加其它类型的数据,java的编译器就会检查,并且提示错误信息,就好像老师检查作业一样,在出错的地方标记红线。
在编译过程,java的编译器都会自动检查添加的数据与我们指定的泛型是否一致,以后再也不怕添加错误的数据了,很赞吧。
2. 避免强转类型转换
如下代码:
//1.定义没有泛型的方法: 创建对象
public static Object createObj(Object obj){
return obj;
}
//2.定义有泛型的方法: 创建对象
public static <T> T createT(T t){
return t;
}
//3.测试
public static void main(String[] args) {
Date date1 = (Date) createObj(new Date());//没有泛型的方法: 类型强转
Date date2 = createT(new Date());//有泛型的方法:不需要类型强转
}
为了加强对比,我们定义了两个方法,一个带泛型,一个不带泛型。
根据测试的结果,显而易见:
所以我们在开发中,特别是在定义创建对象的方法时,一般都使用泛型来进行定义,从而避免后期的类型强转。
3. 可读性和灵活性
如下代码:
public class GenericDemo6 {
public static void main(String[] args) {
//1.带泛型的map集合
Map<String,Student> map = new HashMap<String,Student>();//使用泛型:可读性强
map.put("01号",new Student("乔丹",23));//01号: 学生乔丹
map.put("02号",new Student("皮蓬",36));//02号: 学生皮蓬
//2.不带泛型的map集合
Map map2 = new HashMap();//没有泛型:可读性差
map2.put(new Student("乔丹",23),"01号");//学生在前:编号在后
map2.put("02号",new Student("皮蓬",36));//学生在后:编号在前
}
}
class Student{
String username;
Integer age;
public Student(String username, Integer age) {
this.username = username;
this.age = age;
}
public Student() {
}
}
通过这段代码,我们可以看到泛型能规范代码的书写,让我们的代码可读性更强,便于后期我们对数据的处理。
如果我们使用不带泛型的map集合保存数据,那么map的数据保存很混乱,不便于后期对数据进行处理。
闯关练习
请描述下列哪些选项是泛型的优点:
A: 泛型可以避免类型强转
B: 泛型可以在编译期间进行检查
C: 泛型可以提高代码的可读性
D: 泛型可以提高代码的灵活性
答案:ABCD
小伙伴们,泛型我们已经学到第四关了,是不是感觉泛型很强大啊!但是不要得意,泛型也有“软肋”,
泛型在使用时,有时并不能随意指定任意类型,也就是说,泛型在使用时具有类型限制,具体体现为泛型之飞天 ;
就是泛型的上限,泛型之遁地 ; 就是泛型的下限。
来吧,我们一起来看一下泛型的上限和下限。
在泛型上限和下限的分析过程中,我们会看到 ?符号经常出现,在这里 ?表示通配符,表示任意类型,小伙伴们需要注意一下喔。
1.泛型通配符
切记: ?表示通配符,表示任意的类型。
如下代码:
ArrayList<?> list1 ;
ArrayList<String> list2 = new ArrayList<String>();
ArrayList<Integer> list3 = new ArrayList<Integer>();
list1 = list2;// ? 表示 String类型
list1 = list3;// ? 表示 Integer类型
在这里,小伙伴一定要注意: ?这个符号神通广大,可以用来表示任意的泛型类型。
上述代码中创建了一个带?泛型list1。
讲到这里,小伙伴基本上明白了?符号的含义了,通常 ?会出现在泛型的上限和下限定义中使用中,我们接下来看看?这个通配符,
在泛型的上限和下限过程怎么使用的,let`s go。
2.泛型的上限
定义的基本语法:定义基本语法:类名或者接口名 extends T>,那么此时 ?表示T类型,或者T的子类型
我们通过一段List接口的源码来分析一下泛型的上限,源码如下:
public interface List<E> extends Collection<E> {
/**
* 方法作用:将一个集合 添加到 List中,
* 这时集合的类型 ? extends E ,这里? 表示是E的子类类型,?的上限不能超过E
*/
boolean addAll(Collection<? extends E> c);
----
}
代码演示:
List<Number> list = new ArrayList<Number>();//创建一个集合: 泛型为Number
List<Number> son1 = new ArrayList<Number>() ;
List<Long> son2 = new ArrayList<Long>() ;
List<String> str_list = new ArrayList<String>();
list.addAll(son1);// ? 表示 Number,上限是Number
list.addAll(son2);// ? 表示Long,Long是Number的子类
// list.addAll(str_list);//编译报错: 原因 String 和 Number 没有继承关系
泛型的上限,就是在使用泛型时,不能超过extends后面定义的类型,所以大家在使用时,一定要注意类型的子、父级关系。
3.泛型的下限
定义基本语法: 类名或者接口名 super T>,那么此时 ?表示T类型,或者T的父类型
刚刚看了泛型的上限,那么下限对于小伙伴来说就会简单很多了。
我们还是通过List集合来分析泛型的下限。
比如:
List<? super Integer> list = new ArrayList<Integer>();//创建一个集合
list = new ArrayList<Integer>();// ? 表示Integer
list = new ArrayList<Number>();// ? 表示Number: Number是Integer的父类
list = new ArrayList<Object>();// ? 表示Object: Object是Integer的父类
// list = new ArrayList<String>();//编译报错: ? 表示String: String和Integer没有关系
泛型的下限,就是在使用泛型时,必须高于super后面的定义的类型,所以大家在使用时,一定要注意类型的父、子级关系
闯关练习
请描述 ? 通配符 在泛型中的具体使用(多选)。
A:在定义泛型的上限时: 可以使用?通配符 表示 泛型的子类
B:在定义泛型的下限时: 可以使用?通配符 表示 泛型的父类
C:在定义泛型时,可以不用 ?通配符
D: 以上说法都不对
答案:
ABC
第三节及之后的内容会稍后发布哦~
感兴趣就点个关注吧~