每日一道Java面试题:说一说Java中的泛型?

简介: 今天的每日一道Java面试题聊的是Java中的泛型,泛型在面试的时候偶尔会被提及,频率不是特别高,但在日后的开发工作中,却是是个高频词汇,因此,我们有必要去认真的学习它。

写在开头

今天的每日一道Java面试题聊的是Java中的泛型,泛型在面试的时候偶尔会被提及,频率不是特别高,但在日后的开发工作中,却是是个高频词汇,因此,我们有必要去认真的学习它。

泛型的定义

什么是泛型?

什么是泛型?这是个好问题,JDK5更新时带来了一个新特性-==泛型==,所谓“泛型”就是类型参数化,把类型定义成参数的形式(编译期-类型形参),调用时再传入具体的类型(调用时-类型实参)。

了解了定义之后,我们再来直观的感受一下泛型的魅力吧
不使用泛型

List list = new ArrayList();
list.add(1);
int o1 = (int)list.get(0);

使用泛型

List<String> list1 = new ArrayList<>();
list1.add("1");
String s = list1.get(0);

以上就是使用泛型和不适用泛型时代码的对比,可以看出在使用泛型后无需类型强转,这样更安全,代码可读性也更好啦。

泛型中的通配符

在学习和使用泛型的过程中,想必大家已经发现了泛型的尖括号中间的大写字母的差异吧,如< E >、< T >、< N >、<?>、,这些都称为泛型的通配符

E - Element (在集合中使用,因为集合中存放的是元素)

T - Type(Java 类)

K - Key(键)

V - Value(值)

N - Number(数值类型)

? - 表示不确定的java类型

泛型的使用

学以致用,用学相长,乃是大道!

Java中的泛型通常可使用在类、接口、和方法上,我们一个个的看哈

泛型类

泛型类的命名格式:类名<>;尖括号中可以为T、E、K、V等常用通配符,在实例化泛型类时,必须指定具体类型。

//JDK8中ArrayList的源码
public class ArrayList<E> extends AbstractList<E>
        implements List<E>, RandomAccess, Cloneable, java.io.Serializable
{
   
   
///
}

上面这段源码java.util.ArrayList中的ArrayList,我们可以看到这就是一个典型的泛型类,我们可以在使用的时候指定不同的类型,基本类型或包装类,或引用类型去存储不同类型的数据,那么现在我们自己去设计一个ArrayList看一下。

【代码示例】
class Arraylist<E> {
   
   
    private Object[] elementData;
    private int size = 0;

    public Arraylist(int initialCapacity) {
   
   
        this.elementData = new Object[initialCapacity];
    }

    public boolean add(E e) {
   
   
        elementData[size++] = e;
        return true;
    }

    E elementData(int index) {
   
   
        return (E) elementData[index];
    }
}
【输出】
//测试类调用刚刚手写的ArrayList
public class Test {
   
   
    public static void main(String[] args) {
   
   
        Arraylist<Integer> list = new Arraylist<Integer>(10);
        list.add(666);
        System.out.println(list.elementData(0));
        new ArrayList<>();
    }
}
//输出:666
【注意】
在实例化泛型类的时候,建议指定具体的泛型类型,否则返回所有类的父类Object。
### 泛型接口
泛型接口的定义与泛型类类似,直接上代码!

【代码示例】
public interface Box<T> {
   
   
    public T method();
}

针对泛型接口的实现,我们既可以在实现的时候指定类型,也可以不指定。

【代码示例】
java /** * 实现泛型接口时指定类型为String */ public class BoxImpl implements Box<String>{ public static void main(String[] args) { BoxImpl test = new BoxImpl(); System.out.println(test.method()); } @Override public String method() { return "helloworld"; } }
当然,我们在开发的过程中,也会遇到一个泛型类在实现泛型接口的时候不指定类型,在实例化的时候,在指定也是OK的。

【代码示例】
class Arraylist<E> implements Box<E>{
   
   
      ///
    @Override
    public E method() {
   
   
        return null;
    }
}

泛型方法

泛型方法的定义与使用还是直接上代码去分析哈,清晰明了,哈哈哈!

