写在开头

今天的每日一道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面试题:说一说Java中的泛型?的更多相关文章

  1. java面试题—精选30道Java笔试题解答(二)

    摘要: java面试题-精选30道Java笔试题解答(二) 19. 下面程序能正常运行吗() public class NULL { public static void haha(){ System ...

  2. 最有价值的50道java面试题 适用于准入职Java程序员

    下面的内容是对网上原有的Java面试题集及答案进行了全面修订之后给出的负责任的题目和答案,原来的题目中有很多重复题目和无价值的题目,还有不少的参考答案也是错误的,修改后的Java面试题集参照了JDK最 ...

  3. Java面试题全集(上)(中)(下) (转)+自己总结

    Java面试题 自己总总结 https://www.cnblogs.com/songanwei/p/9366427.html Java面试题全集(上) https://blog.csdn.net/ja ...

  4. 【Java面试题系列】:Java基础知识常见面试题汇总 第一篇

    文中面试题从茫茫网海中精心筛选,如有错误,欢迎指正! 1.前言 ​ 参加过社招的同学都了解,进入一家公司面试开发岗位时,填写完个人信息后,一般都会让先做一份笔试题,然后公司会根据笔试题的回答结果,确定 ...

  5. 【Java面试题系列】:Java基础知识常见面试题汇总 第二篇

    文中面试题从茫茫网海中精心筛选,如有错误,欢迎指正! 第一篇链接:[Java面试题系列]:Java基础知识常见面试题汇总 第一篇 1.JDK,JRE,JVM三者之间的联系和区别 你是否考虑过我们写的x ...

  6. java面试题—精选30道Java笔试题解答(一)

    下面都是我自己的答案非官方,仅供参考,如果有疑问或错误请一定要提出来,大家一起进步啦~~~ 1. 下面哪些是Thread类的方法() A start() B run() C exit() D getP ...

  7. Java面试题整理:这些Java程序员面试中经常遇见的题目,必须掌握才能有好结果

    1.是否可以从一个static方法内部发出对非static方法的调用? 不可以.因为非static方法是要与对象关联在一起的,必须创建一个对象后,才可以在该对象上进行方法调用,而static方法调用时 ...

  8. 【Java面试题系列】:Java中final finally finalize的区别

    本篇为[Java面试题系列]第三篇,文中如有错误,欢迎指正. 第一篇链接:[Java面试题系列]:Java基础知识常见面试题汇总 第一篇 第二篇链接:[Java面试题系列]:Java基础知识常见面试题 ...

  9. 【Java面试题】56 在JAVA中如何跳出当前的多重嵌套循环?

    在Java中,要想跳出多重循环,可以在外面的循环语句前定义一个标号,然后在里层循环体的代码中使用带有标号的break 语句,即可跳出外层循环.例如, public class xunhuan { pu ...

  10. java面试题汇总,不断更新中。。。

    JVM,并发,锁相关: 1.请你谈谈对volatile的理解,volatile是否存在伪共享问题. 2.cas你知道吗? 3.原子类AtomicInteger的ABA问题谈谈?原子更新引用知道吗? 4 ...

随机推荐

  1. signed main 和 int main 的区别

    事实上只是因为有人直接 #define int long long 了...然后int main改成signed main就行了 #define int long long ... signed ma ...

  2. 领域驱动设计(DDD)实践之路(二):事件驱动与CQRS

    本文首发于 vivo互联网技术 微信公众号 链接: https://mp.weixin.qq.com/s/Z3uJhxJGDif3qN5OlE_woA作者:wenbo zhang [领域驱动设计实践之 ...

  3. Hbase结构和原理

     Hbase是什么? HBase是一种构建在Hadoop HDFS之上的分布式.面向列的存储系统.在需要实时读写.随机访问超大规模数据集时,可以使用HBase. HBase依赖Zookeeper,默认 ...

  4. 如何用 Serverless 一键部署 Stable Diffusion?

    思路 其实很简单, 我们只需要将镜像里面的动态路径映射到 NAS文件存储里面即可,利用 NAS 独立存储文件模型,扩展,语言包等,并且我们可以为管理 NAS 单独配置一个可视化的后台,用简单的文件上传 ...

  5. vant-list实现下拉加载更多

    1 <template> 2 <div class="home-wrapper"> 3 <div class="swipe-box" ...

  6. C# 序列化器

    理论知识: 序列化是指将对象转换成字节流,从而存储对象或将对象传输到内存.数据库或文件的过程. 它的主要用途是保存对象的状态,以便能够在需要时重新创建对象. 反向过程称为"反序列化" ...

  7. git或gitee 提交代码到远程仓库

    本文为博主原创,未经允许不得转载: 1. 选中远程仓库,并fork 指定的项目到自己的私仓: fork 之后,打开我的仓库便能看到刚刚fork 的项目. 2. clone 项目代码到自己电脑的本地仓库 ...

  8. 基于Html+腾讯云播SDK开发的m3u8播放器

    周末业余时间在家无事,学习了一下腾讯的云播放sdk,并制作了一个小demo(m3u8播放器),该在线工具是基于腾讯的云播sdk开发的,云播sdk非常牛,可以支持多种播放格式. 预览地址 m3u8pla ...

  9. 【mysql】 解决 auto_increment 字段 Column count doesn't match value count at row 1

    1, 表结构   man +-------+-------------+------+-----+---------+----------------+| id | int(11) | NO | PR ...

  10. Go——语言特性

    golang 简介 来历 很久以前,有一个IT公司,这公司有个传统,允许员工拥有20%自由时间来开发实验性项目.在2007的某一天,公司的几个大牛,正在用c++开发一些比较繁琐但是核心的工作,主要包括 ...