Java泛型编程与多态、重载的同与不同

简介: 从字节码看Java泛型编程与多态、重载的同于不同。

泛型编程

泛型编程在某些语言中也称之为模板编程,比如C++,所以在泛型编程中见到的那个T也就是Template的首字母。
来看一个泛型编程的简单样例。

public static void main(String[] args) {
    StringBuilder sb = new StringBuilder("Test template");
    System.out.println(foo(sb).toString());
}

public static <T> T foo(T t) {
    // bla bla bla
    return t;
}

泛型编程与多态

在面向对象的编程中还有一个概念叫多态,利用这种概念,我们可以将泛型编程中的T替换成所有对象的基类Object,这在某种程度上同样能够达到泛型编程所达到的效果,如下方代码所示。

public static void main(String[] args) {
    StringBuilder sb = new StringBuilder("Test template");
    System.out.println(foo(sb).toString());
    System.out.println(foo2(sb).toString());
}

public static <T> T foo(T t) {
    // bla bla bla
    return t;
}

public static Object foo2(Object o) {
    // bla bla bla
    return o;
}

那么这两种方法有什么区别吗?

我们来看看字节码。

public class template.Template {
  public template.Template();
    Code:
       0: aload_0
       1: invokespecial #1                  // Method java/lang/Object."<init>":()V
       4: return

  public static void main(java.lang.String[]);
    Code:
       0: new           #2                  // class java/lang/StringBuilder
       3: dup
       4: ldc           #3                  // String Test template
       6: invokespecial #4                  // Method java/lang/StringBuilder."<init>":(Ljava/lang/String;)V
       9: astore_1
      10: getstatic     #5                  // Field java/lang/System.out:Ljava/io/PrintStream;
      13: aload_1
      14: invokestatic  #6                  // Method foo:(Ljava/lang/Object;)Ljava/lang/Object;
      17: checkcast     #2                  // class java/lang/StringBuilder
      20: invokevirtual #7                  // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
      23: invokevirtual #8                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V
      26: getstatic     #5                  // Field java/lang/System.out:Ljava/io/PrintStream;
      29: aload_1
      30: invokestatic  #9                  // Method foo2:(Ljava/lang/Object;)Ljava/lang/Object;
      33: invokevirtual #10                 // Method java/lang/Object.toString:()Ljava/lang/String;
      36: invokevirtual #8                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V
      39: return

  public static <T> T foo(T);
    Code:
       0: aload_0
       1: areturn

  public static java.lang.Object foo2(java.lang.Object);
    Code:
       0: aload_0
       1: areturn
}

foo方法调用时的字节码

14: invokestatic  #6                  // Method foo:(Ljava/lang/Object;)Ljava/lang/Object;
17: checkcast     #2                  // class java/lang/StringBuilder
20: invokevirtual #7                  // Method java/lang/StringBuilder.toString:()Ljava/lang/String;

foo2方法调用时的字节码

30: invokestatic  #9                  // Method foo2:(Ljava/lang/Object;)Ljava/lang/Object;
33: invokevirtual #10                 // Method java/lang/Object.toString:()Ljava/lang/String;

不难发现,main函数调用使用了泛型编程的foo方法时,其字节码已不再是T,而是替换为实际的StringBuilder类;反观foo2方法使用的还是Object,那它将是在运行时作类型转换。
结论:泛型编程是编译期替换;多态则是运行期作类型转换。

泛型编程与重载

什么是重载

不废话,形如下方代码即重载。

public static int add(int a, int b) {
    return a + b; 
}
    
public static double add(double a, double b) {
    return a + b;
}

重载的特点:两个及以上方法名相同;参数个数不同,参数顺序不同、类型不同,以上任一种及以上都可以构成重载。仅返回值不同不可构成重载。

测试的完整代码如下:

public static void main(String[] args) {
    StringBuilder sb = new StringBuilder("Test template");
    System.out.println(foo(sb).toString());
    System.out.println(foo2(sb).toString());
    int a = 1;
    int b = 2;
    System.out.println(add(a, b));
        
    double x = 1.1;
    double y = 1.2;
    System.out.println(add(x, y));
}

public static <T> T foo(T t) {
    // bla bla bla
    return t;
}
    
public static Object foo2(Object o) {
    // bla bla bla
    return o;
}

public static int add(int a, int b) {
    return a + b; 
}
    
public static double add(double a, double b) {
    return a + b;
}

接下来看看重载部分的字节码:

48: invokestatic  #11                 // Method add:(II)I
71: invokestatic  #17                 // Method add:(DD)D

总结:看来字节码已经做了相应的转换,重载和泛型编程相似,都是在编译期就做了替换,而不是在运行时,但是重载需要为每一种不同的参数类型重新编写代码,代码复用度不高。

总结

泛型编程和重载是在编译期作了类型替换,多态则是在运行期作类型转换。
泛型编程和多态代码复用度高,重载代码复用度低。

当然,泛型、重载、多态更有其它设计模式的意义,在此不作讨论。

