一文玩转Java 泛型知识
创始人
2024-04-11 14:23:19

✅作者简介:热爱国学的Java后端开发者,修心和技术同步精进。
🍎个人主页:Java Fans的博客
🍊个人信条:不迁怒,不贰过。小知识,大智慧。
💞当前专栏:前端开发者成长之路
✨特色专栏:国学周更-心性养成之路
🥭本文内容:一文玩转Java 泛型知识

文章目录

    • 泛型定义
    • 泛型作用
    • 泛型使用
      • 泛型方法
      • 泛型类
      • 泛型接口
    • 泛型通配符

在这里插入图片描述

泛型定义

  Java 泛型(generics)是 JDK 5 中引入的一个新特性, 泛型提供了编译时类型安全检测机制,该机制允许程序员在编译时检测到非法的类型。

  泛型的本质是参数化类型,就是将类型由原来的具体的类型参数化,然后在使用时再指定此参数具体的值,那样这个类型就可以在使用时决定了。这种参数类型可以用在类、接口、方法中,分别称为泛型类、泛型接口、泛型方法。

假定我们有这样一个需求:写一个排序方法,能够对整型数组、字符串数组甚至其他任何类型的数组进行排序,该如何实现?
~
答案是可以使用 Java 泛型
~
使用 Java 泛型的概念,我们可以写一个泛型方法来对一个对象数组排序。然后,调用该泛型方法来对整型数组、浮点数数组、字符串数组等进行排序

在这里插入图片描述

泛型作用

  泛型的好处是在编译的时候检查类型安全,并且所有的强制转换都是自动和隐式的,提高代码的重用率。
在这里插入图片描述
【1】安全性

  泛型的主要目标是提高 Java 程序的类型安全。通过知道使用泛型定义的变量的类型限制,编译器可以在一个高得多的程度上验证类型假设。没有泛型,这些假设就只存在于程序员的头脑中(或者如果幸运的话,还存在于代码注释中)。

比如:没有泛型的情况下使用集合:

public static void arrayList() {ArrayList list = new ArrayList();list.add("好好学习,天天向上");list.add(5201314); //编译正常
}

有泛型的情况下使用集合:

public static void arrayList() {ArrayList list = new ArrayList<>();list.add("好好学习,天天向上"); list.add(5201314); //编译不通过 
}

  有了泛型后,定义好的集合list在编译的时候add(5201314)就会编译不通过。

  相当于告诉编译器每个集合接收的对象类型是什么,编译器在编译期就会做类型检查,告知是否插入了错误类型的对象,使得程序更加安全,增强了程序的健壮性。

【2】消除转换

  消除强制类型转换。 泛型的一个附带好处是,消除源代码中的许多强制类型转换。这使得代码更加可读,并且减少了出错机会。

比如,以下没有泛型的代码段需要强制转换:

List list = new ArrayList();list.add("hello");String s = (String) list.get(0);

当重写为使用泛型时,代码不需要强制转换:

List list = new ArrayList();list.add("hello");String s = list.get(0); // no cast

【3】提升性能

  避免了不必要的装箱、拆箱操作,提高程序的性能,在非泛型编程中,将筒单类型作为Object传递时会引起Boxing(装箱)和Unboxing(拆箱)操作,这两个过程都是具有很大开销的。引入泛型后,就不必进行Boxing和Unboxing操作了,所以运行效率相对较高,特别在对集合操作非常频繁的系统中,这个特点带来的性能提升更加明显。

  泛型变量固定了类型,使用的时候就已经知道是值类型还是引用类型,避免了不必要的装箱、拆箱操作。

使用泛型前:

//由于是object类型,会自动进行装箱操作。
object a=1;//强制转换,拆箱操作。这样一去一来,当次数多了以后会影响程序的运行效率。 
int b=(int)a;

使用泛型后:

public static T GetValue(T a)
{return a;
}public static void Main()
{//使用这个方法的时候已经指定了类型是int,所以不会有装箱和拆箱的操作。int b=GetValue(1);
}

