循环是流程控制的又一重要结构,“白天-黑夜-白天-黑夜”属于时间上的循环,古人“年复一年、日复一日”的“日出而作、日落而息”便是每天周而复始的生活。计算机程序处理循环结构时,给定一段每次都要执行的代码块,然后分别指定循环的开始条件和结束条件,就形成了常见的循环语句。最简单的循环结构只需一个while关键字设置循环条件即可,下面是根据输入数字决定循环次数的代码例子:

		System.out.println("长夜漫漫,无心睡眠");
System.out.println("请给他设定一个睡醒的年限");
Scanner scan = new Scanner(System.in); // 从控制台接收输入文本
/* nextLine方法表示接收一行文字,以回车键结尾 */
int limit = scan.nextInt();
int year = 0;
while (year < limit) { // 当条件满足时,在循环内部持续处理
System.out.println("已经过去的年份:"+year);
year++;
}
System.out.println("他足足睡了这么多年:"+year);

运行上面的测试代码,输入数字3,连同处理结果的完整日志如下图所示。

单纯的while语句是在每次循环开始前进行条件判断,符合条件则执行一圈代码,不符合的话就退出循环。Java还提供了另一种形式的do-while循环,该循环以do关键字开头,以while语句结尾,并且条件判断也挪到了每次循环结束后再处理了。条件判断的位置变更造成了while循环和do-while循环之间的差异,while循环在开始工作前必定先判断循环条件是否成立,一旦条件不成立,那么当前循环一次都不会执行,直接跳到循环后面的语句了。而do-while循环迟至第一次循环执行完毕才做条件判断,只有发现循环条件不成立时才姗姗来迟退出循环。这意味着,即使循环条件一开始就不满足,do-while循环也一定要进入循环内部,到此一游后才拍拍屁股离开。
下面通过一个实验观察while循环和do-while循环的不同之处,演示代码见下,主要是把循环外围的while改成了do-while:

		int year = 0;
do {
System.out.println("已经过去的年份:"+year);
year++;
} while (year < limit); // 当条件满足时,在循环内部持续处理
System.out.println("他足足睡了这么多年:"+year);

然后运行演示代码,倘若输入正数,则while循环和do-while循环打印的日志信息并无区别;要是输入负数,此时二者打印的日志就分道扬镳了。while循环的日志打印结果如下面的左图所示,do-while循环的日志打印结果如下面的右图所示。

从上面的实验结果看出,在条件不满足的情况之下,do-while循环依旧强行遛达了一圈,这种先斩后奏的特性过于武断,故而限制了它的使用范围。
不管是while循环还是do-while循环,它们的条件判断都独立于循环内部代码,要么在循环开始前判断,要么在循环结束后判断。如果要求在内部代码的中间就插入条件判断,并据此决定是退出循环还是继续循环,那又该当若何?为此Java给循环语句引入了break和continue两个关键字,其中break的作用是跳出整个循环,而continue的作用是跳过本次循环的剩余代码、直接继续下次的循环判断和处理。
现在准备修改之前的代码,打算在println和year++中间进行是否继续循环的判断。首先要把while语句后面的条件式子改为true,表示每次循环前后取消默认的条件校验。然后在println和year++之间插入if语句开展条件判断,条件满足的话,就执行year++以及continue操作;条件不满足的话,就直接跳出整个循环,继续循环以外的代码处理。根据上述思路修改循环代码,改好的代码如下所示:

		int year = 0;
while (true) { // 当条件满足时,在循环内部持续处理
System.out.println("已经过去的年份:"+year);
if (year < limit) { // 这里判断能否跳出循环
year++;
continue; // 继续下一次循环
} else {
break; // 跳出循环。即跳到while循环的右花括号之后
}
}
System.out.println("他足足睡了这么多年:"+year);

接着运行修改之后的代码,打印出来的循环运行日志如下图所示:

可见此时循环内部接连打印了年份1、年份2、年份3,共三行日志。对比修改之前只打印年份1、年份2的两行日志,显然修改之后的代码逻辑更灵活,因为无论循环的内部代码如何千变万化,程序员总能找到合适的地方退出循环处理。

