Java泛型教程–示例类,接口,方法,通配符等

泛型是Java编程的核心功能之一,它是Java 5中引入的。如果您使用的是Java Collections ,并且版本5或更高版本,则可以肯定使用了它。 在集合类中使用泛型非常容易,但是它提供了比仅创建集合类型更多的功能,我们将在本文中尝试学习泛型的功能。 如果我们使用专业术语,对泛型的理解有时会变得混乱,因此,我将尽量保持其简单易懂。
在本教程中,我们将研究泛型的以下主题。 java-generics-tutorial

  1. Java泛型示例
  2. 具有类和接口的泛型
  3. 泛型类型命名约定
  4. 方法和构造函数中的泛型
  5. 泛型有界类型参数
  6. 泛型与继承
  7. 通用类和子类型
  8. 通用通配符
    1. 泛型上界通配符
    2. 泛型无界通配符
    3. 泛型下界通配符
  9. 使用泛型通配符进行子类型化
  10. 类型擦除

Java泛型示例

Java 5中添加了泛型,以提供编译时类型检查并消除使用集合类时常见的ClassCastException风险。 整个收集框架都进行了重写,以使用泛型进行类型安全。 让我们看看泛型如何帮助我们安全地使用集合类。

List list = new ArrayList();
list.add("abc");
list.add(new Integer(5)); //OKfor(Object obj : list){String str=(String) obj; //type casting leading to ClassCastException at runtime
}

上面的代码可以很好地编译,但是在运行时会抛出ClassCastException,因为我们试图将列表中的Object强制转换为String,而其中一个元素是Integer类型。 在Java 5之后,我们使用如下收集类。

List<String> list1 = new ArrayList<String>(); // java 7 ? List<String> list1 = new ArrayList<>(); 
list1.add("abc");
//list1.add(new Integer(5)); //compiler errorfor(String str : list1){//no type casting needed, avoids ClassCastException
}

请注意,在创建列表时,我们已指定列表中元素的类型为String。 因此,如果我们尝试在列表中添加任何其他类型的对象,则该程序将引发编译时错误。 还要注意,在for循环中,我们不需要类型转换列表中的元素,因此在运行时删除了ClassCastException。

具有类和接口的泛型

我们可以使用泛型类型定义自己的类和接口。 泛型类型是通过类型进行参数化的类或接口。 我们使用尖括号(<>)来指定type参数。

为了了解其好处,可以说我们有一个简单的类:

