Java泛型的学习

文章目录

  • 前言
  • 泛型接口
    • 定义实现类时指定泛型具体的数据类型
    • 定义实现类时,泛型与接口相同
    • 泛型接口的示例代码
  • 泛型类
    • 演示代码
  • 泛型方法
    • 演示代码
  • 泛型的通配符
  • 泛型的区间限定
  • 泛型应用
  • 泛型性能
  • 注意事项

前言

泛型的本质就是将数据类型参数化, 普通方法的参数值可以变,但是参数的数据类型是不能变的,那么需要使用不同数据类型参数时,我们通常的做法就是重载方法,而泛型则可以将数据类型也搞成跟参数的值一样可以变的。

所以说泛型表示具体的数据类型未知,泛型其实就是一个参数变量,用来接收数据类型的名称。当我们在定义接口、类、方法时,不知道使用什么数据类型的时候,可以使用泛型。

泛型参数名称可以任何字符,甚至单词,不过通常使用比较多了就是 ETE 表示 ElementT 表示 Type

泛型分为泛型接口、泛型类和泛型方法。泛型接口、泛型类大家都比较熟悉,例如 List 就是泛型接口,而 ArrayList 就是泛型类, 我们经常看到 List<E> 的声明,new ArrayList<E>() 的定义,尖括号里面的 E 就是泛型参数名称,参数值就是你在创建对象的时候指定的具体数据类型,例如你创建集合对象的时候可以指定 EString,或者 Integer,当然也可以指定自己定义的类(例如:CarBean)。在实际的工作中,泛型方法用的比较多,特别是解析自定义数据结构非常有用,类似于 toString 这种场景是百试不爽。

泛型接口

泛型接口中的泛型参数的值是在写实现类的时候指定具体的值,或者在引用实现子类的示例对象时指定具体的值,但是引用实现子类对象时,实现子类也必须是泛型类才行。

定义实现类时指定泛型具体的数据类型

例如,定义一个泛型接口 Interator 和实现类 Scanner 的泛型实现如下:

public interface interator<E> {E next();
}

Scanner 类实现了 Iterator 接口,并指定接口的泛型参数值为 String,所以重写的 next() 方法中的泛型参数值就是 String

public final class Scanner implements Iterator<String> {public String next(){}
}

定义实现类时,泛型与接口相同

在定义实现类时,接口使用什么泛型参数名,实现类就使用什么泛型参数名。其实就是定义一个泛型类,只不过这个泛型的泛型参数名称需要和泛型接口的泛型参数名相同。创建对象的时候确定泛型的具体类型。

注:在创建对象时指定泛型具体的数据类型,那么具体的数据类型应该是保存在所创建的对象中,当访问某个含有泛型参数的变量或者调用某个含有泛型参数的方法时,会把这个数据类型传递过去。

例如,List 接口和 ArrayList 实现类的泛型实现如下:

public interface List<E> {boolean add(E e);E get(int index);
}pubic class ArrayList<E> implements List<E> {public boolean add(E e);public E get(int index);
}

泛型接口的示例代码

泛型接口:

package priv.lwx.javaprac.genericinterface;/*** 泛型接口示例** @author liaowenxiong* @date 2021/11/22 15:02*/public interface GenericInterface<E> {void toString(E e);
}

泛型接口的第一种使用方式:定义实现类时指定泛型具体的数据类型

package priv.lwx.javaprac.genericinterface;/*** 泛型接口的第一种使用方式:* 定义实现类时指定泛型具体的数据类型** @author liaowenxiong* @date 2021/11/22 15:42*/public class GenericInterfaceImpl01 implements GenericInterface<Double>{@Overridepublic void toString(Double aDouble) {System.out.println(aDouble);}
}

泛型接口的第二种使用方式:在定义实现类时,接口使用什么泛型参数名,实现类就使用什么泛型参数名。其实就是定义一个泛型类,只不过这个泛型的泛型参数名称需要和泛型接口的泛型参数名相同。创建对象的时候确定泛型的具体类型。

