转自
http://hellosure.github.io/java/2014/09/10/java-sugar/
语法糖
语法糖方便了程序员的开发,提高了开发效率,提升了语法的严谨也减少了编码出错误的几率。我们不仅仅在平时的编码中依赖语法糖,更要看清语法糖背后程序代码的真实结构,这样才能更好的利用它们。。
泛型
与C#中的泛型相比,Java的泛型可以算是“伪泛型”了。在C#中,不论是在程序源码中、在编译后的中间语言,还是在运行期泛型都是真实存在的。Java则不同,Java的泛型只在源代码存在,只供编辑器检查使用,编译后的字节码文件已擦除了泛型类型,同时在必要的地方插入了强制转型的代码。
public static void main(String[] args) {
List<String> stringList = new ArrayList<String>();
stringList.add("oliver");
System.out.println(stringList.get(0));
}
将上面的代码的字节码反编译后:
public static void main(String args[])
{
List stringList = new ArrayList();
stringList.add("oliver");
System.out.println((String)stringList.get(0));
}
自动拆箱/装箱
自动拆箱/装箱是在编译期,依据代码的语法,决定是否进行拆箱和装箱动作。 1. 装箱过程:把基本类型用它们对应的包装类型进行包装,使基本类型具有对象特征。 2. 拆箱过程:与装箱过程相反,把包装类型转换成基本类型。
需要注意的是:包装类型的“==”运算在没有遇到算数运算符的情况下不会自动拆箱,而其包装类型的equals()方法不会处理数据类型转换,所以:
Integer a = 1;
Integer b = 1;
Long c = 1L;
System.out.println(a == b);
System.out.println(c.equals(a));
这样的代码应该尽量避免自动拆箱与装箱。
循环历遍(foreach)
List<Integer> list = new ArrayList<Integer>();
for(Integer num : list){
System.out.println(num);
}
Foreach要求被历遍的对象要实现Iterable接口,由此可想而知,foreach迭代也是调用底层的迭代器实现的。反编译上面源码的字节码:
List list = new ArrayList();
Integer num;
Integer num;
for (Iterator iterator = list.iterator(); iterator.hasNext(); System.out.println(num)){
num = (Integer) iterator.next();
}
枚举
枚举类型其实并不复杂,在JVM字节码文件结构中,并没有“枚举”这个类型。 其实源程序的枚举类型,会在编译期被编译成一个普通的类。利用继承和反射,这是完全可以做到的。
看下面一个枚举类:
public enum EnumTest {
OLIVER,LEE;
}
反编译字节码后:
public final class EnumTest extends Enum {
private EnumTest(String s, int i) {
super(s, i);
}
public static EnumTest[] values() {
EnumTest aenumtest[];
int i;
EnumTest aenumtest1[];
System.arraycopy(aenumtest = ENUM$VALUES, 0,
aenumtest1 = new EnumTest[i = aenumtest.length], 0, i);
return aenumtest1;
}
public static EnumTest valueOf(String s) {
return (EnumTest) Enum.valueOf(EnumTest, s);
}
public static final EnumTest OLIVER;
public static final EnumTest LEE;
private static final EnumTest ENUM$VALUES[];
static {
OLIVER = new EnumTest("OLIVER", 0);
LEE = new EnumTest("LEE", 1);
ENUM$VALUES = (new EnumTest[] { OLIVER, LEE });
}
}
变长参数
变长参数允许我们传入到方法的参数是不固定个数。
对于这个方法:
public void foo(String str,Object...args){
}
我们可以这样调用:
foo("oliver");
foo("oliver",new Object());
foo("oliver",new Integer(1),"sss");
foo("oliver",new ArrayList(),new Object(),true,1);
参数args可以是任意多个。
其实,在编译阶段,args是会被编译成Object [] args。
public transient void foo(String s, Object aobj[])
{
}
这样,变长参数就可以实现了。 但是要注意的是,变长参数必须是方法参数的最后一项。
分享到:
相关推荐
有意思的是,在编程领域,除了语法糖,还有语法盐和语法糖精的说法,篇幅有限这里不做扩展了。但其实,Java虚拟机并不支持这些语法糖,这些语法糖在编译阶段就会被还原
java语法糖介绍
语法糖(Syntactic Sugar),也称糖衣语法,指在计算机语言中添加的 某种语法 ,这种语法对语言本身的功能来说没有什么影响,只是为了方便程序员进行开发,提高开发...所以在Java中真正支持语法糖的是Java编译器。
实现算法Java中的语法糖
Java 中的语法糖,真甜 语法糖 泛型 自动拆箱和自动装箱 枚举 内部类 变长参数 增强 for 循环 Switch 支持字符串和枚举 条件编译 断言 try-with-resources 字符串相加 学习语法糖的意义
【深入Java虚拟机(6)】Java语法糖编程开发技术共3页.pdf.zip
极客时间 | 深入拆解 Java 虚拟机讲堂 深入拆解 Java 虚拟机 文章详情2018-08-24 郑雨迪朗读人:郑雨迪 09′58′′ | 4.58M在前
- 泛型擦除前的例子把这段Java代码编译成Class文件,然后再用字节码反编译后,將会发现泛型都不见了,又变回了Java泛型出现之前的写法,泛型类型都变回了原
JavaGrande 扩展了 Java 编程语言以支持其他语法。 支持以下内容。 可观察的属性。 Grande 不仅允许您定义新属性,而且任何带有标准 getter 和 setter 的 api 都可以像访问属性一样进行访问。 可选的分号。 任何...
主要介绍了深入理解Java基础之try-with-resource语法糖,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
java基础语法讲解下集,适合初级人员入门学习
主要介绍了不了解这12个语法糖,别说你会Java,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
java基础语法讲解上集,适合初级人员入门学习
首先需要声明的是“语法糖”这个词绝非贬义词,它可以给我带来方便,是一种便捷的写法,编译器会帮我们做转换;而且可以提高开发编码的效率,在性能上也不会带来损失。这让java开发人员羡慕不已,呵呵。
比较详细
1.语法糖:CoreHighLevel\src\main\java\pers\li\syntacticsuger 1.语法糖使得程序员更容易编写程序,更清晰准确表达业务逻辑 2.语法糖不仅是语法的转换,也包括一些小的改进 3.JDk5,6,7,8,11长期稳定版 4.语法糖: 1...
Java 语法糖详解 集合 知识点/面试题总结 : Java 集合常见知识点&面试题总结(上) (必看 ) Java 集合常见知识点&面试题总结(下) (必看 ) Java 容器使用注意事项总结 源码分析 : ArrayList 源码+扩容机制分析 ...
Java 语法糖详解 集合 知识点/面试题总结: Java 集合常见知识点&面试题总结(上) (必看 ) Java 集合常见知识点&面试题总结(下) (必看 ) Java 容器使用注意事项总结 源码分析: ArrayList 源码+扩容机制分析 ...
主要介绍了Java语法糖之个数可变的形参的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
探讨Lambda是否为语法糖 语法 (1) 形参列表。如果没有参数()表示 (2) -> (3) 代码块 用Lambda语法写线程 快速启动(输出其实并不重要) mvn compile exec:java -Dexec.mainClass=...