Java基础---常见的语法糖

news2025/1/14 18:06:17

目录

典型回答

如何解语法糖

糖块一、switch 支持 String 与枚举

糖块二、泛型

糖块三、自动装箱与拆箱

糖块四、方法变长参数

糖块五、枚举

糖块六、内部类

糖块七、条件编译

糖块八、断言

糖块九、数值字面量

糖块十、for-each

糖块十一、try-with-resource

糖块十二、Lambda表达式


  • 典型回答

  • 语法糖(Syntactic sugar),指在计算机语言中添加的某种语法,这种语法对语言的功能并没有影响,但是更方便程序员使用
  • 虽然Java中有很多语法糖,但是Java虚拟机并不支持这些语法糖,所以这些语法糖在编译阶段就会被还原成简单的基础语法结构,这样才能被虚拟机识别,这个过程就是解语法糖
  • 如果看过Java虚拟机的源码,就会发现在编译过程中有一个重要的步骤就是调用desugar(),这个方法就是负责解语法糖的实现
  • 常见的语法糖有switch支持枚举及字符串、泛型、条件编译、断言、可变参数、自动装箱/拆箱、枚举、内部类、增强for循环、try-with-resources语句、lambda表达式等
  • 如何解语法糖

  • 语法糖的存在主要是方便开发人员使用
  • 但其实,Java虚拟机并不支持这些语法糖
  • 这些语法糖在编译阶段就会被还原成简单的基础语法结构,这个过程就是解语法糖
  • 说到编译,Java语言中javac命令可以将后缀名为.java的源文件编译为后缀名为.class的可以运行于Java虚拟机的字节码
  • 如果去看com.sun.tools.javac.main.JavaCompiler的源码,会发现在compile()中有一个步骤就是调用`desugar()`,这个方法就是负责解语法糖的实现的
  • 糖块一、switch 支持 String 与枚举

  • 前面提到过,从Java 7 开始,Java语言中的语法糖在逐渐丰富,其中一个比较重要的就是Java 7中 switch 开始支持 String
  • 在开始coding之前先科普下,Java中的 switch 自身原本就支持基本类型
  • 比如 int 、 char 等
  • 对于 int 类型,直接进行数值的比较
  • 对于 char 类型则是比较其ascii码
  • 所以,对于编译器来说, switch 中其实只能使用整型,任何类型的比较都要转换成整型
  • 比如 byte,short,char(ackii码是整型)以及 int
  • 那么接下来看下 switch 对 String 的支持,有以下代码:

  • 反编译后内容如下:

  • 看到这个代码,就知道原来字符串的switch是通过 equals() 和 hashCode() 方法来实现的
  • 还好 hashCode() 方法返回的是 int,而不是 long
  • 仔细看下可以发现,进行 switch 的实际是哈希值,然后通过使用 equals方法比较进行安全检查,这个检查是必要的,因为哈希可能会发生碰撞
  • 因此它的性能是不如使用枚举进行switch或者使用纯整数常量,但这也不是很差
  • 糖块二、泛型

  • 很多语言都是支持泛型的,但是很多人不知道的是,不同的编译器对于泛型的处理方式是不同的
  • 通常情况下,一个编译器处理泛型有两种方式:Code specialization 和 Code sharing
  • C++和C#是使用 Code specialization 的处理机制
  • 而Java使用的是 Code sharing 的机制
  • Code sharing方式为每个泛型类型创建唯一的字节码表示,并且将该泛型类型的实例都映射到这个唯一的字节码表示上
  • 将多种泛型类形实例映射到唯一的字节码表示是通过类型擦除(type erasue)实现的
  • 也就是说,对于Java虚拟机来说,他根本不认识 Map<String, String> map 这样的语法
  • 需要在编译阶段通过类型擦除的方式进行解语法糖
  • 类型擦除的主要过程如下:
  • 1.将所有的泛型参数用其最左边界(最顶级的父类型)类型替换
  • 2.移除所有的类型参数
  • 有以下代码:

  • 解语法糖之后会变成:

  • 有以下代码:

  • 类型擦除后会变成:

  • 虚拟机中没有泛型,只有普通类和普通方法,所有泛型类的类型参数在编译时都会被擦除,泛型类并没有自己独有的 Class 类对象
  • 比如并不存在 List<String>.class 或是 List<Integer>.class,而只有 List.class
  • 糖块三、自动装箱与拆箱

  • 自动装箱就是Java自动将原始类型值转换成对应的对象
  • 比如将int的变量转换成Integer对象,这个过程叫做装箱
  • 反之将Integer对象转换成int类型值,这个过程叫做拆箱
  • 因为这里的装箱和拆箱是自动进行的非人为转换,所以就称作为自动装箱和拆箱
  • 原始类型byte, short, char, int, long, float, double 和 boolean 对应的封装类为Byte, Short, Character, Integer, Long, Float, Double, Boolean
  • 先来看个自动装箱的代码:

  • 反编译后代码如下:

  • 再来看个自动拆箱的代码:

  • 反编译后代码如下:

  • 从反编译得到内容可以看出,在装箱的时候自动调用的是 Integer 的 valueOf(int) 方法
  • 而在拆箱的时候自动调用的是 Integer 的 intValue 方法
  • 所以装箱过程是通过调用包装器的valueOf方法实现的,而拆箱过程是通过调用包装器的 xxxValue 方法实现的
  • 糖块四、方法变长参数

  • 可变参数( variable arguments )是在Java 1.5中引入的一个特性
  • 它允许一个方法把任意数量的值作为参数
  • 可变参数在被使用的时候,他首先会创建一个数组,数组的长度就是调用该方法是传递的实参的个数,然后再把参数值全部放到这个数组当中,然后再把这个数组作为参数传递到被调用的方法中
  • 糖块五、枚举

  • 在Java中,枚举是一种特殊的数据类型,用于表示有限的一组常量
  • 枚举常量是在枚举类型中定义的,每个常量都是该类型的一个实例
  • Java中的枚举类型是一种安全而优雅的方式来表示有限的一组值
  • 要想看源码,首先得有一个类吧,那么枚举类型到底是什么类呢?
  • 是 enum 吗?答案很明显不是,enum 就和 class 一样,只是一个关键字,他并不是一个类,那么枚举是由什么类维护的呢,简单的写一个枚举:

  • 然后使用反编译,看看这段代码到底是怎么实现的,反编译后代码内容如下:

  • 通过反编译后代码可以看到,public final class T extends Enum
  • 说明,该类是继承了 Enum 类的,同时 final 关键字告诉我们这个类也是不能被继承的
  • 当使用 enmu 来定义一个枚举类型的时候,编译器会自动帮我们创建一个 final 类型的类继承 Enum 类,所以枚举类型不能被继承
  • 糖块六、内部类

  • 内部类又称为嵌套类,可以把内部类理解为外部类的一个普通成员
  • 内部类之所以也是语法糖,是因为它仅仅是一个编译时的概念
  • outer.java 里面定义了一个内部类 inner ,一旦编译成功,就会生成两个完全不同的 .class 文件了,分别是 outer.class 和 outer$inner.class
  • 所以内部类的名字完全可以和它的外部类名字相同

  • 以上代码编译后会生成两个class文件:OutterClass$InnerClass.class、OutterClass.class
  • 当尝试对 OutterClass.class 文件进行反编译的时候,命令行会打印以下内容: Parsing OutterClass.class...Parsing inner class OutterClass$InnerClass.class... Generating OutterClass.jad
  • 他会把两个文件全部进行反编译,然后一起生成一个 OutterClass.jad 文件
  • 文件内容如下:

  • 糖块七、条件编译

  • 一般情况下,程序中的每一行代码都要参加编译
  • 但有时候出于对程序代码优化的考虑,希望只对其中一部分内容进行编译,此时就需要在程序中加上条件,让编译器只对满足条件的代码进行编译,将不满足条件的代码舍弃,这就是条件编译
  • 如在C或CPP中,可以通过预处理语句来实现条件编译
  • 其实在Java中也可实现条件编译
  • 先来看一段代码:

  • 反编译后代码如下:

  • 首先发现,在反编译后的代码中没有 System.out.println("Hello, ONLINE!"); ,这其实就是条件编译
  • 当 if(ONLINE) 为false的时候,编译器就没有对其内的代码进行编译
  • 所以,Java语法的条件编译,是通过判断条件为常量的if语句实现的
  • 其原理也是Java语言的语法糖
  • 根据if判断条件的真假,编译器直接把分支为false的代码块消除
  • 通过该方式实现的条件编译,必须在方法体内实现,而无法在正整个Java类的结构或者类的属性上进行条件编译,这与C/C++的条件编译相比,确实更有局限性
  • 在Java语言设计之初并没有引入条件编译的功能,虽有局限,但是总比没有更强
  • 糖块八、断言

  • 在Java中,assert 关键字是从JAVA SE 1.4 引入的,为了避免和老版本的Java代码中使用了 assert 关键字导致错误,Java在执行的时候默认是不启动断言检查的(这个时候,所有的断言语句都将忽略!),如果要开启断言检查,则需要用开关 -enableassertions 或 -ea 来开启
  • 反编译之后的代码要比我们自己的代码复杂的多,使用了assert这个语法糖我们节省了很多代码
  • 其实断言的底层实现就是if语言,如果断言结果为true,则什么都不做,程序继续执行,如果断言结果为false,则程序抛出AssertError来打断程序的执行
  • -enableassertions 会设置$assertionsDisabled字段的值
  • 糖块九、数值字面量

  • 在java 7中,数值字面量,不管是整数还是浮点数,都允许在数字之间插入任意多个下划线
  • 这些下划线不会对字面量的数值产生影响,目的就是方便阅读
  • 比如:

  • 反编译后:

  • 反编译后就是把 _ 删除了
  • 也就是说编译器并不认识在数字字面量中的 _ ,需要在编译阶段把他去掉
  • 糖块十、for-each

  • 增强for循环( for-each )日常开发经常会用到的,他会比for循环要少写很多代码,那么这个语法糖背后是如何实现的呢?
  • 反编译后代码很简单,for-each的实现原理其实就是使用了普通的for循环和迭代器
  • 糖块十一、try-with-resource

  • Java里,对于文件操作IO流、数据库连接等开销非常昂贵的资源,用完之后必须及时通过close方法将其关闭,否则资源会一直处于打开状态,可能会导致内存泄露等问题
  • 关闭资源的常用方式就是在 finally 块里是释放,即调用 close 方法
  • 比如经常会写这样的代码:

  • 从Java 7开始,jdk提供了一种更好的方式关闭资源,使用 try-with-resources 语句,改写一下上面的代码,效果如下:

  • 这简直是一大福音啊,虽然之前一般使用 IOUtils 去关闭流,并不会使用在 finally 中写很多代码的方式,但是这种新的语法糖看上去好像优雅很多呢
  • 看下他的背后:

  • 其实背后的原理也很简单,那些没有做的关闭资源的操作,编译器都帮我们做了
  • 所以再次印证了,语法糖的作用就是方便程序员的使用,但最终还是要转成编译器认识的语言
  • 糖块十二、Lambda表达式

  • 关于lambda表达式,有人可能会有质疑,因为网上有人说他并不是语法糖
  • Labmda表达式不是匿名内部类的语法糖,但是他也是一个语法糖
  • 实现方式其实是依赖了几个JVM底层提供的lambda相关api
  • 为啥说他并不是内部类的语法糖呢,前面讲内部类说过,内部类在编译之后会有两个class文件,但是包含lambda表达式的类编译后只有一个文件
  • lambda表达式的实现其实是依赖了一些底层的api,在编译阶段,编译器会把lambda表达式进行解糖,转换成调用内部api的方式

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/724543.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

