Java 泛型概述

简介: Java 泛型概述

泛型的作用

第一是泛化。

可以用T代表任意类型Java语言中引入泛型是一个较大的功能增强不仅语言、类型系统和编译器有了较大的变化,以支持泛型,而且类库也进行了大翻修,所以许多重要的类,比如集合框架,都已经成为泛型化的了,这带来了很多好处。

第二是类型安全

。泛型的一个主要目标就是提高Java程序的类型安全,使用泛型可以使编译器知道变量的类型限制,进而可以在更高程度上验证类型假设。如果不用泛型,则必须使用强制类型转换,而强制类型转换不安全,在运行期可能发生ClassCast Exception异常,如果使用泛型,则会在编译期就能发现该错误。

第三是消除强制类型转换。

泛型可以消除源代码中的许多强制类型转换,这样可以使代码更加可读,并减少出错的机会。

第四是向后兼容。

支持泛型的Java编译器(例如JDK1.5中的Javac)可以用来编译经过泛型扩充的Java程序(Generics Java程序),但是现有的没有使用泛型扩充的Java程序仍然可以用这些编译器来编译。

泛型的优点

1、类型安全

泛型的主要目标是提高Java程序的类型安全。通过知道使用泛型定义的变量的类型限制,编译器可以在非常高的层次上验证类型假设。没有泛型,这些假设就只存在于系统开发人员的头脑中。


通过在变量声明中捕获这一附加的类型信息,泛型允许编译器实施这些附加的类型约束。类型错误就可以在编译时被捕获了,而不是在运行时当作ClassCastException展示出来。将类型检查从运行时挪到编译时有助于Java开发人员更早、更容易地找到错误,并可提高程序的可靠性。

2、消除强制类型转换

泛型的一个附带好处是,消除源代码中的许多强制类型转换。这使得代码更加可读,并且减少了出错机会。尽管减少强制类型转换可以提高使用泛型类的代码的累赞程度,但是声明泛型变量时却会带来相应的累赞程度。在简单的程序中使用一次泛型变量不会降低代码累赞程度。但是对于多次使用泛型变量的大型程序来说,则可以累积起来降低累赞程度。所以泛型消除了强制类型转换之后,会使得代码加清晰和筒洁。


3、更高的运行效率

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


4、潜在的性能收益

泛型为较大的优化带来可能。在泛型的初始实现中,编译器将强制类型转换(没有泛型的话,Java系统开发人员会指定这些强制类型转换)插入生成的字节码中。但是更多类型信息可用于编译器这一事实,为未来版本的JVM的优化带来可能

泛型概述

Java SE 5.0以前操作集合的缺点:

当把一个对象存入集合后,集合会“忘记”这个对象的类型,将该对象从集合中取出时,这个对象的编译类型就变成了Object类型。

从集合中取出对象,需要执行类型转换操作:

由于没有类型检查,可以向集合添加任意对象,get方法在取出元素时,如果进行强制类型转换就很容易出错

ArrayList files = new ArrayList();
 . . . 
String filename = (String) files.get(0); 

泛型提供了参数化类型

在定义集合类时,可以使用“<参数化类型>”的方式指定该类中方法操作的数据类型。如

ArrayList<String> files = new ArrayList<String>();

限定了ArrayList集合只能存储String类型元素

取出对象的类型为String,无需执行类型转换。

没有使用泛型的Java代码警告

泛型的使用

如何定义泛型List对象

List<T>

T是类型变量

T也可以是泛型变量类型

泛型经常被称为参数化类型,它能够像方法一样接受不同类型的参数

<Integer>存储IntegerList

        List <Integer> numbers = new ArrayList<Integer>();
        for (int i = 1; i <= 49; i++){
            numbers.add(i);
        }
Collections.shuffle(numbers);//对集合中的元素随机排序
        List<Integer> subnumbers=    numbers.subList(0, 6);
        Collections.sort(subnumbers);
        System.out.println(subnumbers);

泛型中使用通配符

泛型中可以使用“?”通配符作为参数,表示该泛型可以接收任意类型的数据

public static void print(List<?>  list) {
for (Object element : list) {
                     System.out.println(element );
               } }

上界通配符:

泛型中只允许一个类自身(接口)或者其子类(实现该接口的类)作为参数传入。

ArrayList<? extends 父类(接口)>

下界通配符:

泛型中只允许一个类自身或者该类的父类作为参数传入。

ArrayList<? super 子类>

自定义泛型

Point类具有两个成员,类型待定

使用private T x表示x的类型为参数T

public class GerDemo {
public static void main(String[] args) {
Point<Integer> p1=new Point<Integer>();  //点p1
p1.setX(5);
p1.setY(8);
System.out.println("p1:("+p1.getX()+","+p1.getY()+")");
Point<Integer> p2=new Point<Integer>(34,98); //点p2
System.out.println("p2:("+p2.getX()+","+p2.getY()+")");
Point<Double> p3=new Point<Double>(); //点p3
p3.setX(5.9);
p3.setY(8.0);
System.out.println("p3:("+p3.getX()+","+p3.getY()+")");
Point<Double> p4=new Point<Double>(2.1,9.2); //点p4
System.out.println("p4:("+p4.getX()+","+p4.getY()+")");
}
}
public class Point<T>
{  
    private T x;
    private T y;
    public Point() {}
    public Point(T x, T y) {  this.x = x;  this.y = y;  } 
    public T getX() {  return  x;  }
    public T getY() {  return  y;  }    
    public void setX(T x) {  this.x = x;  }
    public void setY(T y) {  this.y = y; }    
}