Java开发笔记(十八)上下求索的while循环的更多相关文章

  1. Java开发笔记(八十九)缓存字节I/O流

    文件输出流FileOutputStream跟FileWriter同样有个毛病,每次调用write方法都会直接写到磁盘,使得频繁的写操作性能极其低下.正如FileWriter搭上了缓存兄弟Buffere ...

  2. Java开发笔记(八十八)文件字节I/O流

    前面介绍了如何使用字符流读写文件,并指出字符流工具的处理局限,进而给出随机文件工具加以改进.随机文件工具除了支持访问文件内部的任意位置,更关键的一点是通过字节数组读写文件数据,采取字节方式比起字符方式 ...

  3. Java开发笔记(八十六)通过缓冲区读写文件

    前面介绍了利用文件写入器和文件读取器来读写文件,因为FileWriter与FileReader读写的数据以字符为单位,所以这种读写文件的方式被称作“字符流I/O”,其中字母I代表输入Input,字母O ...

  4. Java开发笔记(八十五)通过字符流读写文件

    前面介绍了文件的信息获取.管理操作,以及目录下的文件遍历,那么文件内部数据又是怎样读写的呢?这正是本文所要阐述的内容.File工具固然强大,但它并不能直接读写文件,而要借助于其它工具方能开展读写操作. ...

  5. Java开发笔记(八十四)文件与目录的管理

    程序除了处理内存中的数据结构,还要操作磁盘上的各类文件,这里的磁盘是个统称,泛指可以持久保留数据的存储介质,包括但不限于:插在软驱中的软盘.固定在机箱中的硬盘.插在光驱中的光盘.插在USB接口上的U盘 ...

  6. Java开发笔记(八十二)注解的基本单元——元注解

    Java的注解非但是一种标记,还是一种特殊的类型,并且拥有专门的类型定义.前面介绍的五种内置注解,都可以找到对应的类型定义代码,例如查看注解@Override的源码,发现它的代码定义是下面这样的: @ ...

  7. Java开发笔记(八十)利用反射技术操作私有方法

    前面介绍了如何利用反射技术读写私有属性,不单是私有属性,就连私有方法也能通过反射技术来调用.为了演示反射的逆天功能,首先给Chicken鸡类增加下列几个私有方法,简单起见弄来了set***/get** ...

  8. Java开发笔记(八十七)随机访问文件的读写

    前面介绍了字符流读写文件的两种方式,包括文件字符流和缓存字符流,但是它们的写操作都存在一个问题:不管是write方法还是append方法,都只能从文件开头写入,而不能追加到文件末尾或者在文件中间某个位 ...

  9. Java开发笔记(八十三)利用注解技术检查空指针

    注解属于比较高级的Java开发技术,前面介绍的内置注解专用于编译器检查代码,另外一些注解则由各大框架定义与调用,像Web开发常见的Spring框架.Mybatis框架,Android开发常见的Butt ...

  10. Java开发笔记(八十一)如何使用系统自带的注解

    之前介绍继承的时候,提到对于子类而言,父类的普通方法可以重写也可以不重写,但是父类的抽象方法是必须重写的,如果不重写,编译器就直接在子类名称那里显示红叉报错.例如,以前演示抽象类用法之时,曾经把Chi ...

随机推荐

  1. [Codeforces Round #516][Codeforces 1063B/1064D. Labyrinth]

    题目链接:1063B - Labyrinth/1064D - Labyrinth 题目大意:给定一个\(n\times m\)的图,有若干个点不能走,上下走无限制,向左和向右走的次数分别被限制为\(x ...

  2. ndk编译faac生成库

    1.编译脚本如下: NDK=/opt/android-ndk-r9d TOOLCHAIN=$NDK/toolchains/arm-linux-androideabi-4.6/prebuilt/linu ...

  3. 1.7 All components require plug-in?

    In Android, Activity, Service, ContentProvider, and BroadcastReceiver are called as four major compo ...

  4. [Swift]LeetCode73. 矩阵置零 | Set Matrix Zeroes

    Given a m x n matrix, if an element is 0, set its entire row and column to 0. Do it in-place. Exampl ...

  5. [Swift]LeetCode334. 递增的三元子序列 | Increasing Triplet Subsequence

    Given an unsorted array return whether an increasing subsequence of length 3 exists or not in the ar ...

  6. [Swift]LeetCode410. 分割数组的最大值 | Split Array Largest Sum

    Given an array which consists of non-negative integers and an integer m, you can split the array int ...

  7. [Swift]LeetCode528. 按权重随机选择 | Random Pick with Weight

    Given an array w of positive integers, where w[i] describes the weight of index i, write a function  ...

  8. [Swift]LeetCode567. 字符串的排列 | Permutation in String

    Given two strings s1 and s2, write a function to return true if s2 contains the permutation of s1. I ...

  9. [Swift]LeetCode796. 旋转字符串 | Rotate String

    We are given two strings, A and B. A shift on A consists of taking string A and moving the leftmost ...

  10. mybatis 详解------动态SQL

    mybatis 详解------动态SQL   目录 1.动态SQL:if 语句 2.动态SQL:if+where 语句 3.动态SQL:if+set 语句 4.动态SQL:choose(when,o ...