TigerBot和ChatGLM-6B大语言模型

目录 1 TigerBot大语言模型 1.1 环境安装 1.2 模型下载 1.2.1 hugging face 网站下载 1.2.2 百度网盘下载 1.3 推理 2 ChatGLM大语言模型 2.1环境搭建 2.2 模型下载 2.3 推理 1 TigerBot大语言模型 虎博科技”发布自研多模态大模型TigerBot&#xff0c;开源模型、代码…

Focal Loss损失函数

目录 前言 交叉熵损失函数 平衡交叉熵 Focal Loss 代码实现 前言 Focal loss是一个常用的解决类别不平衡问题的损失函数&#xff0c;由何恺明提出的&#xff08;论文名称&#xff1a;Focal Loss for Dense Object Detection&#xff09;&#xff0c;用于图像领域解决one-…

学习系统编程No.29【线程执行过程之页表详解】

引言&#xff1a; 北京时间&#xff1a;2023/7/3/14:09&#xff0c;刚睡醒&#xff0c;放假在家起床时间确实不怎么好调整&#xff0c;根本固定不了一点&#xff0c;当然通俗点说也就是根本起不来&#xff0c;哈哈哈&#xff0c;已经很少见到那种7点起来码字的情形了&#xff…

NanopcT4 系统 人脸检测实验 超详细教程 代码及操作步骤

