泛型进阶: 泛型方法 & 通配符

简介: 泛型进阶: 泛型方法 & 通配符

泛型方法

定义语法

方法限定符 <类型参数列表> 返回值类型 方法名称(形参列表) {...}

示例

public class Test {
    //静态的泛型方法 需要在static后用<>声明泛型参数
    public static <E> void swap(E[] array, int i, int j) {
        E t = array[i];
        array[i] = array[j];
        array[j] = t;
    }
 
    public static void main(String[] args) {
        //完成一个字符数组的倒转,注意这里的类型是包装类
        Character[] arr = {'h', 'e', 'l', 'l', 'o'};
        for(int i = 0; i < arr.length / 2; i++) {
            swap(arr, i, arr.length - 1 - i);
        }
        System.out.println(Arrays.toString(arr));
    }
}

运行结果:

通配符

? 用于在泛型的使用,即为通配符

通配符解决什么问题

class Message<T> {
    private T message;
 
    public T getMessage() {
        return message;
    }
 
    public void setMessage(T message) {
        this.message = message;
    }
}
 
public class Test1 {
    public static void main(String[] args) {
        Message<String> message = new Message<>();
        message.setMessage("我不吃牛肉");
        fun(message);
    }
 
    public static void fun(Message<String> temp) {
        System.out.println(temp.getMessage());
    }
}

以上程序会带来新的问题,如果现在泛型的类型设置不是String, 而是Integer.

public class Test1 {
    public static void main(String[] args) {
        Message<String> message = new Message<>();
        message.setMessage(99);
        fun(message);
    }
 
    public static void fun(Message<String> temp) {
        System.out.println(temp.getMessage());
    }
}

我们需要的解决方案: 可以接收所有的泛型类型, 但又不能让用户随意修改.这种情况就需要使用通配符"?"来处理

范例:使用通配符

public class Test1 {
    public static void main(String[] args) {
        Message<String> message = new Message<>();
        message.setMessage(99);
        fun(message);
    }
    
    //此时使用通配符"?"描述的是它可以接收任意类型,但是由于不确定类型,所以无法修改
    public static void fun(Message<?> temp) {
        //temp.setMessage(100);无法修改!
        System.out.println(temp.getMessage());
    }
}

在"?"的基础上又产生了两个子通配符:

? extends 类: 设置通配符上限

? super 类: 设置了通配符下限

通配符上界

语法:

<? extends 上界>

<? extends Number> //可以传入的实参类型是Number或者Number的子类

举个栗子:

class Food {
 
}
 
class Fruit extends Food {
 
}
 
class Apple extends Fruit {
 
}
 
class Banana extends Fruit {
    
}
 
class Message<T> {//设置泛型
    private T message;
    
    public T getMessage() {
        return message;
    }
    
    public void setMessage(T message) {
        this.message = message;
    }
}
 
 
public class Test2 {
    public static void main(String[] args) {
        Message<Apple> message = new Message<>();
        message.setMessage(new Apple());
        fun(message);
 
        Message<Apple> message2 = new Message<>();
        message2.setMessage(new Banana());
        fun(message2);
    }
    
    //此时使用通配符"?"描述的是它可以接收任意类型, 但是由于不确定类型, 所以无法修改
    public static void fun(Message<? extends Fruit> temp) {
        //temp.setMessage(new Banana()); //仍然无法修改
        //temp.setMessage(new Apple()); //仍然无法修改
        System.out.println(temp.getMessage());
    }
}

此时无法在fun函数中对temp进行添加元素,因为temp接收的是Fruit和它的子类, 此时存储的元素应该是哪个子类无法确定.所以会添加报错!但是仍然可以获取元素.

//此时使用通配符"?"描述的是它可以接收任意类型, 但是由于不确定类型, 所以无法修改
public static void fun(Message<? extends Fruit> temp) {
    //temp.setMessage(new Banana()); //仍然无法修改
    //temp.setMessage(new Apple()); //仍然无法修改
    Fruit b = temp.getMessage();
    System.out.println(b);
}     
   
    
    
    

通配符的上界, 不能写入数据, 只能进行读取数据.

通配符的下界

语法:

<? super 下界>

<? super Integer> //代表可以传入的实参的类型是Integer或者Integer的父类类型

class Food {}
class Fruit extends Food {}
class Apple extends Fruit {}
 
class Plate<T> {
    private T plate;
    
    public T getPlate() {
        return plate;
    }
    
    public void setPlate(T plate) {
        this.plate = plate;
    }
}
 
public class Test {
    public static void main(String[] args) {
        
    }
    
    public static void fun(Plate<? super Fruit> temp) {
        //此时可以修改!添加的是Fruit或者是Fruit的子类
        temp.setPlate(new Apple());//这个是Fruit的子类
        temp.setPlate(new Fruit());//这个是Fruit本身
        //Fruit fruit = temp.getPlate(); 不能直接接收,这里无法确定是哪个父类
        System.out.println(temp.getPlate());//只能直接输出
    }
}

通配符的下界,不能进行读取数据,只能写入数据.

相关文章
|
1天前
|
云安全 人工智能 自然语言处理
AI说的每一句话,都靠谱吗?
阿里云提供AI全栈安全能力,其中针对AI输入与输出环节的安全合规挑战,我们构建了“开箱即用”与“按需增强”相结合的多层次、可配置的内容安全机制。
|
5天前
|
存储 人工智能 安全
AI 越智能,数据越危险?
阿里云提供AI全栈安全能力,为客户构建全链路数据保护体系,让企业敢用、能用、放心用
|
8天前
|
域名解析 人工智能
【实操攻略】手把手教学,免费领取.CN域名
即日起至2025年12月31日,购买万小智AI建站或云·企业官网,每单可免费领1个.CN域名首年!跟我了解领取攻略吧~
|
2天前
|
消息中间件 安全 NoSQL
阿里云通过中国信通院首批安全可信中间件评估
近日,由中国信通院主办的 2025(第五届)数字化转型发展大会在京举行。会上,“阿里云应用服务器软件 AliEE”、“消息队列软件 RocketMQ”、“云数据库 Tair”三款产品成功通过中国信通院“安全可信中间件”系列评估,成为首批获此认证的中间件产品。此次评估覆盖安全可信要求、功能完备性、安全防护能力、性能表现、可靠性与可维护性等核心指标,标志着阿里云中间件产品在多架构适配与安全能力上达到行业领先水平。
296 192
|
2天前
|
安全 Java Android开发
深度解析 Android 崩溃捕获原理及从崩溃到归因的闭环实践
崩溃堆栈全是 a.b.c?Native 错误查不到行号?本文详解 Android 崩溃采集全链路原理,教你如何把“天书”变“说明书”。RUM SDK 已支持一键接入。
311 164
|
1天前
|
开发者
「玩透ESA」ESA启用和加速-ER在加速场景中的应用
本文介绍三种配置方法:通过“A鉴权”模板创建函数并设置触发器路由;在ESA上配置回源302跟随;以及自定义响应头。每步均配有详细截图指引,帮助开发者快速完成相关功能设置,提升服务安全性与灵活性。
291 2
|
7天前
|
数据采集 人工智能 自然语言处理
3分钟采集134篇AI文章!深度解析如何通过云无影AgentBay实现25倍并发 + LlamaIndex智能推荐
结合阿里云无影 AgentBay 云端并发采集与 LlamaIndex 智能分析,3分钟高效抓取134篇 AI Agent 文章,实现 AI 推荐、智能问答与知识沉淀,打造从数据获取到价值提炼的完整闭环。
454 93