3.2.2 多路分支
前面提到条件语句的标准格式为“if (条件) { /*条件成立时的操作代码*/ } else { /*条件不成立时的操作代码*/ }”,乍看之下仿佛只有两个分支,一个是条件成立时的分支,另一个是条件不成立时的分支。很明显仅仅两个分支不能满足复杂的业务需求,自然Java代码也不会傻瓜到固步自封,为此else分支还允许嫁接别的if条件,形如“if (条件一) { } else if (条件二) { } else if (条件三) { } else { }”这样,无论来几个条件分支,都能通过“else if”加以判断和处理。
像之前的猜谜游戏,输入数字1时打印“凉风有信”的谜底,输入数字2时打印“秋月无边”的谜底,现在规定输入其他数字要打印“按键有误”的提示。此时一共存在3个条件分支,往原来的if/else语句添加一个“else if”即可实现3个分支。下面便是改写成3个分支之后的代码例子(完整代码见本章源码的src\com\control\process\Multipath.java):
System.out.println("凉风有信,秋月无边。打二字"); System.out.println("获取“凉风有信”的谜底请按1,获取“秋月无边”的谜底请按2"); Scanner scan=new Scanner(System.in); // 从控制台接收输入文本 int seq=scan.nextInt(); // nextInt方法表示接收一个整数,以回车键结尾 if (seq == 1) { // 条件式子1为真时,进入第一个if分支处理 System.out.println("凉风有信的谜底是“讽”"); } else if (seq == 2){ //否则继续判断条件式子2为真时,进入第二个if分支处理 System.out.println("秋月无边的谜底是“二”"); } else { // 否则(前面的判断条件都不满足)进入else分支处理 System.out.println("您的按键有误"); }
随着分支数量多了起来,岂不是得写一样多的“if else”?这种做法虽然可行,但毕竟大费周章,条理也不够清晰。故而早期的设计师构造了“swicth-case”语句,也被称作多路分支结构。在该控制流程中,对每个分支都指定了一个数值把关,只有设定的变量符合数值要求,才能进入相应的分支处理。这样做的好处是方便编译器在底层优化,既可利用二分查找法加快寻找速度,又可采取地址映射直接找到指定分支。兼容并蓄的Java自然继承了有利于调优的“swicth-case”,多路分支语句除了swicth和case外,还额外增加了break和default两个关键字。break的作用是跳出整个多路分支,不再执行本分支及其余分支的代码;default的作用相当于else,所有不满足已知条件的数值都进入default这个默认分支处理。
仍以猜谜游戏为例,使用“swicth-case”改写后的多路分支代码如下(完整代码见本章源码的src\com\control\process\Multipath2.java):
// switch允许判断某个变量的多个取值,并分别进行单独处理 switch (seq) { case 1: // seq值为1时进入该分支 System.out.println("凉风有信的谜底是“讽”"); break; // 跳出多路分支,即跳到switch分支的右花括号之后 case 2: // seq值为2时进入该分支 System.out.println("秋月无边的谜底是“二”"); break; // 跳出多路分支,即跳到switch分支的右花括号之后 default: // seq值为其他时进入该分支 System.out.println("您的按键有误"); break; // 跳出多路分支,即跳到switch分支的右花括号之后 } System.out.println("猜谜结束");
多路分支固然好用,但要特别注意以下几点:
(1)每个case分支末尾务必要加上break语句,否则即使该分支走完了也不会跳出多路分支,而是继续执行该分支的后面一个分支的代码,显然这并非程序员的本意。
(2)多路分支只能判断整型(含byte、short、int)、字符型、枚举型3种类型的变量,无法判断布尔、浮点、双精度等其他类型的变量。
(3)case语句后面的数值只能做相等判断,不能开展大于、小于等其他关系运算。