【4】重用性

  对于泛型的理解不应该只停留于Java 5之后提供的泛型的特性,只停留在可以使用菱形运算符或者带限制的通配符上面,对于泛型的理解应该基于泛型的思想,即对于代码的重用性的提高上面来。

  比如可以使用Object类来表示泛型,使用接口类型表示泛型等。

泛型使用

  泛型有三种使用方式,分别为:泛型方法、泛型类和泛型接口。
在这里插入图片描述

泛型方法

  你可以写一个泛型方法,该方法在调用时可以接收不同类型的参数。根据传递给泛型方法的参数类型,编译器适当地处理每一个方法调用。

定义泛型方法的规则:

  所有泛型方法声明都有一个类型参数声明部分(由尖括号分隔),该类型参数声明部分在方法返回类型之前(在下面例子中的 )。

  每一个类型参数声明部分包含一个或多个类型参数,参数间用逗号隔开。一个泛型参数,也被称为一个类型变量,是用于指定一个泛型类型名称的标识符。类型参数能被用来声明返回值类型,并且能作为泛型方法得到的实际参数类型的占位符。

  泛型方法体的声明和其他方法一样。注意类型参数只能代表引用型类型,不能是原始类型(像 int、double、char 等)。

java 中泛型标记符:

  • E - Element (在集合中使用,因为集合中存放的是元素)
  • T - Type(Java 类)
  • K - Key(键)
  • V - Value(值)
  • N - Number(数值类型)
  • ? - 表示不确定的 java 类型

在这里插入图片描述

下面的例子演示了如何使用泛型方法打印不同类型的数组元素:

public class GenericMethodTest
{// 泛型方法 printArray                         public static < E > void printArray( E[] inputArray ){// 输出数组元素            for ( E element : inputArray ){        System.out.printf( "%s ", element );}System.out.println();}public static void main( String args[] ){// 创建不同类型数组: Integer, Double 和 CharacterInteger[] intArray = { 1, 2, 3, 4, 5 };Double[] doubleArray = { 1.1, 2.2, 3.3, 4.4 };Character[] charArray = { 'H', 'E', 'L', 'L', 'O' };System.out.println( "整型数组元素为:" );printArray( intArray  ); // 传递一个整型数组System.out.println( "\n双精度型数组元素为:" );printArray( doubleArray ); // 传递一个双精度型数组System.out.println( "\n字符型数组元素为:" );printArray( charArray ); // 传递一个字符型数组} 
}

编译以上代码,运行结果如下所示:

整型数组元素为:
1 2 3 4 5

双精度型数组元素为:
1.1 2.2 3.3 4.4

字符型数组元素为:
H E L L O

有界的类型参数:

  可能有时候,你会想限制那些被允许传递到一个类型参数的类型种类范围。例如,一个操作数字的方法可能只希望接受Number或者Number子类的实例。这就是有界类型参数的目的。

  要声明一个有界的类型参数,首先列出类型参数的名称,后跟extends关键字,最后紧跟它的上界。

  下面的例子演示了"extends"如何使用在一般意义上的意思"extends"(类)或者"implements"(接口)。该例子中的泛型方法返回三个可比较对象的最大值。

public class MaximumTest
{// 比较三个值并返回最大值public static > T maximum(T x, T y, T z){                     T max = x; // 假设x是初始最大值if ( y.compareTo( max ) > 0 ){max = y; //y 更大}if ( z.compareTo( max ) > 0 ){max = z; // 现在 z 更大           }return max; // 返回最大对象}public static void main( String args[] ){System.out.printf( "%d, %d 和 %d 中最大的数为 %d\n\n",3, 4, 5, maximum( 3, 4, 5 ) );System.out.printf( "%.1f, %.1f 和 %.1f 中最大的数为 %.1f\n\n",6.6, 8.8, 7.7, maximum( 6.6, 8.8, 7.7 ) );System.out.printf( "%s, %s 和 %s 中最大的数为 %s\n","pear","apple", "orange", maximum( "pear", "apple", "orange" ) );}
}