文章目录 1.NanopcT4 系统烧写详细操作步骤2.vim 使用与 gcc 使用3.makefile 使用4.GPIO 引脚查看与连接5.使用vim 编写 LED 灯闪烁c 语言程序test1.c6.使用vim 编写 LED 灯闪烁shell 脚本程序test2.sh7.在ARM 上实现人脸检测 1.NanopcT4 系统烧写详细操作步骤 1)准备一张 8G …

口语理解任务源码详解系列(二)利用seq2seq-attention模型实现

利用seq2seq-attention模型实现 写在前面 在前文介绍了项目的数据集构建&#xff1a;传送门 本文利用seq2seq-attention实现&#xff1a;实现细节请参考论文&#xff1a;《Attention-Based Recurrent Neural Network Models for Joint Intent Detection and Slot Filling》 1.意…

喜羊羊贴吧顶帖软件实战教学

喜羊羊贴吧顶帖软件实战教学#贴吧顶帖#贴吧推广 大家好&#xff0c;欢迎来到百收网SEO这期视频&#xff0c;给大家更新一下百度贴最新的一个顶帖视频教程。首先我们今天用的顶帖软件是我们的喜羊羊173 的一个版本&#xff0c;软件的话在我们的群文件去下载&#xff0c;就是我们…

如何查看货物的物流状态