【代码示例】
class Arraylist<E> {
   
   
    ///...
    //在上述手写的ArrayList中增加一个toArray的泛型方法
    public <T> T[] toArray(T[] a) {
   
   
        return (T[]) Arrays.copyOf(elementData, size, a.getClass());
    }
}
【调用】
public class Test {
   
   
    public static void main(String[] args) {
   
   
        Arraylist<Integer> list = new Arraylist<Integer>(3);
        list.add(1);
        list.add(2);
        list.add(3);
        Integer[] i = new Integer[3];
        //调用泛型方法,将数据转为对应类型数组
        i = list.toArray(i);
        for (Integer integer : i) {
   
   
            System.out.println(integer);
        }
    }
}

在定义泛型方法时,我们可以参考如下图(注意:方法返回类型和方法参数类型至少需要一个!)

image.png

静态泛型方法

除了普通的泛型方法外,还有一类静态泛型方法

【代码示例】
  //静态泛型方法
    public static < E > void printArray( E[] inputArray )
    {
   
   
        for ( E element : inputArray ){
   
   
            System.out.printf("%s",element);
        }
    }

在使用静态泛型方法时需要注意:

静态方法的加载先于类的实例化,也就是说类中的泛型还没有传递真正的类型参数,静态的方法的加载就已经完成了,所以静态泛型方法是没有办法使用类上声明的泛型的。只能使用自己声明的 < E >

泛型限定符-extends

在泛型的使用中可以使用关键字 extends 限定子类,看下面一段代码:

【代码示例】
class Grandfather {
   
   
    public String toString() {
   
   
        return "我是爷爷";
    }
}

class Father extends Grandfather{
   
   
    public String toString() {
   
   
        return "我是爹";
    }
}

class Son extends Father{
   
   
    public String toString() {
   
   
        return "我是儿子";
    }
}

先定义三个存在继承关系的爷爷-父亲-儿子的类,然后我们重写ArrayList的泛型

class Arraylist<E extends Father>
【调用】
ArrayList<Father> list = new ArrayList<>();
list.add(new Father());
list.add(new Son());
list.add(new Grandfather());

image.png

当我们添加Grandfather对象时,编译器报错如上,说明在使用extends进行限定的时候,仅能传入类与子类,同理可推出可以使用关键字 super 限定父类!

泛型擦除

在泛型的使用过程中,有个现象需要特别注意一下,那就是泛型擦除,泛型仅存在于编译时,JVM中是不存在泛型的,我们可以将上述ArrayList.class文件进行反编译,可以通过jad反编译工具,也可以通过网上的在线工具均可哈。
反编译后源码:

//反编译后的代码
package com.javabuild.server.pojo;

import java.util.Arrays;

class Arraylist {
   
   

   private Object[] elementData;
   private int size = 0;


   public Arraylist(int initialCapacity) {
   
   
      this.elementData = new Object[initialCapacity];
   }

   public boolean add(Object e) {
   
   
      ++this.size;
      this.elementData[this.size] = e;
      return true;
   }

   Object elementData(int index) {
   
   
      return this.elementData[index];
   }

   public Object[] toArray(Object[] a) {
   
   
      return (Object[])Arrays.copyOf(this.elementData, this.size, a.getClass());
   }

   public static void printArray(Object[] inputArray) {
   
   
      Object[] var1 = inputArray;
      int var2 = inputArray.length;

      for(int var3 = 0; var3 < var2; ++var3) {
   
   
         Object element = var1[var3];
         System.out.printf("%s", new Object[]{
   
   element});
      }

   }
}

可以发现Java中的泛型均被Object替换,因为在JVM解析的过程中会进行泛型的擦除操作。