编写一个泛型方法,能够对数组求最大值和最小值(实现通用的求极值算法)。

class ArrayAlg {
   public static <T extends Comparable> Pair<T> minmax(T[ ] a) {
        if (a == null || a.length == 0) {
            return null;
        }
        T min = a[0];T max = a[0];
        for (int i = 1; i < a.length; i++) {
            if (min.compareTo(a[i]) > 0) {min = a[i];}
            if (max.compareTo(a[i]) < 0) { max = a[i];}
        }
        return new Pair<T>(min, max);
    }
}

使用泛型方法

创建一个日历的数组,求出最大和最小的日期

Pair<GregorianCalendar > mm 使用 GregorianCalendar 为 Pair<T>

传递类型参数

Java.util.GregorianCalendar实现了接口 java.lang.Comparable <T>

        GregorianCalendar[ ] birthdays = {
            new GregorianCalendar(1906, Calendar.DECEMBER, 9), 
            new GregorianCalendar(1815, Calendar.DECEMBER, 10), 
            new GregorianCalendar(1903, Calendar.DECEMBER, 3), 
            new GregorianCalendar(1910, Calendar.JUNE, 22), 
        };
        Pair<GregorianCalendar> mm =   ArrayAlg.minmax(birthdays);
        System.out.println("min = " + mm.getFirst().getTime());
        System.out.println("max = " + mm.getSecond().getTime());

如何求出一个int类型数组的最大值和最小值?

如何求出一个String类型数组的最大值和最小值(字典顺序)

String[ ] words ={"able","word","excel","course","java","c#"};
      Pair<String> mm = ArrayAlg.minmax(words);
      System.out.println("first word = " + mm.getFirst());
      System.out.println("last word = " + mm.getSecond());
目录
相关文章
|
3月前
|
安全 Java 编译器
揭秘JAVA深渊:那些让你头大的最晦涩知识点,从泛型迷思到并发陷阱,你敢挑战吗?
【8月更文挑战第22天】Java中的难点常隐藏在其高级特性中,如泛型与类型擦除、并发编程中的内存可见性及指令重排,以及反射与动态代理等。这些特性虽强大却也晦涩,要求开发者深入理解JVM运作机制及计算机底层细节。例如,泛型在编译时检查类型以增强安全性,但在运行时因类型擦除而丢失类型信息,可能导致类型安全问题。并发编程中,内存可见性和指令重排对同步机制提出更高要求,不当处理会导致数据不一致。反射与动态代理虽提供运行时行为定制能力,但也增加了复杂度和性能开销。掌握这些知识需深厚的技术底蕴和实践经验。
74 2
|
10天前
|
Java API
[Java]泛型
本文详细介绍了Java泛型的相关概念和使用方法,包括类型判断、继承泛型类或实现泛型接口、泛型通配符、泛型方法、泛型上下边界、静态方法中使用泛型等内容。作者通过多个示例和测试代码,深入浅出地解释了泛型的原理和应用场景,帮助读者更好地理解和掌握Java泛型的使用技巧。文章还探讨了一些常见的疑惑和误区,如泛型擦除和基本数据类型数组的使用限制。最后,作者强调了泛型在实际开发中的重要性和应用价值。
12 0
[Java]泛型
|
19天前
|
存储 安全 Java
🌱Java零基础 - 泛型详解
【10月更文挑战第7天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
10 1
|
26天前
|
Java 语音技术 容器
java数据结构泛型
java数据结构泛型
26 5
|
24天前
|
存储 Java 编译器
Java集合定义其泛型
Java集合定义其泛型
17 1
|
2月前
|
Java 编译器 容器
Java——包装类和泛型
包装类是Java中一种特殊类,用于将基本数据类型(如 `int`、`double`、`char` 等)封装成对象。这样做可以利用对象的特性和方法。Java 提供了八种基本数据类型的包装类:`Integer` (`int`)、`Double` (`double`)、`Byte` (`byte`)、`Short` (`short`)、`Long` (`long`)、`Float` (`float`)、`Character` (`char`) 和 `Boolean` (`boolean`)。包装类可以通过 `valueOf()` 方法或自动装箱/拆箱机制创建。
34 9
Java——包装类和泛型
|
26天前
|
存储 Java 编译器
【用Java学习数据结构系列】初识泛型
【用Java学习数据结构系列】初识泛型
18 2
|
2月前
|
安全 Java API
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
String常量池、String、StringBuffer、Stringbuilder有什么区别、List与Set的区别、ArrayList和LinkedList的区别、HashMap底层原理、ConcurrentHashMap、HashMap和Hashtable的区别、泛型擦除、ABA问题、IO多路复用、BIO、NIO、O、异常处理机制、反射
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
|
24天前
|
安全 Java 编译器
Java基础-泛型机制
Java基础-泛型机制
13 0
|
2月前
|
存储 安全 搜索推荐
Java中的泛型
【9月更文挑战第15天】在 Java 中,泛型是一种编译时类型检查机制,通过使用类型参数提升代码的安全性和重用性。其主要作用包括类型安全,避免运行时类型转换错误,以及代码重用,允许编写通用逻辑。泛型通过尖括号 `&lt;&gt;` 定义类型参数,并支持上界和下界限定,以及无界和有界通配符。使用泛型需注意类型擦除、无法创建泛型数组及基本数据类型的限制。泛型显著提高了代码的安全性和灵活性。