Java中泛型概述、泛型方法、泛型接口、类型通配符及可变参数的使用

简介: 泛型概述、泛型方法、泛型接口、类型通配符及可变参数的使用的简单示例

 一、泛型概述和好处

泛型概述

● 是JDK5中引入的特性,它提供了编译时类型安全检测机制,该机制允许在编译时检测到非法的类型

● 它的本质是参数化类型,也就是说操作的数据类型被指定为一个参数,一提到参数,最熟悉的就是定义方法时有形参,然后调用此方法时传递实参。那么参数化类型怎么理解呢?顾名思义,就是将类型由原来的具体的类型参数化,然后在使用/调用时传入具体的类型。这种参数类型可以用在类、方法和接口中,分别被称为泛型类、泛型方法、泛型接口。

泛型定义格式:

<类型>:指定一种类型的格式,这里的类型可以看出是形参

<类型1,类型2...>:指定多种类型的格式,多种类型之间用逗号隔开。这里的类型可以看出是形参

将来具体调用时候给定的类型可以看出是实参,并且实参的类型只能是引用数据类型

泛型的好处:

把运行时期的问题提前到了编译期间

避免了强制类型转换

二、泛型类

定义格式:

修饰符 class 类名<类型> { }

image.gif

示例代码:

泛型类

public class Generic<T> { 
    private T t; 
    public T getT() { 
        return t; 
    }
    public void setT(T t) { 
        this.t = t; 
    } 
}

image.gif

测试类

public class GenericDemo { 
    public static void main(String[] args) { 
        Generic<String> g1 = new Generic<String>(); 
        g1.setT("小明"); 
        System.out.println(g1.getT()); 
        Generic<Integer> g2 = new Generic<Integer>(); 
        g2.setT(30); 
        System.out.println(g2.getT()); 
        Generic<Boolean> g3 = new Generic<Boolean>(); 
        g3.setT(true); 
        System.out.println(g3.getT()); 
    } 
}

image.gif

三、泛型方法

定义格式:

修饰符 <类型> 返回值类型 方法名(类型 变量名) { }

image.gif

示例代码:

带有泛型方法的类

public class Generic { 
    public <T> void show(T t) { 
        System.out.println(t); 
    } 
}

image.gif

测试类

public class GenericDemo { 
    public static void main(String[] args) { 
        Generic g = new Generic(); 
        g.show("小明"); 
        g.show(30); 
        g.show(true); 
        g.show(12.34); 
    } 
}

image.gif

四、泛型接口

定义格式:

修饰符 interface 接口名<类型> { }

image.gif

示例代码:

泛型接口

public interface Generic<T> { 
    void show(T t); 
}

image.gif

泛型接口实现类

public class GenericImpl<T> implements Generic<T> { 
    @Override 
    public void show(T t) { 
        System.out.println(t); 
    } 
}

image.gif

测试类

public class GenericDemo { 
    public static void main(String[] args) { 
        Generic<String> g1 = new GenericImpl<String>(); 
        g1.show("小明"); 
        Generic<Integer> g2 = new GenericImpl<Integer>(); 
        g2.show(30); 
    } 
}

image.gif

五、类型通配符

类型通配符的作用:

为了表示各种泛型List的父类,可以使用类型通配符

类型通配符的分类:

类型通配符:<?>

List<?>:表示元素类型未知的List,它的元素可以匹配任何的类型

这种带通配符的List仅表示它是各种泛型的List的父类,并不能把元素添加到其中

类型通配符上限:<? extends 类型>

List<? extends Nubers>:它表示的类型是Number或者其子类型

类型通配符下限:<? super 类型>

List<? super Number>:它表示的类型是Number或者其父类型

类型通配符的基本使用:

public class GenericDemo { 
    public static void main(String[] args) { 
        //类型通配符:<?> 
        List<?> list1 = new ArrayList<Object>(); 
        List<?> list2 = new ArrayList<Number>(); 
        List<?> list3 = new ArrayList<Integer>(); 
        System.out.println("--------"); 
        //类型通配符上限:<? extends 类型> 
//         List<? extends Number> list4 = new ArrayList<Object>(); 
        List<? extends Number> list5 = new ArrayList<Number>(); 
        List<? extends Number> list6 = new ArrayList<Integer>(); 
        System.out.println("--------"); 
        //类型通配符下限:<? super 类型> 
        List<? super Number> list7 = new ArrayList<Object>(); 
        List<? super Number> list8 = new ArrayList<Number>(); 
//         List<? super Number> list9 = new ArrayList<Integer>(); 
    } 
}

image.gif

六、可变参数

可变参数介绍:

可变参数又称为参数个数可变,用作方法的形参出现,那么方法参数个数就是可变的

可变参数定义格式:

修饰符 返回值类型 方法名(数据类型… 变量名) { }

image.gif

可变参数的注意事项:

这里的变量其实是一个数组

如果一个方法有多个参数,包含可变参数,可变参数要放在最后

可变参数的基础使用:

public class ArgsDemo01 { 
    public static void main(String[] args) { 
        System.out.println(sum(10, 20)); 
        System.out.println(sum(10, 20, 30)); 
        System.out.println(sum(10, 20, 30, 40)); 
        System.out.println(sum(10,20,30,40,50)); 
        System.out.println(sum(10,20,30,40,50,60)); 
        System.out.println(sum(10,20,30,40,50,60,70)); 
        System.out.println(sum(10,20,30,40,50,60,70,80,90,100)); 
    } 
//         public static int sum(int b,int... a) { 
//         return 0; 
//         } 
        public static int sum(int... a) { 
        int sum = 0; 
        for(int i : a) { 
            sum += i; 
        }
        return sum; 
    } 
}

