一、泛型简介
泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数。
在Java SE 1.5之前,没有泛型的情况的下,通过对类型Object的引用来实现参数的“任意化”,“任意化”带来的缺点是要做显式的强制类型转换,而这种转换是要求开发者对实际参数类型可以预知的情况下进行的。对于强制类型转换错误的情况,编译器可能不提示错误,在运行的时候才出现异常,这是一个安全隐患。
泛型的好处是在编译的时候检查类型安全,并且所有的强制转换都是自动和隐式的,以提高代码的重用率。
二、为什么需要泛型
2.1、编译期对数据类型进行检测
首先我们看一个案例,向一个ArrayList中添加字符串,“不小心”添加了整数,如下面代码,
并没有错误:
但是执行时,会报错:“java.lang.classCastException”
因为ArrayList中维护的是一个Object数组, private transient Object[] elementData;
, 使用get()返回的是一个Object对象, 需要强制转换,但是中间混杂一个Integer数值, 导致强制转换失败。这个错误就是由于Object的任意化导致的。
如果能在编译阶段就发现数据类型有错, 那么就很方便,泛型就满足了这个要求:我将这个程序修改一下,ArrayList使用泛型:会发现编译阶段就报错了.
2.2强制转换是自动的
不使用泛型:
package com.chb.fanxing;public class NoGen { private Object ob; public NoGen(Object ob) { this.ob = ob; } getter setter... private void showType() { System.out.println("数据的实际类型是:" + ob.getClass().getName()); } public static void main(String[] args) { NoGen ngInt = new NoGen(88); ngInt.showType(); int i = (int)ngInt.getOb(); System.out.println("value = " + i); System.out.println("---------------"); NoGen ngStr = new NoGen("88"); ngStr.showType(); String str = (String)ngStr.getOb(); System.out.println("value = " + str); } }
使用泛型:
package com.chb.fanxing;public class Gen<T> { private T ob; public Gen(T ob) { this.ob = ob; } getter setter... private void showType() { System.out.println("T的实际类型:"+ob.getClass().getName()); } public static void main(String[] args) { //定义一个Integer版本 Gen<Integer> genInt = new Gen<Integer>(88); genInt.showType(); int i = genInt.getOb();//此处不用强制转换 System.out.println("value = " + i); System.out.println("----------------------"); Gen<String> genStr = new Gen<String>("88"); genStr.showType(); String str = genStr.getOb(); System.out.println("value = "+str); } }
运行结果:
两个例子的运行结果是一致的
数据的实际类型是:java.lang.Integervalue = 88 ---------------数据的实际类型是:java.lang.String value = 88
对比两个例子会发现:
使用泛型,强制转换时自动进行的:
int i = genInt.getOb();//此处不用强制转换
而不使用泛型,必须要进行手动强制转化
int i = (int)ngInt.getOb();
三、深入泛型
3.1 、有两个类,我们需要打印他们的成员变量
class StringDemo { private String s; public StringDemo (String s) { this.s = s; } setter geter.... } class DoubleDemo{ private Double d; public DoubleDemo(Double d) { this.d = d; } setter getter... }
3.2、重构
仔细观察两个类功能基本一致,只是数据类型不一样,考虑到重构,因为Object是所有类的基类,所以可以使用Object作为成员变量,这样代码就可以通用了。重构代码如下:
class ObjectDemo{ private Object ob; public ObjectDemo(Object ob){ this.ob = ob; } setter getter... }
ObjectDemo测试:
public static void ObjectDemoTest(){ ObjectDemo strOD = new ObjectDemo("123"); ObjectDemo dOD = new ObjectDemo(new Double(23)); ObjectDemo od = new ObjectDemo(new Object()); System.out.println((String)strOD.getOb()); System.out.println((Double)dOD.getOb()); System.out.println(od.getOb()); }
运行结果:
3.3使用泛型重构
发现上面的ObjectDemoTest()
中必须要使用强制转换,这比较麻烦,我们还必须事先知道要转换的数据类型,才能进行正确的转换,否则,会出现错误, 业务编译时没有问题,但是一运行,会出现”classCastException”。所以我们需要不用自己进行强制转换,这是泛型就尤为重要。
class GenDemo<T>{ private T t; public GenDemo(T t) { this.t = t; } public void setT(T t) { this.t = t; } public T getT() { return t; } }
测试:省去了手动进行强制转换
public static void GenTest() { GenDemo<String> strOD = new GenDemo<String>("123"); GenDemo<Double> dOD = new GenDemo<Double>(new Double(23)); GenDemo<Object> od = new GenDemo<Object>(new Object()); System.out.println(strOD.getT()); System.out.println(dOD.getT()); System.out.println(od.getT()); }
下面解释一下上面的泛型语法:
使用表示一个类型持有者名称, 相当于一个形参,数据的类型是有实际传入的数据的类型决定,然后T作为成员、参数、方法的返回值的类型。
T仅仅是一个名字,可以随意取的。
class GenDemo , T没有进行任何限制, 实际相当于 Object,
等同于 class GenDemo。
与Object相比,使用泛型所定义的类,在定义和声明,可以使用1b41d1029b94e4c9dab4ab0d953b6030来制定真实的数据类型,如:
GenDemoeafb63d086dd6c9bd19609d76bcc2869 dOD = new GenDemoeafb63d086dd6c9bd19609d76bcc2869(new Double(23));
也可以不指定,那么就需要进行强制转换。
下文我们将继续java的泛型
限制泛型
多接口限制
通配符泛型
一、泛型简介
泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数。
在Java SE 1.5之前,没有泛型的情况的下,通过对类型Object的引用来实现参数的“任意化”,“任意化”带来的缺点是要做显式的强制类型转换,而这种转换是要求开发者对实际参数类型可以预知的情况下进行的。对于强制类型转换错误的情况,编译器可能不提示错误,在运行的时候才出现异常,这是一个安全隐患。
泛型的好处是在编译的时候检查类型安全,并且所有的强制转换都是自动和隐式的,以提高代码的重用率。
二、为什么需要泛型
2.1、编译期对数据类型进行检测
首先我们看一个案例,向一个ArrayList中添加字符串,“不小心”添加了整数,如下面代码,
并没有错误:
但是执行时,会报错:“java.lang.classCastException”
因为ArrayList中维护的是一个Object数组, private transient Object[] elementData;
, 使用get()返回的是一个Object对象, 需要强制转换,但是中间混杂一个Integer数值, 导致强制转换失败。这个错误就是由于Object的任意化导致的。
如果能在编译阶段就发现数据类型有错, 那么就很方便,泛型就满足了这个要求:我将这个程序修改一下,ArrayList使用泛型:会发现编译阶段就报错了.
2.2强制转换是自动的
不使用泛型:
package com.chb.fanxing;public class NoGen { private Object ob; public NoGen(Object ob) { this.ob = ob; } getter setter... private void showType() { System.out.println("数据的实际类型是:" + ob.getClass().getName()); } public static void main(String[] args) { NoGen ngInt = new NoGen(88); ngInt.showType(); int i = (int)ngInt.getOb(); System.out.println("value = " + i); System.out.println("---------------"); NoGen ngStr = new NoGen("88"); ngStr.showType(); String str = (String)ngStr.getOb(); System.out.println("value = " + str); } }
使用泛型:
package com.chb.fanxing;public class Gen<T> { private T ob; public Gen(T ob) { this.ob = ob; } getter setter... private void showType() { System.out.println("T的实际类型:"+ob.getClass().getName()); } public static void main(String[] args) { //定义一个Integer版本 Gen<Integer> genInt = new Gen<Integer>(88); genInt.showType(); int i = genInt.getOb();//此处不用强制转换 System.out.println("value = " + i); System.out.println("----------------------"); Gen<String> genStr = new Gen<String>("88"); genStr.showType(); String str = genStr.getOb(); System.out.println("value = "+str); } }
运行结果:
两个例子的运行结果是一致的
数据的实际类型是:java.lang.Integervalue = 88 ---------------数据的实际类型是:java.lang.String value = 88
对比两个例子会发现:
使用泛型,强制转换时自动进行的:
int i = genInt.getOb();//此处不用强制转换
而不使用泛型,必须要进行手动强制转化
int i = (int)ngInt.getOb();
三、深入泛型
3.1 、有两个类,我们需要打印他们的成员变量
class StringDemo { private String s; public StringDemo (String s) { this.s = s; } setter geter.... } class DoubleDemo{ private Double d; public DoubleDemo(Double d) { this.d = d; } setter getter... }
3.2、重构
仔细观察两个类功能基本一致,只是数据类型不一样,考虑到重构,因为Object是所有类的基类,所以可以使用Object作为成员变量,这样代码就可以通用了。重构代码如下:
class ObjectDemo{ private Object ob; public ObjectDemo(Object ob){ this.ob = ob; } setter getter... }
ObjectDemo测试:
public static void ObjectDemoTest(){ ObjectDemo strOD = new ObjectDemo("123"); ObjectDemo dOD = new ObjectDemo(new Double(23)); ObjectDemo od = new ObjectDemo(new Object()); System.out.println((String)strOD.getOb()); System.out.println((Double)dOD.getOb()); System.out.println(od.getOb()); }
运行结果:
3.3使用泛型重构
发现上面的ObjectDemoTest()
中必须要使用强制转换,这比较麻烦,我们还必须事先知道要转换的数据类型,才能进行正确的转换,否则,会出现错误, 业务编译时没有问题,但是一运行,会出现”classCastException”。所以我们需要不用自己进行强制转换,这是泛型就尤为重要。
class GenDemo<T>{ private T t; public GenDemo(T t) { this.t = t; } public void setT(T t) { this.t = t; } public T getT() { return t; } }
测试:省去了手动进行强制转换
public static void GenTest() { GenDemo<String> strOD = new GenDemo<String>("123"); GenDemo<Double> dOD = new GenDemo<Double>(new Double(23)); GenDemo<Object> od = new GenDemo<Object>(new Object()); System.out.println(strOD.getT()); System.out.println(dOD.getT()); System.out.println(od.getT()); }
下面解释一下上面的泛型语法:
使用表示一个类型持有者名称, 相当于一个形参,数据的类型是有实际传入的数据的类型决定,然后T作为成员、参数、方法的返回值的类型。
T仅仅是一个名字,可以随意取的。
class GenDemo , T没有进行任何限制, 实际相当于 Object,
等同于 class GenDemo。
与Object相比,使用泛型所定义的类,在定义和声明,可以使用1b41d1029b94e4c9dab4ab0d953b6030来制定真实的数据类型,如:
GenDemoeafb63d086dd6c9bd19609d76bcc2869 dOD = new GenDemoeafb63d086dd6c9bd19609d76bcc2869(new Double(23));
也可以不指定,那么就需要进行强制转换。
以上就是java之泛型的内容,更多相关内容请关注PHP中文网(www.php.cn)!

本文讨论了使用Maven和Gradle进行Java项目管理,构建自动化和依赖性解决方案,以比较其方法和优化策略。

本文使用Maven和Gradle之类的工具讨论了具有适当的版本控制和依赖关系管理的自定义Java库(JAR文件)的创建和使用。

本文讨论了使用咖啡因和Guava缓存在Java中实施多层缓存以提高应用程序性能。它涵盖设置,集成和绩效优势,以及配置和驱逐政策管理最佳PRA

本文讨论了使用JPA进行对象相关映射,并具有高级功能,例如缓存和懒惰加载。它涵盖了设置,实体映射和优化性能的最佳实践,同时突出潜在的陷阱。[159个字符]

Java的类上载涉及使用带有引导,扩展程序和应用程序类负载器的分层系统加载,链接和初始化类。父代授权模型确保首先加载核心类别,从而影响自定义类LOA

本文解释了用于构建分布式应用程序的Java的远程方法调用(RMI)。 它详细介绍了接口定义,实现,注册表设置和客户端调用,以解决网络问题和安全性等挑战。

本文详细介绍了用于网络通信的Java的套接字API,涵盖了客户服务器设置,数据处理和关键考虑因素,例如资源管理,错误处理和安全性。 它还探索了性能优化技术,我

本文详细介绍了创建自定义Java网络协议。 它涵盖协议定义(数据结构,框架,错误处理,版本控制),实现(使用插座),数据序列化和最佳实践(效率,安全性,维护


热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

螳螂BT
Mantis是一个易于部署的基于Web的缺陷跟踪工具,用于帮助产品缺陷跟踪。它需要PHP、MySQL和一个Web服务器。请查看我们的演示和托管服务。

ZendStudio 13.5.1 Mac
功能强大的PHP集成开发环境

SublimeText3汉化版
中文版,非常好用

PhpStorm Mac 版本
最新(2018.2.1 )专业的PHP集成开发工具

SecLists
SecLists是最终安全测试人员的伙伴。它是一个包含各种类型列表的集合,这些列表在安全评估过程中经常使用,都在一个地方。SecLists通过方便地提供安全测试人员可能需要的所有列表,帮助提高安全测试的效率和生产力。列表类型包括用户名、密码、URL、模糊测试有效载荷、敏感数据模式、Web shell等等。测试人员只需将此存储库拉到新的测试机上,他就可以访问到所需的每种类型的列表。