java中的泛型【T】与通配符【?】概念入门

使用泛型的目的是利用Java编译机制,在编译过程中帮我们检测代码中不规范的有可能导致程序错误的代码。例如,我们都知道List容器可以持有任何类型的数据,所以我们可以把String和Integer等类型同时放入同一个List容器中,但是这种做法是极其危险的。在泛型机制中,这种操作就会导致编译不通过,会强制要求你将List容器中的数据类型修改为统一类型。这种机制可以帮助我们减少程序运行中隐藏的Bug。

泛型【T】

泛型在代码中使用广泛。

泛型的用法

根据泛型使用的位置,即使用在类(Class),属性(Field)和方法(Method)的不同位置,可以将泛型的用法总结为以下几种:

1.泛型类。在类名后添加泛型标识(<T...>),用于表示该类持有的一种类型。

2.泛型属性。泛型属性必须结合泛型类使用,用于接收泛型类持有的类型T。

3.泛型方法。在方法的返回值前声明泛型(<T extends 父类名>),该泛型是对该方法的参数T的一种限定。

泛型用法的补充

1.如果泛型T没有被extends修饰(包括类和方法),我们称之为无界泛型;如果被extend修饰,我们就称之为有界泛型。

2.如果方法参数中有泛型T,而方法的返回类型前没有泛型T,该类型就不是泛型方法,而是泛型类。

3.泛型方法常用在工具类中(即该方法只是一种工具),与类的实例对象无关。

4.当泛型方法中的泛型T与类中的泛型T同名时会产生警报,因为编译器不确定你要使用哪个持有对象。

有界泛型

相较于无界泛型(没有限定类型)的用法,我们可以使用有界泛型(<T extends 父类名>)来限定持有对象的范围,或泛型方法传入该方法参数的范围,以此保证业务逻辑的正确执行。

有界泛型只有上界(extends),而没有下界的用法(相对于通配符【?】)。

泛型中的继承

用一段代码说明泛型中继承的使用。

ArrayList<String> arrayList = new ArrayList<>();
Object object = new Object();
arrayList.add(object); // 编译器报错

因为 ArrayList<String>不是 ArrayList<Object>的子类 ,因此编译器会报错,错误信息为The method add(String) in the type ArrayList<String> is not applicable for the arguments (Object)。

通配符【?】

