在Switch中可以使用的类型有枚举、字符串类型与整形int类型,下面来具体看这几个类型。

1、switch为枚举类型

枚举类:

enum Fruit {
    APPLE,ORINGE
}  

调用javac生成的代码如下:

enum Fruit{
	private <init>(/*synthetic*/ String $enum$name,
			       /*synthetic*/ int $enum$ordinal) {
	    super($enum$name, $enum$ordinal);
	}
	 /*public static final*/ APPLE /* = new Fruit("APPLE", 0) */
	 /*public static final*/ ORINGE /* = new Fruit("ORINGE", 1) */
	 /*synthetic*/ private static final Fruit[] $VALUES =
	                     new Fruit[]{Fruit.APPLE, Fruit.ORINGE}
	public static Fruit[] values() {
	    return (Fruit[])$VALUES.clone();
	}
	public static Fruit valueOf(String name) {
	    return (Fruit)Enum.valueOf(Fruit.class, name);
	}
}

Enum在Java中也被当作一个类来处理,并且Enum也算是Java中的语法糖,主要通过Javac中的Lower类来处理枚举这个语法糖的。

查看javac的英文说明,如下:

This map gives a translation table to be used for enum switches.

For each enum that appears as the type of a switch expression, we maintain an EnumMapping(举例) to assist in the translation, as exemplified by the following example:

we translate

Fruit fruit = Fruit.APPLE;
switch (fruit) {
case APPLE:
    System.out.println("apple");
    break;
case ORANGE:
    System.out.println("orange");
    break;
default:
    System.out.println("unknow");
}

into

switch (com.test19.TestEnumClass$1.$SwitchMap$com$test19$Fruit[(fruit).ordinal()]) {
case 1:
    System.out.println("apple");
    break;

case 2:
    System.out.println("orange");
    break;

default:
    System.out.println("unknow");

}

with the auxiliary table initialized as follows:

/*synthetic*/ class TestEnumClass$1 {
		    /*synthetic*/ static final int[] $SwitchMap$com$test19$Fruit = new int[Fruit.values().length];
		    static {
		        try {
		            com.test19.TestEnumClass$1.$SwitchMap$com$test19$Fruit[Fruit.APPLE.ordinal()] = 1;
		        } catch (NoSuchFieldError ex) {
		        }
		        try {
		            com.test19.TestEnumClass$1.$SwitchMap$com$test19$Fruit[Fruit.ORANGE.ordinal()] = 2;
		        } catch (NoSuchFieldError ex) {
		        }
		    }
}

class EnumMapping provides mapping data and support methods for this translation.

2、switch为字符串类型

下面继续来看switch对于字符串的处理。

The general approach used is to translate a single string switch statement into a series of two chained switch statements:

使用的一般方法是将一个字符串类型的switch语句翻译为两个switch语句:

the first a synthesized statement switching on the argument string's hash value and
computing a string's position in the list of original case labels, if any, followed by a second switch on the
computed integer value.

The second switch has the same code structure as the original string switch statement
except that the string case labels are replaced with positional integer constants starting at 0.

The first switch statement can be thought of as an inlined map from strings to their position in the case
label list. An alternate implementation would use an actual Map for this purpose, as done for enum switches.

With some additional effort, it would be possible to use a single switch statement on the hash code of the
argument, but care would need to be taken to preserve the proper control flow in the presence of hash
collisions and other complications, such as fallthroughs. Switch statements with one or two
alternatives could also be specially translated into if-then statements to omit the computation of the hash
code.

The generated code assumes that the hashing algorithm of String is the same in the compilation environment as
in the environment the code will run in. The string hashing algorithm in the SE JDK has been unchanged
since at least JDK 1.2. Since the algorithm has been specified since that release as well, it is very
unlikely to be changed in the future.

Different hashing algorithms, such as the length of the strings or a perfect hashing algorithm over the
particular set of case labels, could potentially be used instead of String.hashCode.

举一个具体的例子,如下:

class TestStringSwitch {
	public void test() {
		String fruit = "";
		switch (fruit) {
		case "banana":
		case "apple":
			System.out.println("banana or orange");
			break;
		case "orange":
			System.out.println("orange");
			break;
		default:
			System.out.println("default");
			break;
		}
	}
} 

翻译后的最终结果为:

    /*synthetic*/ final String s99$ = (fruit);
    /*synthetic*/ int tmp99$ = -1;
    switch (s99$.hashCode()) {
    case -1396355227:
        if (s99$.equals("banana")) tmp99$ = 0;
        break;

    case 93029210:
        if (s99$.equals("apple")) tmp99$ = 1;
        break;

    case -1008851410:
        if (s99$.equals("orange")) tmp99$ = 2;
        break;

    }
    switch (tmp99$) {
    case 0:
    case 1:
        System.out.println("banana or orange");
        break;

    case 2:
        System.out.println("orange");
        break;

    default:
        System.out.println("default");
        break;

    }

哈希函数在映射的时候可能存在冲突,多个字符串的哈希值可能是一样的,所以还要通过字符串比较来保证。

 

  