目录
相关文章
|
1月前
|
Java 数据库连接 API
2025 更新必看:Java 编程基础入门级超级完整版指南
本教程为2025更新版Java编程基础入门指南,涵盖开发环境搭建(SDKMAN!管理JDK、VS Code配置)、Java 17+新特性(文本块、Switch表达式增强、Record类)、面向对象编程(接口默认方法、抽象类与模板方法)、集合框架深度应用(Stream API高级操作、并发集合)、模式匹配与密封类等。还包括学生成绩管理系统实战项目,涉及Maven构建、Lombok简化代码、JDBC数据库操作及JavaFX界面开发。同时提供JUnit测试、日志框架使用技巧及进阶学习资源推荐,助你掌握Java核心技术并迈向高级开发。
133 5
|
2月前
|
JavaScript 前端开发 Java
Java 编程进阶实操中工具集整合组件封装方法与使用指南详解
本文详细介绍Hutool工具集和图书管理系统相关组件的封装方法及使用示例。通过通用工具类封装(如日期格式化、字符串处理、加密等)、数据库操作封装(结合Hutool DbUtil与MyBatis)、前端Vue组件封装(图书列表与借阅表单)以及后端服务层封装(业务逻辑实现与REST API设计),帮助开发者提升代码复用性与可维护性。同时,提供最佳实践建议,如单一职责原则、高内聚低耦合、参数配置化等,助力高效开发。适用于Java编程进阶学习与实际项目应用。
106 10
|
1月前
|
Oracle Java 关系型数据库
java 编程基础入门级超级完整版教程详解
这份文档是针对Java编程入门学习者的超级完整版教程,涵盖了从环境搭建到实际项目应用的全方位内容。首先介绍了Java的基本概念与开发环境配置方法,随后深入讲解了基础语法、控制流程、面向对象编程的核心思想,并配以具体代码示例。接着探讨了常用类库与API的应用,如字符串操作、集合框架及文件处理等。最后通过一个学生成绩管理系统的实例,帮助读者将理论知识应用于实践。此外,还提供了进阶学习建议,引导学员逐步掌握更复杂的Java技术。适合初学者系统性学习Java编程。资源地址:[点击访问](https://pan.quark.cn/s/14fcf913bae6)。
119 2
|
22天前
|
缓存 安全 算法
2025 年 Java 秋招面试必看 Java 并发编程面试题实操篇
Java并发编程是Java技术栈中非常重要的一部分,也是面试中的高频考点。本文从基础概念、关键机制、工具类、高级技术等多个方面进行了介绍,并提供了丰富的实操示例。希望通过本文的学习,你能够掌握Java并发编程的核心知识,在面试中取得好成绩。同时,在实际工作中,也能够运用这些知识设计和实现高效、稳定的并发系统。
37 0
|
22天前
|
存储 安全 Java
2025 年 Java 秋招面试必看的 Java 并发编程面试题汇总
文章摘要: 本文系统梳理Java并发编程核心知识点,助力2025年秋招面试。内容涵盖:1)基础概念,包括线程/进程区别、创建线程的3种方式(Thread/Runnable/Callable)、6种线程状态及转换;2)关键机制,对比sleep()与wait()的锁行为差异,解释start()而非run()启动线程的原因;3)工具类与典型应用场景。通过技术原理与代码示例结合的方式,帮助开发者深入理解并发模型、线程同步等核心问题,为高并发系统设计打下坚实基础。(150字)
61 0
|
1月前
|
人工智能 Java API
Java并发编程之Future与FutureTask
本文深入解析了Future接口及其实现类FutureTask的原理与使用。Future接口定义了获取任务结果、取消任务及查询任务状态的规范,而FutureTask作为其核心实现类,结合了Runnable与Future的功能。文章通过分析FutureTask的成员变量、状态流转、关键方法(如run、set、get、cancel等)的源码,展示了异步任务的执行与结果处理机制。最后,通过示例代码演示了FutureTask的简单用法,帮助读者更直观地理解其工作原理。适合希望深入了解Java异步编程机制的开发者阅读。
|
Java 编译器
Java的重载与重写的区别
Java的重载与重写的区别
108 0
|
Java 编译器
【面试题精讲】Java重载和重写有什么区别?
【面试题精讲】Java重载和重写有什么区别?
|
前端开发 Java
店铺业务场景分析、BigDecimal是Java提供的一个不变的、任意精度的有符号十进制数对象。它提供了四个构造器,有两个是用BigInteger构造、接口怎么使用的、重载与重写的区别?分别是什么?
店铺业务场景分析 一、协同店铺、竞争店铺极海数据返回给前端数据结构不一样 导入的数据结构 很有可能和自定义采集得到的数据结构不一样
216 1
店铺业务场景分析、BigDecimal是Java提供的一个不变的、任意精度的有符号十进制数对象。它提供了四个构造器,有两个是用BigInteger构造、接口怎么使用的、重载与重写的区别?分别是什么?
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等

登录插画

登录以查看您的控制台资源

管理云资源
状态一览
快捷访问