首页 > 编程语言 >每日一道Java面试题:说一说Java中的泛型?

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

时间:2024-01-31 22:14:41浏览次数:31  
标签:面试题 Java ArrayList Object list 泛型 new public

写在开头

今天的每日一道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);   

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

泛型中的通配符

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

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();
}

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

【代码示例】

/**
* 实现泛型接口时指定类型为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);
        }
    }
}

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

静态泛型方法

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

【代码示例】

  //静态泛型方法
    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());

当我们添加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解析的过程中会进行泛型的擦除操作。

标签:面试题,Java,ArrayList,Object,list,泛型,new,public
From: https://www.cnblogs.com/JavaBuild/p/18000217

相关文章

  • Go官方放出泛型slices包
    阅读本文大概需要6分钟。slices 标准库是Go1.21新增的一个包,它提供了许多对切片(slices)进行常见操作的泛型函数,可以适用于任何元素类型的切片。切片是Go语言中一种重要的数据结构,它可以动态地存储和管理一组相同类型的元素。切片的底层实现是一个数组,但是切片可以根据需要......
  • Java中比较两个字符串==和.equals()区别
    ​在Java中,==和.equals()都是用于比较两个字符串是否相等的运算符,==比较的是两个字符串的引用地址,而.equals()比较的是两个字符串的内容。只有当两个字符串变量指向同一个字符串对象时,==和.equals()才会返回相同的结果 参考文档:Java中比较两个字符串==和.equals()区......
  • Java学习----基本语法
    1.注释有哪几种形式(1)单行注释:通常用于解释方法内某行代码的作用(2)多行注释:通常用于解释一段代码的作用(3)文档注释:通常用于生成Java开发文档2.标识符和关键字的区别(1)标识符就是一个名字(2)关键字不可以当做名字,不可修改,关键字是被赋予特殊含义的标识符3.自增(自减运算符)(1)后自增,......
  • Java的面向对象
    面向对象什么是面向对象面向对象思想:物以类聚,分类的思维方式。面向对象适合处理复杂的问题,适合处理需要多人协作的问题属性+方法=类面对象的本质:以类的方式组织代码,以对象的组织(封装)数据。三大特性:封装继承多态回顾方法及加深方法的定义:修饰符返回类型break:跳出sw......
  • Java 异常
    异常Error和ExceptionJava把异常当作对象处理,并定义一个基类java.lang.Throwable作为所有异常的超类异常类分为两大类:Error错误和Exception异常Error通常是灾难性的致命的错误,是程序无法控制和处理的,当出现这些异常时,JAVA虚拟机(JVM)一般会选择终止线程Exception通常情况下可以......
  • Java的方法详解
    Java方法详解什么是方法Java方法是语句的集合,它们在一块执行一个功能。方法是解决一类问题的步骤的有序组合方法包含于类或者对象中方法在程序中被创建,在其他地方被引用方法命名规则:驼峰原则publicclassDemo01{//main方法publicstaticvoidmain(String[]......
  • Java 数组
    数组数组是相同类型数据的有序集合。数组的声明和创建publicclassDemo01{//变量的类型变量的名字=变量的值//数组类型publicstaticvoidmain(String[]args){//首先声明数组变量int[]nums;//定义,首选这种intnums2[]......
  • Java的基础语法
    Java基础语法注释:书写注释是一个非常好的习惯,平时写代码一定要规范单行注释://多行注释:/**/文档注释JavaDoc:/***/标识符不能使用关键字作为变量名和方法名标识符大小写敏感首字母以字母、_以及$开头可以使用中文名,但是不建议,也不建议使用拼音,很low数据类型......
  • Java的流程控制
    Java流程控制Scanner对象我们通过Scanner类获取用户的输入。基础语法:Scanners=newScanner(System.in);通过Scanner类的next()与nextLine()方法获取输入的字符串,在读取前我们一般需要使用hasNext()与hasNextLine()判断是否还有输入的数据。next():一定要读取到有效字符才可以结束......
  • Java 的开始
    Java的jdk、jre、jvmjdk:JavaDelelopmentKitjre:Java运行环境jvm:JavaVirtualMachine卸载jdk删除java的安装目录删除Java_HOME删除path下关于Java的目录通过java-version检查安装jdk百度搜索JDK8,找到下载地址同意协议下载电脑对应的版本双击安装JDK......