想不想有一个一键批量查询快递号的软件&#xff1f;今天&#xff0c;小编向您介绍一款软件&#xff1a;“固乔快递查询助手”&#xff0c;该软件是固乔工作室正式推出的专业快递和物流单号出货信息批量查询软件。这款软件功能实用&#xff0c;操作简单&#xff0c;页面简单&…

使用 Jackson 库对日期时间的动态序列化反序列化操作

0.背景 因某项目中的数据报表功能在创建年报 和月报时需要生成不同的日期格式&#xff0c;但数据结构未变&#xff0c;为避免类的冗余定义&#xff0c;故使用如下方式来动态设置日期格式&#xff0c;在不同报表是使用不同格式的时间格式来保存数据。 1.代码介绍 PS:此介绍有Cha…

深度学习技巧应用23-利用latex对深度学习各种网络模型的图像进行绘制

大家好,我是微学AI,今天给大家介绍一下深度学习技巧应用23-利用latex对深度学习各种网络模型的图像进行绘制,大家知道那些好看的模型结构图是怎么画的吗?今天就给大家手把手的利用latex绘画出深度学习模型图出来,我将利用latex画出AlexNet,LeNet,VGG16,U-Net高清的网络模型…

微服务一 实用篇 - 5.1 ElasticSearch安装

《微服务一 实用篇 - 5.1 ElasticSearch安装》 提示: 本材料只做个人学习参考,不作为系统的学习流程,请注意识别!!! 《微服务一 实用篇 - 5.1 ElasticSearch安装》 《微服务一 实用篇 - 5.1 ElasticSearch安装》1.部署单点es1.1.创建网络1.2.加载镜像1.3.运行 2.部署kibana2.1…

