java三目运算符以及语言上的思考
- 三目运算符,也叫三元运算符,英文名叫做tertiary operator,表示的是通过 ?和 :来为变量赋值。语法上不做过多介绍,不过建议,尽量不要出现一个赋值语句中出现多重三元运算符,可读性贼差。
简单表示以下一个三元运算符和它等价的if语句赋值,if语句的赋值方式:
int king;
int temp =0;
if (temp ==0)
king =1;
else
king =0;
三元运算符的赋值方式:
int king2 = temp == 0 ? 1 : 0;
这两段语句是等价的,这是用Java写的。为什么要强调是Java写的呢,每种语言都可以这样写的,不是吗?这个时候,讲一下,编程语言这样的东西吧,我对编程语言的理解。
-
计算机语言,分为好几层,下图是《计算机组成原理》教材中的分层结构
1555649792(1).jpg
C语言就属于高级语言机器。我们使用的Java语言,我个人感觉再建立了一层,Java利用高级语言构建一层JVM,Java程序写完之后,会编译成.class文件,.class文件中的字节码虽然类似汇编,但是.class文件的字节码是提供给JVM看的,真正的运行,需要转换成计算机所识别的内容,这一层工作是由JVM来做的。
再说这个三元运算符,我们现在要看三元运算符赋值和if语句赋值,哪个更快,效率更好,应该怎么做,当然是benchmark啦,但是大家不用了解细节都知道,这两种方式,效率肯定都差不多的。先拿出这一段代码编译过后的字节码,再来谈论一下和语言层面相关的问题。
22: iconst_0
23: istore_1
24: iconst_0
25: istore_2
26: iload_2
27: ifne 35
30: iconst_1
31: istore_1
32: goto 37
35: iconst_0
36: istore_1
37: iload_2
38: ifne 45
41: iconst_1
42: goto 46
45: iconst_0
46: istore_3
解释一下命令,iconst,istore,iload,都是属于本地变量(或常量池),操作数栈上互相赋值的操作,ifne 就是if(temp==0),符合就进入否则就跳转到goto。可以看到,用if语句的话,声明变量什么的其实也有一定消耗的,三元运算符这些消耗要少一些,不过这也是我这种赋值这种方式导致的。
总结一下,不论 if 方式来赋值,还是三元运算符赋值,它们的效率问题只有变量值拷贝的问题。但是总体来说,三元运算符,和if语句效率是差不多的,几乎不会有多少效率差距。
上面这个结论,是基于jvm对不同语法的定义,所作出的结论。请注意,并没有谈及,jvm实现这种语法的消耗,当然,我个人觉得不太重要,所以我也没有benchmark去验证哪种效率更好。
我更多的想说我对语言的理解,高级计算机语言的分类,简单分可以说就是:解释型语言和编译型语言,我个人觉得更通俗一点可以说脚本语言和非脚本语言。
从上面那张分层的图能看到,不论你是什么高级语言,都是会走那么多层,才能让计算机识别到你想要传达的意思。非脚本语言,或者编译型语言,需要将你所使用的语言,编译成计算机所能识别的方式(例:字节码),计算机再读编译后的方式,实际上并不是真正的在跑你所写的代码,而是在跑你所写的被编译过的代码。
假如你在Linux上跑一个Java代码,你首先需要把Java代码编译成.class文件,再打包成可执行jar文件,才能执行,但是你执行一个js文件,或者python就简单得多,机器上安装了node环境,或者python环境,可以直接执行你所写的代码,可以不需要编译打包这些环节(主要是编译)。这就是脚本语言。
脚本语言,我所接触的脚本语言不多,但是我所理解的脚本语言,是需要一种叫做解释器的东西。你实际上跑的代码,是把代码丢在了解释器里面跑,在解释器里面,他们把你的代码转换成计算机所识别的方式在跑,你不用知道是怎么回事,只需要把你的代码写好就行。所以脚本语言的语法会更丰富一点,而且解释器所做的工作实在太多,脚本语言中,你对语法的选择有时应该大于对算法和数据结构的选择。脚本语言虽然很方便,但是脚本语言对语法的依赖太大。当然,Java现在对语法的依赖也很大,假如Java并发流处理方式到后面优化到比你自己写的多线程处理数据更快了,到时候不论是代码的简洁性还是效率上,你肯定还是会选择利用jdk提供的api。这没有什么弊处,只是面对数据结构和算法的时候,需要你去实现会更少,需要你会选择对你写代码更重要。当然,会实现各种数据结构和算法肯定更好。