码出高效:Java开发手册-第2章(11)

简介: 本章开始讲解面向对象思想,并以Java 为载体讲述面向对象思想在具体编程语言中的运用与实践。当前主流的编程语言有50 种左右,主要分为两大阵营:面向对象编程与面向过程编程。面向对象编程(Object-Oriented Programming,OOP)是划时代的编程思想变革,推动了高级语言的快速发展和工业化进程。OOP 的抽象、封装、继承、多态的理念使软件大规模化成为可能,有效地降低了软件开发成本、维护成本和复用成本。面向对象编程思想完全不同于传统的面向过程编程思想,使大型软件的开发就像搭积木一样隔离可控、高效简单,是当今编程领域的一股势不可......

2.6 泛型

      泛型的本质是类型参数化,解决不确定具体对象类型的问题。在面向对象编程语言中,允许程序员在强类型校验下定义某些可变部分,以达到代码复用的目的。泛型(generic)、天才(genius)、基因(gene)三个英文单词的词根都是gen,最神奇的是,它们无论是拼写还是发音都十分相像,在沟通中往往比较含糊。可以这样理解,泛型就是这些拥有天才基因的大师们发明的。

      Java 在引入泛型前,表示可变类型,往往存在类型安全的风险。举一个生活中的例子,微波炉最主要的功能是加热食物,即加热肉、加热汤都有可能。在没有泛型的场景中,往往会写出:

class Stove {

      public static Object heat(Object food) {

             System.out.println(food + "is done");

             return food;

      }

      public static void main(String[] args) {

             Meat meat = new Meat();

             meat = (Meat)Stove.heat(meat);

             Soup soup = new Soup();

             soup = (Soup)Stove.heat(soup);

      }

}

      为了避免给每种食材定义一个加热方法,如heatMeat()、heatSoup() 等,将heat()的参数和返回值定义为Object,用“向上转型”的方式,让其具备可以加热任意类型对象的能力。这种方式增强了类的灵活性,但却会让客户端产生困惑,因为客户端对加热的内容一无所知,在取出来时进行强制转换就会存在类型转换风险。泛型则可以完美地解决这个问题。

      泛型可以定义在类、接口、方法中,编译器通过识别尖括号和尖括号内的字母来解析泛型。在泛型定义时,约定俗成的符号包括:E 代表Element,用于集合中的元素;T 代表the Type of object,表示某个类;K 代表Key、V 代表Value,用于键值对元素。我们用一个示例彻底地记住泛型定义的概念,对泛型不再有恐惧心理。如果下面代码编译出错,请指出编译出错的位置在哪里:

public class GenericDefinitionDemo<T> {

      static <String, T, Alibaba> String get(String string, Alibaba alibaba) {

             return string;

      }

      public static void main(String[] args) {

             Integer first = 222;

             Long second = 333L;

             // 调用上方定义的get 方法

             Integer result = get(first, second);

      }

}

      事实上,以上代码编译正确且能够正常运行,get() 是一个泛型方法,first 并非是

java.lang.String 类型,而是泛型标识<String>,second 指代 Alibaba。get() 中其他有被用到的泛型符号并不会导致编译出错,类名后的T 与尖括号内的T 相同也是合法的。当然在实际应用时,并不会存在这样的定义方式,这里只是期望能够对以下几点加深理解:

(1)尖括号里的每个元素都指代一种未知类型。String 出现在尖括号里,它就不是java.lang.String,而仅仅是一个代号。类名后方定义的泛型<T> 和get() 前方定义的<T> 是两个指代,可以完全不同,互不影响。

(2)尖括号的位置非常讲究,必须在类名之后或方法返回值之前。

(3)泛型在定义处只具备执行Object 方法的能力。因此想在get() 内部执行string.longValue() + alibaba.intValue() 是做不到的,此时泛型只能调用Object 类中的方法,如toString()。

(4)对于编译之后的字节码指令,其实没有这些花头花脑的方法签名,充分说明了泛型只是一种编写代码时的语法检查。在使用泛型元素时,会执行强制类型转换:

INVOKESTATIC com/alibaba/easy/coding/generic/GenericDefinitionDemo.get

(Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;

CHECKCAST java/lang/Integer

      这就是坊间盛传的类型擦除。CHECKCAST 指令在运行时会检查对象实例的类型是否匹配,如果不匹配,则抛出运行时异常ClassCastException。与C++ 根据模板类生成不同的类的方式不同,Java 使用的是类型擦除的方式。编译后,get() 的参数是两个Object,返回值也是Object,尖括号里很多内容消失了,参数中也没有String 和Alibaba 两个类型。数据返回给Integer result 时,进行了类型强制转化。因此,泛型就是在编译期增加了一道检查而已,目的是促使程序员在使用泛型时安全放置和使用数据。使用泛型的好处包括:

  • 类型安全。放置的是什么,取出来的自然是什么,不用担心会抛出ClassCastException 异常。
  • 提升可读性。从编码阶段就显式地知道泛型集合、泛型方法等处理的对象类型是什么。
  • 代码重用。泛型合并了同类型的处理代码,使代码重用度变高。

      回到本节开头微波炉加热食材的例子,使用泛型可以很好地实现,示例代码如下:

public class Stove {

      public static <T> T heat(T food) {

             System.out.println(food + "is done");

             return food;

      }

      public static void main(String[] args) {

             Meat meat = new Meat();

             meat = Stove.heat(meat);

             Soup soup = new Soup();

             soup = Stove.heat(soup);

      }

}

      通过使用泛型,既可以避免对加热肉和加热汤定义两种不同的方法,也可以避免使用Object 作为输入和输出,带来强制转换的风险。只要这种强制转换的风险存在,依据墨菲定律,就一定会发生ClassCastException 异常。特别是在复杂的代码逻辑中,会形成网状的调用关系,如果任意使用强制转换,无论可读性还是安全性都存在问题。

      最后,泛型与集合的联合使用,可以把泛型的功能发挥到极致,很多程序员不清楚List、List<Object>、List<?> 三者的区别, 更加不能区分<? extends T> 与<? super T> 的使用场景。具体请参考第6.5 节。

目录
打赏
0
0
0
0
28
分享
相关文章
|
4月前
|
如何使用Java开发抖音API接口?
在数字化时代,社交媒体平台如抖音成为生活的重要部分。本文详细介绍了如何用Java开发抖音API接口,从创建开发者账号、申请API权限、准备开发环境,到编写代码、测试运行及注意事项,全面覆盖了整个开发流程。
620 10
一天成为Java开发高手:用飞算JavaAI实现十倍提效
“一天成为Java开发高手”曾被视为天方夜谭,但飞算JavaAI的出现改变了这一局面。这款AI开发助手通过智能引导、需求分析、自动化逻辑处理和完整代码工程生成,大幅简化了Java开发流程。它不仅帮助新手快速上手,还让资深开发者提高效率,减少调试时间。现在,参与“飞算JavaAI炫技赛”,展示你的开发实力,赢取丰厚奖品!
如何使用Java语言快速开发一套智慧工地系统
使用Java开发智慧工地系统,采用Spring Cloud微服务架构和前后端分离设计,结合MySQL、MongoDB数据库及RESTful API,集成人脸识别、视频监控、设备与环境监测等功能模块,运用Spark/Flink处理大数据,ECharts/AntV G2实现数据可视化,确保系统安全与性能,采用敏捷开发模式,提供详尽文档与用户培训,支持云部署与容器化管理,快速构建高效、灵活的智慧工地解决方案。
智慧产科一体化管理平台源码,基于Java,Vue,ElementUI技术开发,二开快捷
智慧产科一体化管理平台覆盖从备孕到产后42天的全流程管理,构建科室协同、医患沟通及智能设备互联平台。通过移动端扫码建卡、自助报道、智能采集数据等手段优化就诊流程,提升孕妇就诊体验,并实现高危孕产妇五色管理和孕妇学校三位一体化管理,全面提升妇幼健康宣教质量。
44 12
Java高级应用开发:基于AI的微服务架构优化与性能调优
在现代企业级应用开发中,微服务架构虽带来灵活性和可扩展性,但也增加了系统复杂性和性能瓶颈。本文探讨如何利用AI技术,特别是像DeepSeek这样的智能工具,优化Java微服务架构。AI通过智能分析系统运行数据,自动识别并解决性能瓶颈,优化服务拆分、通信方式及资源管理,实现高效性能调优,助力开发者设计更合理的微服务架构,迎接未来智能化开发的新时代。
菜鸟之路day02-04拼图小游戏开发一一JAVA基础综合项目
本项目基于黑马程序员教程,涵盖面向对象进阶、继承、多态等知识,历时约24小时完成。项目去除了登录和注册模块,专注于单机游戏体验。使用Git进行版本管理,代码托管于Gitee。项目包含窗体搭建、事件监听、图片加载与打乱、交互逻辑实现、菜单功能及美化界面等内容。通过此项目,巩固了Java基础并提升了实际开发能力。 仓库地址:[https://gitee.com/zhang-tenglan/puzzlegame.git](https://gitee.com/zhang-tenglan/puzzlegame.git)
46 6
【潜意识Java】javaee中的SpringBoot在Java 开发中的应用与详细分析
本文介绍了 Spring Boot 的核心概念和使用场景,并通过一个实战项目演示了如何构建一个简单的 RESTful API。
47 5
Java后端开发-使用springboot进行Mybatis连接数据库步骤
本文介绍了使用Java和IDEA进行数据库操作的详细步骤,涵盖从数据库准备到测试类编写及运行的全过程。主要内容包括: 1. **数据库准备**:创建数据库和表。 2. **查询数据库**:验证数据库是否可用。 3. **IDEA代码配置**:构建实体类并配置数据库连接。 4. **测试类编写**:编写并运行测试类以确保一切正常。
69 2

热门文章

最新文章

AI助理

你好,我是AI助理

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