package com.journaldev.generics;public class GenericsTypeOld {private Object t;public Object get() {return t;}public void set(Object t) {this.t = t;}public static void main(String args[]){GenericsTypeOld type = new GenericsTypeOld();type.set("Pankaj"); String str = (String) type.get(); //type casting, error prone and can cause ClassCastException}
}

请注意,在使用此类时,我们必须使用类型转换,并且它可以在运行时产生ClassCastException。 现在,我们将使用泛型来重写具有泛型类型的相同类,如下所示。

package com.journaldev.generics;public class GenericsType<T> {private T t;public T get(){return this.t;}public void set(T t1){this.t=t1;}public static void main(String args[]){GenericsType<String> type = new GenericsType<>();type.set("Pankaj"); //validGenericsType type1 = new GenericsType(); //raw typetype1.set("Pankaj"); //validtype1.set(10); //valid and autoboxing support}
}

注意main方法中GenericsType类的使用。 我们不需要进行类型转换,并且可以在运行时删除ClassCastException。 如果我们在创建时未提供类型,则编译器将发出警告,“ GenericsType是原始类型。 泛型类型GenericsType <T>的引用应参数化”。 当我们不提供类型时,该类型将成为Object ,因此它同时允许String和Integer对象,但是我们应始终尝试避免这种情况,因为在处理会产生运行时错误的原始类型时,我们必须使用类型转换。

提示 :我们可以使用@SuppressWarnings("rawtypes")注释来抑制编译器警告,请参阅Java注释教程

还要注意,它支持Java自动装箱 。

可比接口是接口中泛型的一个很好的例子,它写为:

package java.lang;
import java.util.*;public interface Comparable<T> {public int compareTo(T o);
}

以类似的方式,我们可以在接口和类中使用泛型。 我们也可以像Map界面一样具有多个类型参数。 同样,我们也可以为参数化类型提供参数化值,例如new HashMap<String, List<String>>(); 已验证。

泛型类型命名约定

命名约定可以帮助我们轻松地理解代码,拥有命名约定是Java编程语言的最佳实践之一。 因此,泛型也带有它自己的命名约定。 通常,类型参数名称是单个大写字母,以使其易于与Java变量区分开。 最常用的类型参数名称为:

  • E –元素(由Java Collections Framework广泛使用,例如ArrayList,Set等)
  • K –键(在地图中使用)
  • N –数字
  • T –类型
  • V –值(在地图中使用)
  • S,U,V等–第二,第三,第四类型

方法和构造函数中的泛型

有时我们不希望整个类都被参数化,在这种情况下,我们也可以在方法中使用泛型类型。 由于构造函数是一种特殊的方法,因此我们也可以在构造函数中使用泛型类型。

这是一个类,显示方法中的泛型类型的示例。

package com.journaldev.generics;public class GenericsMethods {//Generics in methodpublic static <T> boolean isEqual(GenericsType<T> g1, GenericsType<T> g2){return g1.get().equals(g2.get());}public static void main(String args[]){GenericsType<String> g1 = new GenericsType<>();g1.set("Pankaj");GenericsType<String> g2 = new GenericsType<>();g2.set("Pankaj");boolean isEqual = GenericsMethods.<String>isEqual(g1, g2);//above statement can be written simply asisEqual = GenericsMethods.isEqual(g1, g2);//This feature, known as type inference, allows you to invoke a generic method as an ordinary method, without specifying a type between angle brackets.//Compiler will infer the type that is needed}
}

注意isEqual方法签名显示了在方法中使用泛型类型的语法。 还要注意如何在我们的java程序中使用这些方法。 我们可以在调用这些方法时指定类型,也可以像普通方法一样调用它们。 Java编译器足够聪明,可以确定要使用的变量的类型,这种功能称为类型推断

泛型有界类型参数

假设我们要限制可以在参数化类型中使用的对象的类型,例如在比较两个对象的方法中,并且我们要确保接受的对象是可比较的。 要声明一个有界的类型参数,请列出类型参数的名称,然后列出extends关键字,再加上其上限,类似于下面的方法。

public static <T extends Comparable<T>> int compare(T t1, T t2){return t1.compareTo(t2);}

这些方法的调用与无界方法类似,不同之处在于,如果我们尝试使用任何非Comparable的类,则会引发编译时错误。

绑定类型参数可以与方法以及类和接口一起使用。

泛型也支持多个范围,即<T扩展A&B&C>。 在这种情况下,A可以是接口或类。 如果A是类,则B和C应该是接口。 在多个范围内,我们不能有多个类。

泛型与继承

我们知道,如果A是B的子类,则Java继承允许我们将变量A分配给另一个变量B。因此,我们可能认为可以将A的任何泛型类型分配给B的泛型类型,但事实并非如此。 让我们用一个简单的程序看看。

package com.journaldev.generics;public class GenericsInheritance {public static void main(String[] args) {String str = "abc";Object obj = new Object();obj=str; // works because String is-a Object, inheritance in javaMyClass<String> myClass1 = new MyClass<String>();MyClass<Object> myClass2 = new MyClass<Object>();//myClass2=myClass1; // compilation error since MyClass<String> is not a MyClass<Object>obj = myClass1; // MyClass<T> parent is Object}public static class MyClass<T>{}}

我们不允许将MyClass <String>变量分配给MyClass <Object>变量,因为它们不相关,实际上MyClass <T>的父对象是Object。

通用类和子类型

我们可以通过扩展或实现来泛型一个通用类或接口。 一个类或接口的类型参数与另一类或接口的类型参数之间的关系由extend和Implements子句确定。

例如,ArrayList <E>实现了扩展Collection <E>的List <E>,因此ArrayList <String>是List <String>的子类型,而List <String>是Collection <String>的子类型。

只要不更改type参数,子类型关系就会保留,下面显示了多个type参数的示例。

interface MyList<E,T> extends List<E>{
}

List <String>的子类型可以是MyList <String,Object>,MyList <String,Integer>等。

通用通配符

问号(?)是泛型中的通配符,表示未知类型。 通配符可以用作参数,字段或局部变量的类型,有时还可以用作返回类型。 在调用通用方法或实例化通用类时,不能使用通配符。 在以下各节中,我们将学习上界通配符,下界通配符和通配符捕获。

泛型上界通配符

上限通配符用于在方法中放宽对变量类型的限制。 假设我们要编写一个将返回列表中数字总和的方法,那么我们的实现将是这样的。

public static double sum(List<Number> list){double sum = 0;for(Number n : list){sum += n.doubleValue();}return sum;}

现在,上述实现的问题在于它不适用于List of Integers或Doubles,因为我们知道List <Integer>和List <Double>不相关,这在使用上限通配符很有用时。 我们将泛型通配符与extends关键字和上限类或接口一起使用,这将允许我们传递上限或其子类类型的参数。

可以像下面的程序一样修改上面的实现。

package com.journaldev.generics;import java.util.ArrayList;
import java.util.List;public class GenericsWildcards {public static void main(String[] args) {List<Integer> ints = new ArrayList<>();ints.add(3); ints.add(5); ints.add(10);double sum = sum(ints);System.out.println("Sum of ints="+sum);}public static double sum(List<? extends Number> list){double sum = 0;for(Number n : list){sum += n.doubleValue();}return sum;}
}

就像按照接口编写代码一样,在上述方法中,我们可以使用上限类Number的所有方法。 请注意,对于上界列表,除null之外,我们不允许将任何对象添加到列表中。 如果我们尝试在sum方法内将元素添加到列表中,则该程序将无法编译。

泛型无界通配符

有时,我们希望通用方法适用于所有类型,在这种情况下,可以使用无界通配符。 与使用<?相同 扩展Object>。

public static void printData(List<?> list){for(Object obj : list){System.out.print(obj + "::");}}

我们可以为PrintData方法提供List <String>或List <Integer>或任何其他类型的Object列表参数。 与上限列表类似,我们不允许在列表中添加任何内容。

泛型下界通配符

假设我们要在方法中将整数添加到整数列表中,我们可以将参数类型保持为List <Integer>,但它将与Integers捆绑在一起,而List <Number>和List <Object>也可以容纳整数,因此我们可以使用下限通配符来实现此目的。 我们使用带有super关键字和下限类的泛型通配符(?)来实现此目的。

在这种情况下,我们可以传递下界或下界的任何超级类型作为参数,java编译器允许将下界对象类型添加到列表中。

public static void addIntegers(List<? super Integer> list){list.add(new Integer(50));}

使用泛型通配符进行子类型化

List<? extends Integer> intList = new ArrayList<>();
List<? extends Number>  numList = intList;  // OK. List<? extends Integer> is a subtype of List<? extends Number>

类型擦除

添加了泛型以在编译时提供类型检查,并且在运行时没有使用,因此Java编译器使用类型擦除功能来删除字节码中的所有泛型类型检查代码,并在必要时插入类型转换。 类型擦除可确保不会为参数化类型创建新的类; 因此,泛型不会产生运行时开销。

例如,如果我们有如下通用类;

public class Test<T extends Comparable<T>> {private T data;private Test<T> next;public Test(T d, Test<T> n) {this.data = d;this.next = n;}public T getData() { return this.data; }
}

Java编译器用第一个绑定接口Comparable替换有界类型参数T,如下代码:

public class Test {private Comparable data;private Test next;public Node(Comparable d, Test n) {this.data = d;this.next = n;}public Comparable getData() { return data; }
}

多数民众赞成在Java中的泛型,泛型是一个非常广泛的话题,需要大量时间才能有效地理解和使用它。 本文旨在提供泛型的基本细节,以及如何使用泛型来扩展程序的类型安全性。

参考: Java泛型教程–示例类,接口,方法,通配符以及我们的JCG合作伙伴 Pankaj Kumar在Developer Recipes博客上的更多内容。

翻译自: https://www.javacodegeeks.com/2013/07/java-generics-tutorial-example-class-interface-methods-wildcards-and-much-more.html

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

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

相关文章

html5中音乐播放器怎么写,打造属于自己的音乐播放器 HTML5之audio标签

我的音乐播放器HTML5中增加了Audio和Video标签&#xff0c;这两个标签的用法非常相似。功能却是相当强大&#xff0c;我们先来看一下Audio标签各个浏览器的支持情况。这里用的依然是Can I Use这个在线网站&#xff0c;相信学习前端的同学应该都不陌生。Can I Use我们可以看到&a…

初识react(四) react中异步解决方案之 redux-saga

回顾 初识react(一) 揭开jsx语法和虚拟DOM面纱初识react(二) 实现一个简版的html redux.js的demo初识react(三)在 react中使用redux来实现简版计数器初识react(四) react中异步解决方案之 redux-saga初识react(五) 数据流终极解决方案 dva(零配置) 今天demo是实现一个异步的计…

C# WinFrom 关于MDI

dev是一个牛B 到没边的控件 我们正常用winform做个原始mdi窗体 一点都不好看 但 用的dev只需要一个控件 就可让显示舒服多了 建一个项目 上边放一个 xtraTabbedMdiManager1 一个button1 button1.click如下&#xff1a; Form frm new Form(); frm.MdiParent this; frm.Text &…

Jfinal 文件上传

JFinal上传文件 uploadify 可以在http://www.uploadify.com/ 下载。 在原项目的基础上。 uploadify使用&#xff1a; <input id"file_upload_1" name"file_upload" type"file" multiple"true"> /** param uploader 文件上传方法…

轻量级的开源集成:Apache Camel还是Spring集成?

首先&#xff0c;为全面披露信息&#xff0c;在过去的1.5年中&#xff0c; 我一直担任 FuseSource&#xff08;现为Red Hat&#xff09; 的顾问&#xff0c;为零售&#xff0c;运输&#xff0c;银行/金融等不同行业的大型和小型公司提供SOA和集成项目支持。我的专长是使用该领域…

WePY:在质疑中前进 | 文末福利

WePY 作者介绍 Q: 先介绍一下自己吧~ Gcaufy: 我 2011 年大学毕业之后&#xff0c;阴错阳差的走上了 Web 开发的道路。15 年之前算是自由职业 SOHO 工作&#xff0c;主要给一些国外的大公司做外包系统&#xff0c;更多的是做后端开发。15 年之后以前端工程师的身份加入腾讯&a…

MySQL/MariaDB表表达式(3):视图

视图是表表达式的一种&#xff0c;所以它也是虚拟表。对视图操作的时候会通过语句动态的从表中临时获取数据。 1.创建、修改视图 CREATE [OR REPLACE][ALGORITHM {UNDEFINED | MERGE | TEMPTABLE}]VIEW [IF NOT EXISTS] view_name [(column_list)]AS select_statement[WITH [C…

Event Loop 其实也就这点事

前段时间在网上陆续看了很多关于 Event loop 的文章&#xff0c;看完也就混个眼熟&#xff0c;可能内心深处对这种偏原理的知识有一些抵触心情&#xff0c;看完后也都没有去深入理解。最近在看 Vue 的源码&#xff0c;在读到关于 nextTick 的实现时&#xff0c;总有一种似曾相识…

Kudu系列: Kudu主键选择策略

每个Kudu 表必须设置Pimary Key(unique), 另外Kudu表不能设置secondary index, 经过实际性能测试, 本文给出了选择Kudu主键的几个策略, 测试结果纠正了我之前的习惯认知. 简单介绍测试场景: 表中有一个unqiue字段Id, 另外还有一个日期维度字段histdate, 有三种设置kudu PK的方法…

OSS网页上传和断点续传(OSS配置篇)

OSS网页上传和断点续传主要根据BrowserJS-SDK和相关文档整理而得&#xff0c;快速构建OSS上传应用 一、Bucket设置 浏览器中直接访问OSS需要开通Bucket的CORS设置 将allowed origins设置成 *将allowed methods设置成 PUT, GET, POST, DELETE, HEAD将allowed headers设置成 *将e…

小程序各种姿势实现登录

喜闻乐见的背景时间--由于最近接触小程序比较多&#xff0c;又刚好经历过小程序的自动登录时代以及现在的点击登录时代。结合自己的实践以及观察到其他小程序的做法&#xff0c;就有了这篇小分享~ 本文可能涉及的内容-- 更新 首先感谢shaonialife同学的精彩评论~ 可能由于用词…

BBS-登录

from django.db import models# Create your models here. from django.contrib.auth.models import AbstractUser#用户 class UserInfo(AbstractUser):nidmodels.AutoField(primary_keyTrue)telephonemodels.CharField(max_length32)avatarmodels.FileField(upload_toavatar/,…

使用Mockito和BeanPostProcessors在Spring注入测试双打

我非常确定&#xff0c;如果您曾经使用过Spring并且熟悉单元测试&#xff0c;那么您会遇到与您不想修改的Spring应用程序上下文中注入模拟/间谍&#xff08;测试双打&#xff09;有关的问题。 本文介绍了一种使用Spring组件解决此问题的方法。 项目结构 让我们从项目结构开始&…

二叉搜索时与双向链表python_JZ26-二叉搜索树与双向链表

1、中序遍历&#xff0c;当前结点&#xff0c;以及左侧排好序的双向链表&#xff0c;再调整当前结点的指针指向最前结点/* struct TreeNode {int val;struct TreeNode *left;struct TreeNode *right;TreeNode(int x) :val(x), left(NULL), right(NULL) {} };*/ class Solution …

html右缩进怎么设置,WPS中怎么设置右缩进两个字符?

回答&#xff1a;打开我们的Word文档&#xff0c;调整好我们的文字内容&#xff0c;然后全选我们的文字内容&#xff0c;注意要分段时按下键盘上的回车键另起一行。请点击输入图片描述接着&#xff0c;我们点击顶部菜单栏的“开始”菜单&#xff0c;在开始菜单下面的子菜单中找…

VS2013专业版+QT5.6.3+qt-vs-addin-1.2.5环境搭建

一、工具资料&#xff1a; 1.vs2013专业版地址&#xff1a;http://download.csdn.net/download/u010368556/10238145 2.qt各版本地址&#xff1a;http://download.qt.io/archive/qt/ 3.qt-vs插件地址&#xff1a;http://download.qt.io/archive/vsaddin/ 二、环境搭建过程&…

使用ActiveMQ和HornetQ通过WebSocket通过STOMP轻松进行消息传递

消息传递是用于构建不同级别的分布式软件系统的极其强大的工具。 通常&#xff0c;至少在Java生态系统中&#xff0c;客户端&#xff08;前端&#xff09;从不直接与消息代理&#xff08;或交换&#xff09;进行交互&#xff0c;而是通过调用服务器端&#xff08;后端&#xff…

【laravel】【转发】laravel 导入导出excel文档

1、简介 Laravel Excel 在 Laravel 5 中集成 PHPOffice 套件中的 PHPExcel &#xff0c;从而方便我们以优雅的、富有表现力的代码实现Excel/CSV文件的导入和 导出 。 该项目的GitHub地址是&#xff1a; https://github.com/Maatwebsite/Laravel-Excel 。 本文我们将在Laravel中…

你真的了解css像素嘛?

在日常开发中&#xff0c;px一定是大家接触过最多的css单位&#xff0c;但是你真的了解px嘛&#xff1f;1px在屏幕中到底是多大呢&#xff1f;另外不知道大家有没有过下面这些疑惑: 为什么一个元素在pc上和移动端的物理尺寸不一样&#xff0c;但是两者的视觉效果上却差不多呢&…

mysql for mac中文_mysql for Mac 下创建数据表中文显示为?的解决方法

在我的绝版Mac mini下安装了mysql 5.7版本&#xff0c;实例中&#xff0c;在通过load data 导入数据时发现表中的中文显示为 &#xff1f;通过百度&#xff0c;发现多个版本的解决方法&#xff0c;将其中一个成功解决的方法贴上来&#xff1a;大多方法都是这样&#xff1a;需要…