Javac语法糖之EnumSwitch的更多相关文章

  1. Javac语法糖之内部类

    在Javac中解语法糖主要是Lower类来完成,调用这个类的入口函数translateTopLevelClass即可.这个方法只是JavacCompiler类的desugar方法中进行了调用. 首先来 ...

  2. Javac语法糖之增强for循环

    加强的for循环有两种,遍历数组和实现了Iterable接口的容器.javac通过visitForeachLoop()方法来实现解语法糖,代码如下: /** Translate away the fo ...

  3. Javac语法糖之其它

    1.变长参数 class VarialbeArgumentsDemo { public static void doWork(int... a) {//可变参数 } public static voi ...

  4. Javac语法糖之TryCatchFinally

    https://docs.oracle.com/javase/specs/jls/se7/html/jls-14.html#jls-14.20.3 Optionally replace a try s ...

  5. Javac语法糖之Enum类

    枚举类在Javac中是被当作类来看待的. An enum type is implicitly final unless it contains at least one enum constant ...

  6. Java中的10颗语法糖

    语法糖(Syntactic Sugar):也称糖衣语法,指在计算机语言中添加的某种语法,这种语法对语言的功能没有影响,但是更方便程序员使用.通常来说,使用语法糖能够增加程序的可读性,减少程序代码出错的 ...

  7. 转:【深入Java虚拟机】之六:Java语法糖

    转载请注明出处:http://blog.csdn.net/ns_code/article/details/18011009 语法糖(Syntactic Sugar),也称糖衣语法,是由英国计算机学家P ...

  8. 【转】剖析异步编程语法糖: async和await

    一.难以被接受的async 自从C#5.0,语法糖大家庭又加入了两位新成员: async和await. 然而从我知道这两个家伙之后的很长一段时间,我甚至都没搞明白应该怎么使用它们,这种全新的异步编程模 ...

  9. Java的语法糖

    1.前言 本文记录内容来自<深入理解Java虚拟机>的第十章早期(编译期)优化其中一节内容,其他的内容个人觉得暂时不需要过多关注,比如语法.词法分析,语义分析和字节码生成的过程等.主要关注 ...

随机推荐

  1. windows下命令提示符中有空格路径的解决方法

    1)用缩写.比如c:\Program Files 缩写为c:\Progra~1 再来刨根问底查查这个命名是否有规则,于是找到: 文件夹(sub-directry)名称,以前是不允许带空白的,后来允许带 ...

  2. web service 项目 和 普通 web项目 的 区别

    web service 面向的是开发者(需要再次开发) 普通web 面向的是用户(直接使用)

  3. Android+PHP开发最佳实践

    本书以一个完整的微博应用项目实例为主线,由浅入深地讲解了Android客户端开发和PHP服务端开发的思路和技巧.从前期的产品设计.架构设计,到客户端和服务器的编码实现,再到性能测试和系统优化,以及最后 ...

  4. 解决Error creating bean with name 'huayuanjingguanDaoimp' defined in file [D:\apache-tomcat-7.0.52\webapps\landscapings\WEB-INF\classes\com\itheima\landscaping\dao\imp\huayuanjingguanDaoimp.class]: Invo

    问题描述: 10:23:13,585 ERROR ContextLoader:307 - Context initialization failedorg.springframework.beans. ...

  5. 移动 APP 网络优化概述

    一般开发一个 APP,会直接调用系统提供的网络请求接口去服务端请求数据,再针对返回的数据进行一些处理,或者使用AFNetworking/OKHttp这样的网络库,管理好请求线程和队列,再自动做一些数据 ...

  6. Scala类型检查与转换

    Scala类型检查与转换 isInstanceOf:检查某个对象是否属于某个给定的类. asInstanceOf:将引用转换为子类的引用. classOf:如果想测试p指向的是一个Employee对象 ...

  7. Linux下的ICMP反弹后门:PRISM

    搜索的时候无意中看见的这款基于ping的ICMP后门.于是到作者的github上看看,居然是阴文的,为了过级,只能强忍着看了,学生狗伤不起.还好比较简单易懂,正如简介说的一样:“PRISM is an ...

  8. 转:ObjectInputStream类和ObjectInputStream类的使用

    ObjectInputStream和ObjectInputStream类创建的对象被称为对象输入流和对象输出流. 创建文件输出流代码: FileOutputStream file_out = new ...

  9. 可移植类库无法使用async、await关键字

    今天遇到了如题所示的问题,平台已经选择了.net 4.5了,可是就是编译不通过,await关键字下出现了红色下划线. 解决方法:安装一个Bcl的补丁包. https://www.nuget.org/p ...

  10. Ubuntu/Debian下通过Apt-get简单安装Oracle JDK

    近几年本人对各种Arm小板,开发板不明原因中毒,基本以Linux系统为主,本篇文章以记录在32位Arm的Debian8上,通过Apt-get的简单命令安装Oracle JDK8并成功的记录. 1.首先 ...