image.gif

可变参数的使用

Arrays工具类中有一个静态方法:

● public static List asList(T... a):返回由指定数组支持的固定大小的列表

● 返回的集合不能做增删操作,可以做修改操作

Lsit接口中有一个静态方法:

● public static List of(E... element):返回包含任意数量元素的不可变列表

● 返回的集合不能做增删改操作

Set集合中有一个静态方法:

● public static Set of(E... elements):返回一个包含任意数量元素的不可变集合

● 再给元素的时候,不能给重复的元素

● 返回的集合不能做增删操作,没有修改的方法

示例 代码:

public class ArgsDemo02 {
    public static void main(String[] args) {
        //public static <T > List < T > asList(T...a):返回由指定数组支持的固定大小的列表
        List<String> list = Arrays.asList("hello", "world", "java");
        list.add("javaee"); //UnsupportedOperationException 
        list.remove("world"); //UnsupportedOperationException 
        list.set(1, "javaee");
        System.out.println(list);
        // public static <E > List < E > of(E...elements):返回包含任意数量元素的不可变列表
        List<String> list2 = List.of("hello", "world", "java", "world");
        list2.add("javaee");//UnsupportedOperationException 
        list2.remove("java");//UnsupportedOperationException 
        list2.set(1, "javaee");//UnsupportedOperationException 
        System.out.println(list2);
        //public static <E > Set < E > of(E...elements) :返回一个包含任意数量元素的不可变集合
        Set<String> set1 = Set.of("hello", "world", "java", "world");//IllegalArgumentException
        Set<String> set2 = Set.of("hello", "world", "java");
        set1.add("javaee");//UnsupportedOperationException 
        set2.remove("world");//UnsupportedOperationException 
        System.out.println(set1);
    }
}

image.gif

目录
相关文章
|
8天前
|
JSON Java Apache
非常实用的Http应用框架,杜绝Java Http 接口对接繁琐编程
UniHttp 是一个声明式的 HTTP 接口对接框架,帮助开发者快速对接第三方 HTTP 接口。通过 @HttpApi 注解定义接口,使用 @GetHttpInterface 和 @PostHttpInterface 等注解配置请求方法和参数。支持自定义代理逻辑、全局请求参数、错误处理和连接池配置,提高代码的内聚性和可读性。
|
12天前
|
存储 Java 开发者
Java 中 Set 类型的使用方法
【10月更文挑战第30天】Java中的`Set`类型提供了丰富的操作方法来处理不重复的元素集合,开发者可以根据具体的需求选择合适的`Set`实现类,并灵活运用各种方法来实现对集合的操作和处理。
|
9天前
|
Java
java线程接口
Thread的构造方法创建对象的时候传入了Runnable接口的对象 ,Runnable接口对象重写run方法相当于指定线程任务,创建线程的时候绑定了该线程对象要干的任务。 Runnable的对象称之为:线程任务对象 不是线程对象 必须要交给Thread线程对象。 通过Thread的构造方法, 就可以把任务对象Runnable,绑定到Thread对象中, 将来执行start方法,就会自动执行Runable实现类对象中的run里面的内容。
23 1
|
14天前
|
Java 开发者
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
41 4
|
13天前
|
Java 编译器 开发者
Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面
本文探讨了Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面,帮助开发者提高代码质量和程序的健壮性。
29 2
|
20天前
|
安全 Java
在 Java 中使用实现 Runnable 接口的方式创建线程
【10月更文挑战第22天】通过以上内容的介绍,相信你已经对在 Java 中如何使用实现 Runnable 接口的方式创建线程有了更深入的了解。在实际应用中,需要根据具体的需求和场景,合理选择线程创建方式,并注意线程安全、同步、通信等相关问题,以确保程序的正确性和稳定性。
|
19天前
|
Java
Java基础(13)抽象类、接口
本文介绍了Java面向对象编程中的抽象类和接口两个核心概念。抽象类不能被实例化,通常用于定义子类的通用方法和属性;接口则是完全抽象的类,允许声明一组方法但不实现它们。文章通过代码示例详细解析了抽象类和接口的定义及实现,并讨论了它们的区别和使用场景。
|
19天前
|
Java 测试技术 API
Java零基础-接口详解
【10月更文挑战第19天】Java零基础教学篇,手把手实践教学!
18 1
|
24天前
|
Java
在Java多线程编程中,实现Runnable接口通常优于继承Thread类
【10月更文挑战第20天】在Java多线程编程中,实现Runnable接口通常优于继承Thread类。原因包括:1) Java只支持单继承,实现接口不受此限制;2) Runnable接口便于代码复用和线程池管理;3) 分离任务与线程,提高灵活性。因此,实现Runnable接口是更佳选择。
33 2
|
24天前
|
Java
Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口
【10月更文挑战第20天】《JAVA多线程深度解析:线程的创建之路》介绍了Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口。文章详细讲解了每种方式的实现方法、优缺点及适用场景,帮助读者更好地理解和掌握多线程编程技术,为复杂任务的高效处理奠定基础。
28 2