package priv.lwx.javaprac.genericinterface;/*** 泛型接口的第二种使用方式:* 在定义实现类时,接口使用什么泛型参数名,实现类就* 使用什么泛型参数名。其实就是定义一个泛型类,只不过这个泛型的泛型参数名称需* 要和泛型接口的泛型参数名相同。创建对象的时候确定泛型的具体类型。** @author liaowenxiong* @date 2021/11/22 15:04*/public class GenericInterfaceImpl02<E> implements GenericInterface<E> {@Overridepublic void toString(E e) {System.out.println(e);}
}

测试类代码:

package priv.lwx.javaprac.genericinterface;/*** 泛型接口测试** @author liaowenxiong* @date 2021/11/22 15:31*/public class GenericInterfaceDemo01 {public static void main(String[] args) {/*泛型接口的第一种使用方式测试:定义实现类时指定了具体的泛型类型,那么使用接口类型的变量接收实现类对象,在调用方法时会有风险*/GenericInterface gi1 = new GenericInterfaceImpl01();gi1.toString(123); // 参数类型是Object,编译通过,运行时报错:Integer cannot be cast to Double/*只能使用实现类变量接收对象才能避免风险,参数类型是确定的*/GenericInterfaceImpl01 gi2 = new GenericInterfaceImpl01();gi2.toString(23.99);/*泛型接口第二种使用方式测试:定义实现类时跟随接口的泛型,那么使用接口类型的变量接收实现类对象时,接口指定什么数据类型,那么创建的对象中有关变量或方法的泛型参数就自动使用什么数据类型*/GenericInterface<String> gi3 = new GenericInterfaceImpl02<>();gi3.toString("liaowenxiong");GenericInterfaceImpl02<Double> gi4 = new GenericInterfaceImpl02<>();gi4.toString(2.9);}
}

泛型类

声明定义泛型类,那么在创建泛型类对象时,指定具体的泛型参数值,即具体的数据类型。如果创建对象的时候不指定具体的数据类型,默认数据类型是 Object

例如,ArrayList 类在声明定义的时候不知道集合中会存储什么类型的数据,所以类型使用泛型:

在这里插入图片描述

注:在创建对象时指定泛型具体的数据类型,那么具体的数据类型应该是保存在所创建的对象中,当访问某个含有泛型参数的变量或者调用某个含有泛型参数的方法时,会把这个数据类型传递过去。

演示代码

先声明定义两个 JavaBean 类型:

1.CarBean

public class CarBean {private String brand;private String name;private String price;public CarBean() {}public CarBean(String brand, String name, String price) {this.brand = brand;this.name = name;this.price = price;}@Overridepublic boolean equals(Object o) {if (this == o) return true;if (o == null || getClass() != o.getClass()) return false;CarBean carBean = (CarBean) o;return Objects.equals(brand, carBean.brand) &&Objects.equals(name, carBean.name) &&Objects.equals(price, carBean.price);}@Overridepublic int hashCode() {return Objects.hash(brand, name, price);}public String getBrand() {return brand;}public void setBrand(String brand) {this.brand = brand;}public String getName() {return name;}public void setName(String name) {this.name = name;}public String getPrice() {return price;}public void setPrice(String price) {this.price = price;}
}

2.HouseBean

public class HouseBean {private String brand;private String name;private String price;public HouseBean() {}public HouseBean(String brand, String name, String price) {this.brand = brand;this.name = name;this.price = price;}public String getBrand() {return brand;}public void setBrand(String brand) {this.brand = brand;}public String getName() {return name;}public void setName(String name) {this.name = name;}public String getPrice() {return price;}public void setPrice(String price) {this.price = price;}@Overridepublic String toString() {return "HouseBean{" +"brand='" + brand + '\'' +", name='" + name + '\'' +", prive='" + price + '\'' +'}';}@Overridepublic boolean equals(Object o) {if (this == o) return true;if (o == null || getClass() != o.getClass()) return false;HouseBean houseBean = (HouseBean) o;return brand.equals(houseBean.brand) &&name.equals(houseBean.name) &&price.equals(houseBean.price);}@Overridepublic int hashCode() {return Objects.hash(brand, name, price);}
}

再声明定义一个泛型类:

public class GenericGoods<T> { // 尖括号表示声明泛型参数名(或泛型变量名)为 Tprivate T goods;private String info_goods;public GenericGoods(T goods) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException {this.goods = goods;this.info_goods = setData(goods);}public GenericGoods(T goods, String info_goods) {this.goods = goods;this.info_goods = info_goods;}/*** 声明定义一个方法,将指定的goods对象中的所有成员变量的值取出来,并以“变量名=变量值”这样的格式进行拼接后返回该字符处。* 这里使用泛型声明参数类型,这样就可以接收任意类型的参数,然后利用反射技术获取对象所属类型的变量名和变量值,再拼接成字符串返回*/private String setData(T goods) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {StringBuffer sb = new StringBuffer();String getterMethodName = "";String fieldName = "";Method mObject = null;String info_goods = "";// 获取对象所属类型的全部字段(字段是指类所声明的静态变量和成员变量)Field[] fields = goods.getClass().getDeclaredFields();if (fields.length > 0) {for (Field f : fields) {// 设置允许访问私有域f.setAccessible(true);// 获取字段名称fieldName = f.getName();// 将字段名称按符号“#”进行拼接sb.append(fieldName + "#");// 拼接getter方面名称if (fieldName.length() > 1) {// 组装getter方法名// substring(0,1)表示截取索引值从0到1的字符串,但是不含索引值为1的字符,其实就是截取第一个字符,并且转换成大写// substring(1)表示截取索引值为1后面的所有字符串,包含索引值为1的字符getterMethodName = "get" + fieldName.substring(0, 1).toUpperCase() + fieldName.substring(1);// 将拼接的getter方法名打印出来System.out.println(getterMethodName);} else {// 否则说明字段名称仅有一个字符,直接转换成大写,再拼接 get 即可getterMethodName = "get" + fieldName.toUpperCase();// 将拼接的getter方法名打印出来System.out.println(getterMethodName);}// 获取goods所属类型中指定名称的方法对象mObject = goods.getClass().getMethod(getterMethodName);// 获取对象goods的成员变量的值,并且使用“变量名=变量值&”这样的格式拼接起来// mObject.invoke(goods)是指调用对象goods的方法mObject,其实就是调用getter方法获取相关成员变量的值info_goods += fieldName + "=" + mObject.invoke(goods) + "&";// 截取info_goods中索引值从0到info_goods最后一个索引值的子字符串,不包含最后一个索引值的字符,其实就是将最后一个字符“&”过滤掉info_goods = info_goods.substring(0, info_goods.length() - 1);}// 打印输出所有字段名称System.out.println(sb);}return info_goods;}public T getGoods() {return goods;}public void setGoods(T goods) {this.goods = goods;}public String getInfo_goods() {return info_goods;}public void setInfo_goods(String info_goods) {this.info_goods = info_goods;}@Overridepublic boolean equals(Object o) {if (this == o) return true;if (o == null || getClass() != o.getClass()) return false;GenericGoods<?> that = (GenericGoods<?>) o;return Objects.equals(goods, that.goods) &&Objects.equals(info_goods, that.info_goods);}@Overridepublic int hashCode() {return Objects.hash(goods, info_goods);}
}

main 方法测试:

public class Demo04Generic {public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException {// 创建一个汽车产品CarBean cb = new CarBean();cb.setBrand("奔驰");cb.setName("S60");cb.setPrice("100000.00");// 创建一个通用商品对象,指定泛型参数具体的值,即指定具体的泛型类型,即指定具体的数据类型,那么创建的对象中有使用泛型参数的地方,具体的数据类型就已经明确了GenericGoods<CarBean> gg = new GenericGoods<>(cb); String info_goods = gg.getInfo_goods();System.out.println(info_goods);}
}

泛型方法

泛型方法的定义语法格式:修饰符 <泛型变量名> 返回值类型 方法名(泛型变量名 参数名) {方法体}
调用方法的时候确定泛型参数的值(即具体的数据类型),即调用方法的时候,传递什么类型的参数,泛型就是什么类型。

演示代码

public <T> String toString(T obj) { // 方法所声明的泛型参数名和类的泛型参数名相同,不会造成冲突StringBuffer info_obj = new StringBuffer();StringBuffer fieldNames = new StringBuffer();Method m = null;String getMethodName = "";String fieldName = "";Field[] fields = obj.getClass().getDeclaredFields();System.out.println("对象所属类型是:" + obj.getClass().getName());if (fields.length > 0) for (int i = 0; i < fields.length; i++) {fieldName = fields[i].getName();if (i == fields.length - 1) {fieldNames.append(fieldName);} else {fieldNames.append(fieldName + "#");}if (fieldName.length() == 1) {getMethodName = "get" + fieldName.toUpperCase();} else {getMethodName = "get" + fieldName.substring(0, 1).toUpperCase() + fieldName.substring(1);}try {m = obj.getClass().getMethod(getMethodName);if (i == fields.length - 1) {info_obj.append(fieldName + "=" + m.invoke(obj));} else {info_obj.append(fieldName + "=" + m.invoke(obj) + "&");}} catch (NoSuchMethodException e) {e.printStackTrace();} catch (IllegalAccessException e) {e.printStackTrace();} catch (InvocationTargetException e) {e.printStackTrace();}}}return info_obj.toString();}

main 方法测试代码:

public class Demo05Generic {public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException {HouseBean hb = new HouseBean();hb.setBrand("万科");hb.setName("皇帝宫廷");hb.setPrice("200000.00");GenericGoods<CarBean> gg = new GenericGoods(); // 泛型参数指定了类型 CarBeanSystem.out.println(gg.toString(hb)); // 调方法的时候,依旧根据传参类型来确定泛型参数的值System.out.println("=============================");CarBean cb = new CarBean();cb.setBrand("奔驰");cb.setName("S60");cb.setPrice("200000.00");System.out.println(gg.toString(cb));}
}

泛型的通配符

下面的例子,演示了如何使用泛型的通配符接收不同数据类型的 ArrayList 参数:

public class Demo02Generic {public static void main(String[] args) {ArrayList<Integer> list1 = new ArrayList<>();list1.add(15);list1.add(25);ArrayList<String> list2 = new ArrayList<>();list2.add("迪丽热巴");list2.add("古力娜扎");printArrayList(list1);printArrayList(list2);}// 定义一个方法可以遍历所有数据类型的 Arraylistpublic static void printArrayList(ArrayList<?> list) {//使用迭代器遍历集合Iterator<?> it = list.iterator();while (it.hasNext()) {Object obj = it.next();System.out.println(obj);}}
}

泛型的区间限定

1.泛型的上限限定:? extends E 代表使用的泛型只能是类型 E 的子类或者本身
2.泛型的下限限定:? super E 代表使用的泛型只能是类型 E 的父类或者本身

下面的例子,就是演示了如何使用泛型的区间限定:

public class Demo03Generic {public static void main(String[] args) {Collection<Integer> list1 = new ArrayList<>();Collection<String> list2 = new ArrayList<>();Collection<Number> list3 = new ArrayList<>();Collection<Object> list4 = new ArrayList<>();getElement1(list1);getElement1(list2); // String 不是 Number 的子类,所以报错getElement1(list3);getElement1(list4); // Object 不是 Number 的类,所以报错getElement2(list1); // Integer 不是 Number 的父类,所以报错getElement2(list2); // String 不是 Number 的父类,所以报错getElement2(list3); getElement2(list4);}private static void getElement1(Collection<? extends Number> list1) {}private static void getElement2(Collection<? super Number> list1) {}// 定义一个方法可以遍历所有数据类型的 Arraylistpublic static void printArrayList(ArrayList<?> list) {//使用迭代器遍历集合Iterator<?> it = list.iterator();while (it.hasNext()) {Object obj = it.next();System.out.println(obj);}}
}

泛型应用

在这里插入图片描述

泛型性能

可以参考这篇文章: http://blog.csdn.net/hejiangtao/article/details/7173838

注意事项

1.泛型参数(或叫泛型变量)的值只能是引用类型,不能是基本数据类型

2.泛型参数(或叫泛型变量)可以有多个

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/333340.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

html2canvas截取图片不完整,html2canvas关于图片不能一般截取

题目起首说说遇到了什么题目。起首有这么一个需求。须要前端依据后端传过来数据&#xff0c;动态的天生图片。图片中的案牍、背景图片、用户头像全部都是经由过程后端的接口猎取。然则运用 html2canvas 天生的canvas有些图片胜利的在canvas里天生了。然则有些图片无论如何都显现…

shebang_Shebang来Java了吗?

shebang尽管添加对Unix风格的shebang &#xff08; #! &#xff09;的支持从来不是 JEP 330 [“启动单文件源代码程序”]的主要目标&#xff0c;但围绕JEP 330“单个文件源程序”的潜在功能存在一些问题。为了支持Unix风格的shebang &#xff0c;在OpenJDK jdk-dev邮件列表上引…

beetl模板引擎之自定义html标签,Beetl模板引擎自定义分页标签

1)在beetl.properties文件中配置tagTAG.page xx.xx.xx.xx.PageTag2)在模板中引用标签3)增加标签类继承自GeneralVarTagBinding类import org.beetl.core.Context;import org.beetl.core.GeneralVarTagBinding;import org.beetl.core.statement.Statement;import java.io.IOExce…

Java中常用集合类对比_集合比较

集合实现类集合接口有序性唯一性线程安全空值性能排序数据结构索引应用场景备注ArrayListList有序可重复不安全允许空值查找效率高&#xff0c;添加/删除效率低使用 Collections.sort() 传入ArrayList&#xff0c;会采用默认的方式进行排序&#xff08;字典序&#xff09;。自己…

java nio教程_Java NIO教程

java nio教程1.简介 Java NIO是Java 1.4引入的一个库。 自从Java NIO推出以来&#xff0c;它提供了另一种方法来处理I / O和网络事务。 它被认为是Java网络和Java IO库的替代方法。 开发Java NIO的目的是使输入和输出的事务异步和非阻塞。 阻塞和非阻塞IO的概念将在后面的部分…

qq html制作,jquery学习练习:制作QQ简易聊天框

使用html( )获取和设置页面内容使用val( )获取聊天内容使用addClass( )为指定元素追加样式使用数组保存聊天人员头像和昵称使用随机函数获取聊天人员的头像和昵称在输入框中输入内容&#xff0c;单击“发送”按钮&#xff0c;显示聊天内容&#xff0c;使用addClass( )为聊天内容…

IntelliJ IDEA 删除了快捷键后,如何找回来?

文章目录修改冲突快捷键删除了快捷键&#xff0c;如何找回修改冲突快捷键 删除了快捷键&#xff0c;如何找回 如上图所示&#xff0c;如果你删除了【Restore Default Layout】的快捷键后&#xff0c;如何找回来进行重新设置呢&#xff1f; 你可以按组合键【Command ,】&…

移动端设置html的字体尺寸,移动端开发元素及字体尺寸适配基础知识

刚看了一篇真正教会你开发移动端页面的文章(二)&#xff0c;文章写的很清楚&#xff0c;现总结下他的核心思想及自己所查阅的一些资料&#xff1a;假设设计稿是iPhone6(iPhone6设备像素为750px&#xff1b;设备像素比为2&#xff0c;即其适口尺寸为375px)上有200px的方块box&am…

气味识别应用_解决气味

气味识别应用有时&#xff0c;我们在代码库中可能会遇到一些巨大的if语句。 这些语句必须维护并一次又一次地更改相同的代码块。 在if语句检查变量是否属于某个值范围内的情况下&#xff0c;这也是常见的。 假设您有一个枚举 public enum FoodType {FRUIT,VEGETABLES,RED_MEA…

IntelliJ IDEA for Mac 文件结构侧边窗口/类文件内部结构(File Structure)

文章目录显示或者隐藏类中的属性信息显示或者隐藏类中的非公开信息显示或者隐藏类中的字段信息显示或者隐藏继承自父类的内容显示或者隐藏匿名内部类单击导航的开关始终选择打开的元素通过定义类型进行方法分组类内部结构的节点排序类内部结构窗口可以按组合键【Command 7】打…

java.awt.api_Java SE 11(18.9)中的API更新

java.awt.apiJava SE 11也被命名为18.9&#xff08;基于使用发布年份和月份的新命名方案&#xff09;&#xff0c;预计将在9月的最后一周发布GA。 频繁发布新JDK版本的新方法是允许语言创建者引入新功能&#xff0c;并向开发人员社区更快地更新API。 通常&#xff0c;API更新不…

卡西欧82es计算机怎么玩游戏,卡西欧计算器fx82es-如何使用卡西欧fx-82es计算器计算矩阵 – 手机爱问...

2009-12-24有关于卡西欧计算器的问题卡西欧计算机中&#xff0c;坐标变换计算方法:如将极坐标中距离为2&#xff0c;角度为60度的点&#xff0c;转换为直角坐标&#xff0c;可参考方式&#xff1a;选定角度单位&#xff0c;如选定“DEG”即使用“度、分、秒”&#xff1b;输入距…

Java中,为什么子类的构造方法中必须调父类的构造方法?

1.有父才会有子&#xff0c;即便父类什么东西都没有&#xff0c;也必须调父类的构造方法&#xff0c;这样符合现实的生命传承关系&#xff0c;符合 Java 的面向对象和继承的思想 2.如果父类有一些变量&#xff0c;那么调用父类的构造方法&#xff0c;就可以通知 JVM 把父类加载…

java批处理 异常处理_Java批处理教程

java批处理 异常处理在当今世界&#xff0c;互联网已经改变了我们的生活方式&#xff0c;其主要原因之一是大部分日常琐事都使用互联网。 这导致大量数据可用于处理。 其中涉及大量数据的一些示例是处理工资单&#xff0c;银行对帐单&#xff0c;利息计算等。因此&#xff0c;…

html 按钮防止多次提交,HTML点击提交按钮两次

我最近添加了一个CSS3焦点样式到一个表单的元素(它垂直增长&#xff0c;给更多的空间来写)。然而&#xff0c;现在当用户点击提交按钮时&#xff0c;textarea会失去焦点(和缩小)&#xff0c;但表单不会提交&#xff0c;用户必须再次点击提交。有没有解决这个问题的方法&#xf…

Java的子类可以继承父类的私有变量和私有方法吗?

答&#xff1a;实际上继承了。 父类的成员变量&#xff08;包含私有变量&#xff09;和成员方法地址&#xff08;包含私有成员方法&#xff09;都存放在子类对象中。父类的私有变量和私有方法虽然在子类对象中&#xff0c;但是对于子类对象而言是不可见的&#xff0c;无法直接…

html关于超链接的问题,关于超链接的一些问题

xhtml css页面制作过程中问题的解决方案&#xff0c;说是解决方案应该有点过了&#xff0c;充其量只不过是给刚刚开始学标准页面制作的朋友们的一些小建议&#xff0c;如 2018-04-14XHTML里面的img标记应该这样写&#xff1a;&#xff0c;这种写法也就是所谓的自关闭&#xff0…

java中无限大_Java 9中的无限集

java中无限大一套 甲Set是元素的集合&#xff0c;从而在任何给定的元件Set只出现一次。 更正式地说&#xff0c;集合不包含元素e1和e2对&#xff0c;因此e1.equals(e2) 。 我们可以像这样在Java 9中轻松创建Set &#xff1a; final Set<Integer> s Set.of(1, 2, 3); …

Java中父类的私有数据和静态数据在内存中是如何存储的?

1.父类的成员变量&#xff08;包含私有变量&#xff09;和成员方法地址&#xff08;包含私有成员方法&#xff09;都存放在子类对象中。父类的私有变量和私有方法虽然在子类对象中&#xff0c;但是对于子类对象而言是不可见的&#xff0c;无法直接访问和调用&#xff0c;必须通…

在计算机中 用户程序,在用户计算机上的万维网客户程序是()

摘要&#xff1a;修理需检后不测&#xff0c;户计户程行修即进应立理&#xff0c;修理自行解体铅封拆动。维网客注意简述计的基本及其R引原则要点物设。和鱼观察态的动水色&#xff0c;户计户程天早塘一养殖应每晚巡池塘次。...修理需检后不测&#xff0c;户计户程行修即进应立…