通配符代表的是一种未知的类型,常用在方法上(注意与泛型方法的区别,其不需要在方法的返回类型前声明。

上界通配

上界通配即定义通配符的上界,用关键字extends声明。

public static void process(List<? extends Foo> list){/ * ... * /}

无界通配

无界通配即不限制通配符的界限,不需要任何关键字去修饰【?】。

public static void printList(List<?> list){/*........*/}

通配符的功能形式和声明类型为Object是有区别的。

public static void printList(List<Object> list){/*........*/}

List<Object>和List<?>是不一样的,前者可以插入Object或任何Object对象的子类,但是后者在类型不匹配的情况下只能够插入null。

public class TestWildCard {
    public void printList(List<String> list) {
        for (Object elem : list)
            System.out.print(elem + " ");
        System.out.println();
    }

    public void printList2(List<?> list) {
        for (Object elem : list)
            System.out.print(elem + " ");
        System.out.println();
    }

    public static void main(String[] args) {
        TestWildCard testWildcard = new TestWildCard();

        ArrayList<? extends Object> arrayList = new ArrayList<>();
        arrayList.add(null);
        // arrayList.add(testWildcard); // 报错

        ArrayList<Object> arrayList2 = new ArrayList<>();
        arrayList2.add(null);
        arrayList2.add("2");

        List<Integer> li = Arrays.asList(1, 2, 3);
        testWildcard.printList2(li);
        // testWildcard.printList(li); // 报错

        List<String> ls = Arrays.asList("one", "two", "three");
        testWildcard.printList2(ls);
        testWildcard.printList(ls);
    }
}

下界通配

下界通配即定义通配符的下super界,用关键字extends声明。

public static void addNumbers(List<? super Integer> list){}

泛型【T】与通配符【?】的区别

泛型和通配符最根本的区别就是Java编译器会把泛型【T】推断成具体类型,而把通配符【?】推断成未知类型。Java编辑器只能操作具体类型,不能操作未知类型,如果有对参数做修改的操作就必须要使用泛型,如果仅仅是查看就可以使用通配符。

这样,我们可以利用通配符特性设计出安全的接口。比如在一个接口的方法中定义了通配符参数,则继承该接口的所有方法都不能修改该方法传递过来的参数。

interface GInterface {
    <T> void foo(List<? extends T> list);
}

public class GInterfaceImpl implements GInterface{
    @Override
    public <T> void foo(List<? extends T> list) {
        // 只能遍历list,不能修改list
        for (T t : list) {
            System.out.println(t);
        }

        // list.add(new Object()); // 编译器报错
    }

    public static void main(String[] args) {
        GInterfaceImpl gIterfaceImpl = new GInterfaceImpl();
        ArrayList<String> list = new ArrayList<>();
        list.add("1");
        list.add("2");
        list.add("8");
        gIterfaceImpl.foo(list);
    }
}

"你弱的时候,坏人最多,这个世界的温柔,都来自与你的强大。"

原文地址:https://www.cnblogs.com/yanggb/p/10961968.html

时间: 2024-11-09 03:46:18

java中的泛型【T】与通配符【?】概念入门的相关文章

Java中的泛型 (上) - 基本概念和原理

本节我们主要来介绍泛型的基本概念和原理 后续章节我们会介绍各种容器类,容器类可以说是日常程序开发中天天用到的,没有容器类,难以想象能开发什么真正有用的程序.而容器类是基于泛型的,不理解泛型,我们就难以深刻理解容器类.那,泛型到底是什么呢? 什么是泛型? 一个简单泛型类 我们通过一个简单的例子来说明泛型类的基本概念.实现原理和好处. 基本概念 我们直接来看代码: public class Pair<T> { T first; T second; public Pair(T first, T se

Java 中的泛型详解-Java编程思想

Java中的泛型参考了C++的模板,Java的界限是Java泛型的局限. 2.简单泛型 促成泛型出现最引人注目的一个原因就是为了创造容器类. 首先看一个只能持有单个对象的类,这个类可以明确指定其持有的对象的类型 class Holder1 { private Circle a; public Holder1(Circle a) { this.a = a; } Circle get() { return a; } } 上面的类的可重用性不怎么样,无法持有其他类型的任何对象,下面通过持有Object

Java中的泛型 --- Java 编程思想

前言 ? 我一直都认为泛型是程序语言设计中一个非常基础,重要的概念,Java 中的泛型到底是怎么样的,为什么会有泛型,泛型怎么发展出来的.通透理解泛型是学好基础里面中非常重要的.于是,我对<Java编程思想>这本书中泛型章节进行了研读.可惜遗憾的是,自己没有太多的经验,有些东西看了几次也是有点懵.只能以后有机会,再进行学习了.但是自己也理解了挺多的.下面就是自己对于泛型的理解与感悟.如有不对,望指出. 概念 由来: Java 一开始设计之初是没有泛型这个特性的,直到jdk 1.5中引入了这个特

泛型及java中的泛型

当作笔记整理的~~~ 首先,引出堆对象这个概念. 什么是堆对象,就是程序在运行过程中可以随时建立或者删除的对象,可以用new运算符(或malloc函数)或者delete运算符(或free函数).泛型可以看作是一类堆对象. 泛型是程序设计语言的一种特性.允许程序员在强类型程序设计语言中编写代码时定义一些可变部分,那些部分在使用前必须作出指明. 各种程序设计语言和其编译器.运行环境对泛型的支持均不一样.将类型参数化以达到代码复用提高软件开发工作效率的一种数据类型. 泛型的定义主要有两种:1.在程序编

java中的泛型(转)

什么是泛型? 泛型(Generic type 或者 generics)是对 Java 语言的类型系统的一种扩展,以支持创建可以按类型进行参数化的类.可以把类型参数看作是使用参数化类型时指定的类型的一个占位符,就像方法的形式参数是运行时传递的值的占位符一样. 可以在集合框架(Collection framework)中看到泛型的动机.例如,Map 类允许您向一个 Map 添加任意类的对象,即使最常见的情况是在给定映射(map)中保存某个特定类型(比如 String)的对象. 因为 Map.get(

Java开发知识之Java中的泛型

Java开发知识之Java中的泛型 一丶简介什么是泛型. 泛型就是指泛指任何数据类型. 就是把数据类型用泛型替代了. 这样是可以的. 二丶Java中的泛型 Java中,所有类的父类都是Object类.所以定义泛型的时候,设计长须的话传入的值与返回的值都是Object类型为主.如果是用具体的实例,就要进行转换了.具体参考向上转型,跟向下转型. JDK 1.5版本才有了泛型机制. 语法如下: class 类名<T >{ public T a; public T b; public void Set

夯实Java基础系列13:深入理解Java中的泛型

目录 泛型概述 一个栗子 特性 泛型的使用方式 泛型类 泛型接口 泛型通配符 泛型方法 泛型方法的基本用法 类中的泛型方法 泛型方法与可变参数 静态方法与泛型 泛型方法总结 泛型上下边界 泛型常见面试题 参考文章 微信公众号 Java技术江湖 个人公众号:黄小斜 本系列文章将整理到我在GitHub上的<Java面试指南>仓库,更多精彩内容请到我的仓库里查看 https://github.com/h2pl/Java-Tutorial 喜欢的话麻烦点下Star.Fork.Watch三连哈,感谢你的

Java基础之Java中的泛型

1.为什么要使用泛型 这里我们俩看一段代码; List list = new ArrayList(); list.add("CSDN_SEU_Cavin"); list.add(100); for (int i = 0; i < list.size(); i++) { String name = (String) list.get(i); //取出Integer时,运行时出现异常 System.out.println("name:" + name); } 本例

java中的泛型的使用与理解

什么是泛型? 泛型是程序设计语言的一种特性.允许程序员在强类型程序设计语言中编写 体验泛型代码时定义一些可变部份,那些部份在使用前必须作出指明.各种程序设计语言和其编译器.运行环境对泛型的支持均不一样.将类型参数化以达到代码复用提高软件开发工作效率的一种数据类型.泛型类是引用类型,是堆对象,主要是引入了类型参数这个概念. 定义: 泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数.这种参数类型可以用在类.接口和方法的创建中,分别称为泛型类.泛型