编译以上代码,运行结果如下所示:

3, 4 和 5 中最大的数为 5

6.6, 8.8 和 7.7 中最大的数为 8.8

pear, apple 和 orange 中最大的数为 pear

泛型类

  泛型类的声明和非泛型类的声明类似,除了在类名后面添加了类型参数声明部分。

  和泛型方法一样,泛型类的类型参数声明部分也包含一个或多个类型参数,参数间用逗号隔开。一个泛型参数,也被称为一个类型变量,是用于指定一个泛型类型名称的标识符。因为他们接受一个或多个参数,这些类被称为参数化的类或参数化的类型。
在这里插入图片描述

如下实例演示了我们如何定义一个泛型类:

public class Box {private T t;public void add(T t) {this.t = t;}public T get() {return t;}public static void main(String[] args) {Box integerBox = new Box();Box stringBox = new Box();integerBox.add(new Integer(10));stringBox.add(new String("Java泛型"));System.out.printf("整型值为 :%d\n\n", integerBox.get());System.out.printf("字符串为 :%s\n", stringBox.get());}
}

编译以上代码,运行结果如下所示:

整型值为 :10

字符串为 :Java泛型

泛型接口

  泛型接口的定义和泛型类的定义一样,区别就在于一个是接口需要实现各个接口方法,一个是类,需要实现对应的抽象方法。

1、在定义一个接口的时候如果某些类型不能确定,那么就使用占位符标记,在具体使用的时候再指定泛型的类型。

2、接口的泛型常用的使用方式:

  • 直接在实现类中指定泛型的具体类型
  • 在实现类中继续使用泛型,在实例化实现类对象的时候指定泛型的具体类型
  • 在接口继承接口中指定泛型的具体类型。

在这里插入图片描述

案例代码:

public class GenericTypeInterface {public static void main(String[] args) {Person p1 = new Person("tom", 16, 5000);Person p2 = new Person("jack", 18, 4000);System.out.println(p1.isBetter(p2));}
}/** 泛型接口*/
interface CompareInterface {public boolean isBetter(T t);
}//指定T为Person类型
class Person implements CompareInterface { //属性private String name;private int age;private int salary;//构造方法public Person() {super();}public Person(String name, int age, int salary) {super();this.name = name;this.age = age;this.salary = salary;}//getter和setterpublic String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public int getSalary() {return salary;}public void setSalary(int salary) {this.salary = salary;}// t为Person类型@Overridepublic boolean isBetter(Person t) { if (this.age < t.getAge() && this.salary > t.getSalary()) {return true;}return false;}}

泛型通配符

  1、泛型通配符一般是使用 ? 代替具体的类型参数。例如 List 在逻辑上是 List,List 等所有 List<具体类型实参> 的父类。

案例代码:

import java.util.*;public class GenericTest {public static void main(String[] args) {List name = new ArrayList();List age = new ArrayList();List number = new ArrayList();name.add("icon");age.add(18);number.add(314);getData(name);getData(age);getData(number);}public static void getData(List data) {System.out.println("data :" + data.get(0));}
}

输出结果为:
data :icon
data :18
data :314

  解析: 因为 getData() 方法的参数是 List 类型的,所以 name,age,number 都可以作为这个方法的实参,这就是通配符的作用。

