全国新农村建设中心网站,交换链接或称互惠链接,3d模型素材库,做花生的网站JAVA基础学习笔记-day12-泛型 1. 泛型概述1.1 泛型的引入 2. 使用泛型举例2.1 集合中使用泛型2.2 相关使用说明 3. 自定义泛型结构3.1 泛型的基础说明3.2 自定义泛型类或泛型接口3.2.1 说明3.2.2 注意 3.3 自定义泛型方法3.3.1 说明 4. 泛型在继承上的体现5. 通配符的使用5.1 通… JAVA基础学习笔记-day12-泛型 1. 泛型概述1.1 泛型的引入 2. 使用泛型举例2.1 集合中使用泛型2.2 相关使用说明 3. 自定义泛型结构3.1 泛型的基础说明3.2 自定义泛型类或泛型接口3.2.1 说明3.2.2 注意 3.3 自定义泛型方法3.3.1 说明 4. 泛型在继承上的体现5. 通配符的使用5.1 通配符的理解5.2 通配符的读与写5.3 使用注意点5.4 有限制的通配符 博文主要是自己学习JAVA基础中的笔记供自己以后复习使用参考的主要教程是B站的 尚硅谷宋红康2023大数据教程 水之积也不厚则其负大舟也无力。—庄子《庄子·逍遥游》
1. 泛型概述
1.1 泛型的引入
在Java中我们在声明方法时当在完成方法功能时如果有未知的数据需要参与这些未知的数据需要在调用方法时才能确定那么我们把这样的数据通过形参表示。在方法体中用这个形参名来代表那个未知的数据而调用者在调用时对应的传入实参就可以了。 受以上启发JDK1.5设计了泛型的概念。泛型即为“类型参数”这个类型参数在声明它的类、接口或方法中代表未知的某种通用类型。 举例1
集合类在设计阶段/声明阶段不能确定这个容器到底实际存的是什么类型的对象所以在JDK5.0之前只能把元素类型设计为ObjectJDK5.0时Java引入了“参数化类型Parameterized type”的概念允许我们在创建集合时指定集合元素的类型。比如ListString这表明该List只能保存字符串类型的对象。 使用集合存储数据时除了元素的类型不确定其他部分是确定的例如关于这个元素如何保存如何管理等。
java.lang.Comparable接口和java.util.Comparator接口是用于比较对象大小的接口。这两个接口只是限定了当一个对象大于另一个对象时返回正整数小于返回负整数等于返回0但是并不确定是什么类型的对象比较大小。JDK5.0之前只能用Object类型表示使用时既麻烦又不安全因此 JDK5.0 给它们增加了泛型。 其中T就是类型参数即泛型。 所谓泛型就是允许在定义类、接口时通过一个标识表示类中某个属性的类型或者是某个方法的返回值或参数的类型。这个类型参数将在使用时例如继承或实现这个接口、创建对象或调用方法时确定即传入实际的类型参数也称为类型实参。 2. 使用泛型举例
自从JDK5.0引入泛型的概念之后对之前核心类库中的API做了很大的修改例如JDK5.0改写了集合框架中的全部接口和类、java.lang.Comparable接口、java.util.Comparator接口、Class类等。为这些接口、类增加了泛型支持从而可以在声明变量、创建对象时传入类型实参。
2.1 集合中使用泛型
集合中没有使用泛型时 集合中使用泛型时 Java泛型可以保证如果程序在编译时没有发出警告运行时就不会产生ClassCastException异常。即把不安全的因素在编译期间就排除了而不是运行期既然通过了编译那么类型一定是符合要求的就避免了类型转换。 同时代码更加简洁、健壮。 把一个集合中的内容限制为一个特定的数据类型这就是generic背后的核心思想。 2.2 相关使用说明
在创建集合对象的时候可以指明泛型的类型。 具体格式为List list new ArrayList();JDK7.0时有新特性可以简写为 List list new ArrayList(); //类型推断泛型也称为泛型参数即参数的类型只能使用引用数据类型进行赋值。不能使用基本数据类型可以使用包装类替换集合声明时声明泛型参数。在使用集合时可以具体指明泛型的类型。一旦指明类或接口内部凡是使用泛型参数的位置都指定为具体的参数类型。如果没有指明的话看做是Object类型。
3. 自定义泛型结构
3.1 泛型的基础说明
1、类型这种语法形式就叫泛型。 类型的形式我们称为类型参数这里的类型习惯上使用T表示是Type的缩写。即。 代表未知的数据类型我们可以指定为 等。 类比方法的参数的概念我们把称为类型形参将 称为类型实参有助于我们理解泛型 这里的T可以替换成KV等任意字母。
2、在哪里可以声明类型变量T
声明类或接口时在类名或接口名后面声明泛型类型我们把这样的类或接口称为泛型类或泛型接口。
【修饰符】 class 类名类型变量列表 【extends 父类】 【implements 接口们】{}
【修饰符】 interface 接口名类型变量列表 【implements 接口们】{}//例如
public class ArrayListE
public interface MapK,V{....
} 声明方法时在【修饰符】与返回值类型之间声明类型变量我们把声明了类型变量的方法称为泛型方法。
[修饰符] 类型变量列表 返回值类型 方法名([形参列表])[throws 异常列表]{//...
}//例如java.util.Arrays类中的
public static T ListT asList(T... a){....
}3.2 自定义泛型类或泛型接口
当我们在类或接口中定义某个成员时该成员的相关类型是不确定的而这个类型需要在使用这个类或接口时才可以确定那么我们可以使用泛型类、泛型接口。
3.2.1 说明
① 我们在声明完自定义泛型类以后可以在类的内部比如属性、方法、构造器中使用类的泛型。
② 我们在创建自定义泛型类的对象时可以指明泛型参数类型。一旦指明内部凡是使用类的泛型参数的位置都具体化为指定的类的泛型类型。
③ 如果在创建自定义泛型类的对象时没有指明泛型参数类型那么泛型将被擦除泛型对应的类型均按照Object处理但不等价于Object。
经验泛型要使用一路都用。要不用一路都不要用。
④ 泛型的指定中必须使用引用数据类型。不能使用基本数据类型此时只能使用包装类替换。
⑤ 除创建泛型类对象外子类继承泛型类时、实现类实现泛型接口时也可以确定泛型结构中的泛型参数。
如果我们在给泛型类提供子类时子类也不确定泛型的类型则可以继续使用泛型参数。
我们还可以在现有的父类的泛型参数的基础上新增泛型参数。
3.2.2 注意
① 泛型类可能有多个参数此时应将多个参数一起放在尖括号内。比如E1,E2,E3
② JDK7.0 开始泛型的简化操作ArrayList flist new ArrayList();
③ 如果泛型结构是一个接口或抽象类则不可创建泛型类的对象。
④ 不能使用new E[]。但是可以E[] elements (E[])new Object[capacity];
参考ArrayList源码中声明Object[] elementData而非泛型参数类型数组。
⑤ 在类/接口上声明的泛型在本类或本接口中即代表某种类型但不可以在静态方法中使用类的泛型。
⑥ 异常类不能是带泛型的。
3.3 自定义泛型方法
如果我们定义类、接口时没有使用泛型参数但是某个方法形参类型不确定时这个方法可以单独定义泛型参数。
3.3.1 说明
泛型方法的格式
[访问权限] 泛型 返回值类型 方法名([泛型标识 参数名称]) [抛出的异常]{}方法也可以被泛型化与其所在的类是否是泛型类没有关系。泛型方法中的泛型参数在方法被调用时确定。泛型方法可以根据需要声明为static的。
4. 泛型在继承上的体现
如果B是A的一个子类型子类或者子接口而G是具有泛型声明的类或接口G并不是G的子类型
比如String是Object的子类但是List并不是List的子类。
5. 通配符的使用
当我们声明一个变量/形参时这个变量/形参的类型是一个泛型类或泛型接口例如Comparator类型但是我们仍然无法确定这个泛型类或泛型接口的类型变量的具体类型此时我们考虑使用类型通配符 ? 。
5.1 通配符的理解
使用类型通配符
比如List?Map?,?
List?是ListString、ListObject等各种泛型List的父类。
5.2 通配符的读与写
写操作
将任意元素加入到其中不是类型安全的
Collection? c new ArrayListString();c.add(new Object()); // 编译时错误因为我们不知道c的元素类型我们不能向其中添加对象。add方法有类型参数E作为集合的元素类型。我们传给add的任何参数都必须是一个未知类型的子类。因为我们不知道那是什么类型所以我们无法传任何东西进去。
唯一可以插入的元素是null因为它是所有引用类型的默认值。
读操作
另一方面读取List?的对象list中的元素时永远是安全的因为不管 list 的真实类型是什么它包含的都是Object。
5.3 使用注意点
注意点1编译错误不能用在泛型方法声明上返回值类型前面不能使用?
public static ? void test(ArrayList? list){
}注意点2编译错误不能用在泛型类的声明上
class GenericTypeClass?{
}注意点3编译错误不能用在创建对象上右边属于创建集合对象
ArrayList? list2 new ArrayList?();5.4 有限制的通配符 ? 允许所有泛型的引用调用 通配符指定上限? extends 类/接口 使用时指定的类型必须是继承某个类或者实现某个接口即 通配符指定下限? super 类/接口 使用时指定的类型必须是操作的类或接口或者是操作的类的父类或接口的父接口即 说明 ? extends Number //(无穷小 , Number]
//只允许泛型为Number及Number子类的引用调用? super Number //[Number , 无穷大)
//只允许泛型为Number及Number父类的引用调用? extends Comparable
//只允许泛型为实现Comparable接口的实现类的引用调用举例1 class Creature{}
class Person extends Creature{}
class Man extends Person{}class PersonTest {public static T extends Person void test(T t){System.out.println(t);}public static void main(String[] args) {test(new Person());test(new Man());//The method test(T) in the type PersonTest is not //applicable for the arguments (Creature)test(new Creature());}
}