目录
相关文章
|
4月前
|
缓存 Java 关系型数据库
2025 年最新华为 Java 面试题及答案,全方位打造面试宝典
Java面试高频考点与实践指南(150字摘要) 本文系统梳理了Java面试核心考点,包括Java基础(数据类型、面向对象特性、常用类使用)、并发编程(线程机制、锁原理、并发容器)、JVM(内存模型、GC算法、类加载机制)、Spring框架(IoC/AOP、Bean生命周期、事务管理)、数据库(MySQL引擎、事务隔离、索引优化)及分布式(CAP理论、ID生成、Redis缓存)。同时提供华为级实战代码,涵盖Spring Cloud Alibaba微服务、Sentinel限流、Seata分布式事务,以及完整的D
208 1
|
3月前
|
缓存 Java API
Java 面试实操指南与最新技术结合的实战攻略
本指南涵盖Java 17+新特性、Spring Boot 3微服务、响应式编程、容器化部署与数据缓存实操,结合代码案例解析高频面试技术点,助你掌握最新Java技术栈,提升实战能力,轻松应对Java中高级岗位面试。
340 0
|
12天前
|
安全 Java
Java之泛型使用教程
Java之泛型使用教程
120 10
|
3月前
|
安全 Java API
在Java中识别泛型信息
以上步骤和示例代码展示了怎样在Java中获取泛型类、泛型方法和泛型字段的类型参数信息。这些方法利用Java的反射API来绕过类型擦除的限制并访问运行时的类型信息。这对于在运行时进行类型安全的操作是很有帮助的,比如在创建类型安全的集合或者其他复杂数据结构时处理泛型。注意,过度使用反射可能会导致代码难以理解和维护,因此应该在确有必要时才使用反射来获取泛型信息。
126 11
|
4月前
|
算法 架构师 Java
Java 开发岗及 java 架构师百度校招历年经典面试题汇总
以下是百度校招Java岗位面试题精选摘要(150字): Java开发岗重点关注集合类、并发和系统设计。HashMap线程安全可通过Collections.synchronizedMap()或ConcurrentHashMap实现,后者采用分段锁提升并发性能。负载均衡算法包括轮询、加权轮询和最少连接数,一致性哈希可均匀分布请求。Redis持久化有RDB(快照恢复快)和AOF(日志更安全)两种方式。架构师岗涉及JMM内存模型、happens-before原则和无锁数据结构(基于CAS)。
119 5
|
4月前
|
Java API 微服务
2025 年 Java 校招面试全攻略:从面试心得看 Java 岗位求职技巧
《2025年Java校招最新技术要点与实操指南》 本文梳理了2025年Java校招的核心技术栈,并提供了可直接运行的代码实例。重点技术包括: Java 17+新特性(Record类、Sealed类等) Spring Boot 3+WebFlux响应式编程 微服务架构与Spring Cloud组件 Docker容器化部署 Redis缓存集成 OpenAI API调用 通过实际代码演示了如何应用这些技术,如Java 17的Record类简化POJO、WebFlux构建响应式API、Docker容器化部署。
161 5
|
4月前
|
缓存 NoSQL Java
Java Redis 面试题集锦 常见高频面试题目及解析
本文总结了Redis在Java中的核心面试题,包括数据类型操作、单线程高性能原理、键过期策略及分布式锁实现等关键内容。通过Jedis代码示例展示了String、List等数据类型的操作方法,讲解了惰性删除和定期删除相结合的过期策略,并提供了Spring Boot配置Redis过期时间的方案。文章还探讨了缓存穿透、雪崩等问题解决方案,以及基于Redis的分布式锁实现,帮助开发者全面掌握Redis在Java应用中的实践要点。
211 6
|
4月前
|
安全 Java API
2025 年 Java 校招面试常见问题及详细答案汇总
本资料涵盖Java校招常见面试题,包括Java基础、并发编程、JVM、Spring框架、分布式与微服务等核心知识点,并提供详细解析与实操代码,助力2025校招备战。
203 1
|
4月前
|
算法 Java 微服务
2025 年 Java 面试宝典社招春招秋招实操全方位攻略
2025年Java面试宝典涵盖核心技术及最新趋势,分为四大板块:1. Java基础:深入数据类型、多态等特性,结合学生信息管理等实例;2. JVM核心:解析内存模型与GC算法,附多线程转账等场景应用;3. 高并发方案:详解synchronized与线程池配置,提供Web服务器优化案例;4. Spring生态:剖析IoC/AOP原理,演示微服务架构实现。特别新增Java 17+特性实操,包括Record类、密封接口等语法糖,整合Spring Boot 3、响应式编程及云原生技术,通过订单状态机、API网关配置。
272 1
|
4月前
|
NoSQL Java 微服务
2025 年最新 Java 面试从基础到微服务实战指南全解析
《Java面试实战指南:高并发与微服务架构解析》 本文针对Java开发者提供2025版面试技术要点,涵盖高并发电商系统设计、微服务架构实现及性能优化方案。核心内容包括:1)基于Spring Cloud和云原生技术的系统架构设计;2)JWT认证、Seata分布式事务等核心模块代码实现;3)数据库查询优化与高并发处理方案,响应时间从500ms优化至80ms;4)微服务调用可靠性保障方案。文章通过实战案例展现Java最新技术栈(Java 17/Spring Boot 3.2)的应用.
239 9

热门文章

最新文章