达芬奇快编键盘使用指南——个人白皮书分享

文章目录 序言常见问题&#xff08;陆续更新&#xff09;1.没有指示灯&#xff0c;不知道怎么开机 功能分区和翻译查询A快编工具&#xff08;Smart Insrt&#xff09;B修剪工具&#xff08;Trim In&#xff09;C多功能按键&#xff08;Esc&#xff09;D多机位区&#xff08;Cam…

新加坡访问学者签证申请的三个注意事项

新加坡是一个独特而美丽的国家&#xff0c;吸引了许多学者和研究人员前来访问和交流。如果您计划前往新加坡进行学术交流&#xff0c;下面是知识人网小编整理的关于新加坡访问学者签证申请的三个注意事项&#xff0c;希望对您有所帮助。 1. 签证申请和文件准备&#xff1a; 在…

Scala入门到放弃—01—概述

文章目录 概述什么是是Scala&#xff1f;为什么要使用Scala? 配置环境安装测试附 基本语法定义变量基本数据类型lazy在Scala中的应用 概述 什么是是Scala&#xff1f; https://www.scala-lang.org/ Scala combines object-oriented and functional programming in one conci…

ELK实验部署过程

ELK集群部署环境准备 配置ELK日志分析系统 192.168.1.51 elk-node1 es、logstash、kibana 192.168.1.52 elk-node2 es、logstash 192.168.1.53 apache logstash &#xff08;我这里是把虚拟机的配置全部都改为2核3G的&#xff09; 2台linux 第1台&#xff1a;elk-nod…

大数据开发工程师前景如何?

大数据需求越来越多&#xff0c;只有技术在手不愁找不到工作。 大数据工程师的技术要求如下&#xff1a; 1、掌握至少一种数据库开发技术&#xff1a;Oracle、Teradata、DB2、Mysql等&#xff0c;灵活运用SQL实现海量数据ETL加工处理&#xff1b; 2、熟悉Linux系统常规shell…

软件工程——第8章维护知识点整理

本专栏是博主个人笔记&#xff0c;主要目的是利用碎片化的时间来记忆软工知识点&#xff0c;特此声明&#xff01; 文章目录 1.维护阶段的基本任务是&#xff1f; 2.大型软件的维护成本大概是开发成本的几倍&#xff1f; 3.什么是软件维护&#xff1f; 4.软件维护分为哪几类…

python代码练习:猜成语游戏

python代码练习&#xff1a;猜成语游戏 题目结果展示源代码 题目 成语填填乐&#xff0c;随机输出一条包含一个空格的成语&#xff0c;填写答案并判断是否正确&#xff0c;正确加2分&#xff0c;输出“正确&#xff0c;你真棒”&#xff0c;错误减2分&#xff0c;输出“错了”…

git tag详解

文章目录 前言1. Git tag的基本概念和用法1.1. 什么是 Git tag&#xff1f;1.2. Git 标签有什么作用&#xff1f; 2. 创建 Git tag2.1. 创建轻量级tag2.2. 创建带有注释的tag2.3. 编辑已有的tag 3. 列出和检出 Git tag3.1. 列出所有tag3.3. 列出匹配的tag3.4. 检出tag 4. 将 Gi…

可视化 | Flask+Mysql+PyEcharts 电影Top250数据分析系统

文章目录 &#x1f3f3;️‍&#x1f308; 1. 数据库数据表1. 电影信息数据表2. 用户数据表 &#x1f3f3;️‍&#x1f308; 2. 各子界面1. 登陆界面2. 注册界面3. 电影信息概要界面4. 搜索界面5. 各年份上映电影数量6. 电影榜单TOP107. 评价人数TOP208. 地区电影TOP109. 电影…

uniapp 微信小程序导航功能(单个地址)

获取终点的坐标&#xff0c;根据终点的坐标&#xff0c;终点名称&#xff0c;终点详细地址&#xff0c;调起地图导航到第三方APP 1、针对单个地址导航 <template><view click"toGetLocation"></view><view click"toNavigation">&…