  2、泛型通配符上限通过形如List来定义,如此定义就是通配符泛型值接受Number及其下层子类类型。

案例代码:

import java.util.*;public class GenericTest {public static void main(String[] args) {List name = new ArrayList();List age = new ArrayList();List number = new ArrayList();name.add("icon");age.add(18);number.add(314);//getUperNumber(name);//1getUperNumber(age);//2getUperNumber(number);//3}public static void getData(List data) {System.out.println("data :" + data.get(0));}public static void getUperNumber(List data) {System.out.println("data :" + data.get(0));}
}

输出结果:
data :18
data :314

  解析: 在 //1 处会出现错误,因为 getUperNumber() 方法中的参数已经限定了参数泛型上限为 Number,所以泛型为 String 是不在这个范围之内,所以会报错。

  3、泛型通配符下限通过形如 List 来定义,表示类型只能接受 Number 及其上层父类类型,如 Object 类型的实例。

在这里插入图片描述

相关内容

热门资讯

埃菲尔铁塔在哪 中国仿建埃菲尔... 2019年4月26日,广西南宁市,街头惊现一座巨型山寨版埃菲尔铁塔,高约20米,白色塔身,造型逼真,...
苗族的传统节日 贵州苗族节日有... 【岜沙苗族芦笙节】岜沙,苗语叫“分送”,距从江县城7.5公里,是世界上最崇拜树木并以树为神的枪手部落...
北京的名胜古迹 北京最著名的景... 北京从元代开始,逐渐走上帝国首都的道路,先是成为大辽朝五大首都之一的南京城,随着金灭辽,金代从海陵王...
长白山自助游攻略 吉林长白山游... 昨天介绍了西坡的景点详细请看链接:一个人的旅行,据说能看到长白山天池全凭运气,您的运气如何?今日介绍...
应用未安装解决办法 平板应用未... ---IT小技术,每天Get一个小技能!一、前言描述苹果IPad2居然不能安装怎么办?与此IPad不...
脚上的穴位图 脚面经络图对应的... 人体穴位作用图解大全更清晰直观的标注了各个人体穴位的作用,包括头部穴位图、胸部穴位图、背部穴位图、胳...
猫咪吃了塑料袋怎么办 猫咪误食... 你知道吗?塑料袋放久了会长猫哦!要说猫咪对塑料袋的喜爱程度完完全全可以媲美纸箱家里只要一有塑料袋的响...
demo什么意思 demo版本... 618快到了,各位的小金库大概也在准备开闸放水了吧。没有小金库的,也该向老婆撒娇卖萌服个软了,一切只...
世界上最漂亮的人 世界上最漂亮... 此前在某网上,选出了全球265万颜值姣好的女性。从这些数量庞大的女性群体中,人们投票选出了心目中最美...
埃菲尔铁塔在哪 中国仿建埃菲尔... 2019年4月26日,广西南宁市,街头惊现一座巨型山寨版埃菲尔铁塔,高约20米,白色塔身,造型逼真,...
苗族的传统节日 贵州苗族节日有... 【岜沙苗族芦笙节】岜沙,苗语叫“分送”,距从江县城7.5公里,是世界上最崇拜树木并以树为神的枪手部落...
北京的名胜古迹 北京最著名的景... 北京从元代开始,逐渐走上帝国首都的道路,先是成为大辽朝五大首都之一的南京城,随着金灭辽,金代从海陵王...
长白山自助游攻略 吉林长白山游... 昨天介绍了西坡的景点详细请看链接:一个人的旅行,据说能看到长白山天池全凭运气,您的运气如何?今日介绍...
世界上最漂亮的人 世界上最漂亮... 此前在某网上,选出了全球265万颜值姣好的女性。从这些数量庞大的女性群体中,人们投票选出了心目中最美...
应用未安装解决办法 平板应用未... ---IT小技术,每天Get一个小技能!一、前言描述苹果IPad2居然不能安装怎么办?与此IPad不...
脚上的穴位图 脚面经络图对应的... 人体穴位作用图解大全更清晰直观的标注了各个人体穴位的作用,包括头部穴位图、胸部穴位图、背部穴位图、胳...
demo什么意思 demo版本... 618快到了,各位的小金库大概也在准备开闸放水了吧。没有小金库的,也该向老婆撒娇卖萌服个软了,一切只...
猫咪吃了塑料袋怎么办 猫咪误食... 你知道吗?塑料袋放久了会长猫哦!要说猫咪对塑料袋的喜爱程度完完全全可以媲美纸箱家里只要一有塑料袋的响...