05-循环

05-循环

  • 5.1 引言
    • 循环可以让一个程序反复的执行语句。
    • Java提供了三中类型的循环语句:while语句、do-while循环和for循环。
  • 5.2 while循环
    • while循环在条件为真的情况下,重复的执行语句。
    • while循环的语法如下:
    ```java
    while(循环继续语句){
    //循环体
    语句(组);
    }
    ```
- 循环中包含的重复执行的语句部分称为**循环体**。循环体的每一次执行都被认为是一次循环的迭代。每个循环都含有**循环继续条件**,循环继续条件是一个布尔表达式,控制循环体的执行。在循环体执行前总是先计算循环条件以决定是否执行它。若条件是true,则执行循环体;若条件是flase,则终止整个循环,并且程序控制转移到while循环后的下一条语句。
![在这里插入图片描述](https://upload-images.jianshu.io/upload_images/24494503-28694886c305b893?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

- **循环继续条件应该总是放在圆括号内。只有当循环体只包含一条语句或不包含语句时,循环体的花括号才可以省略**。
- 要保证循环继续条件最终可以变为false,以便程序能够结束。一个常见的错误是无限循环(也就是说,循环会永远执行下去)。如果你的程序运行了过长的时间而不结束,可能其中就有无限循环。如果你是从命令窗口运行程序的,按CTRL+C来结束运行。
- 程序员经常会犯的错误就是使循环多执行一次或者少执行一次。这种情况通常称为差一错误。
- 示例:提示用户为两个个位数相加的问题给出答案。现在你可以使用循环编写程序,让用户重复输入新的答案,直到答案正确为止。


    ```java
    package chapter05;
    
    import java.util.Scanner;
    
    public class RepeatAddtionQuiz {
        public static void main(String[] args){
            int number1 = (int)(Math.random() * 10);
            int number2 = (int)(Math.random() *10 );
            Scanner input = new Scanner(System.in);
            System.out.print("What is " + number1 + " + " + number2 + "? ");
            int answer = input.nextInt();
            while (number1 + number2 != answer){
                System.out.print("Wrong answer.Try again.What is " + number1 + " + " + number2 + "? ");
                answer = input.nextInt();
            }
            System.out.println("You got it! ");
        }
    }
    
    ```
  • 5.3 示例学习:猜数字


    在这里插入图片描述
```java
package chapter05;

import java.util.Scanner;

public class GuessNumber {
    public static void main(String[] args){
        int number = (int)(Math.random() * 101);
        Scanner input = new Scanner(System.in);
        System.out.println("Guess a magic number between 0 and 100");
        int guess = -1;
        while (guess != number){
            System.out.print("\nEnter your guess: ");
            guess = input.nextInt();

            if (guess == number)
                System.out.println("Yes, the number is " + number);
            else if (guess > number)
                System.out.println("Your guess is too high");
            else
                System.out.println("Your guess is too low");
        }
    }
}

```
  • 5.4 循环设计策略
    • 编写循环时应该考虑三个步骤:
      • 1、确定需要重复的语句
      • 2、将这些语句放在一个循环中
      • 3、为循环继续条件编码,并为控制循环添加合适的语句
    • 给出以下一个程序:可以产生5个问题,在学生回答完所有5个问题后,报告回答正确的题数。这个程序还显示该测试所花费的时间,并列出所有的题目。
    ```java
    package chapter05;
    
    import java.util.Scanner;
    
    public class SubtractionQuizLoop {
        public static void main(String[] args){
            final int NUMBER_OF_QUESTION = 5;
            int correctCount = 0;
            int count = 0;
            long stratTime = System.currentTimeMillis();
            String output = " ";
            Scanner input = new Scanner(System.in);
    
            while (count < NUMBER_OF_QUESTION){
                int number1 = (int)(Math.random() * 10);
                int number2 = (int)(Math.random() * 10);
                if (number1 < number2){
                    int temp = number1;
                    number1 = number2;
                    number2 = temp;
                }
                System.out.print("What is " + number1 + " - " + number2 + "? ");
                int answer = input.nextInt();
    
                if (number1 - number2 == answer){
                    System.out.println("You are correct!");
                    correctCount++;
                }
                else
                    System.out.println("Your answer is wrong.\n" + number1 + " - " + number2 + "should be " + (number1 + number2));
                count++;
                output += "\n" + number1 + "-" + number2 + "=" + answer + ((number1 - number2 == answer) ? "correct":"wrong");
            }
            long endTime = System.currentTimeMillis();
            long testTime = endTime - stratTime;
            System.out.println("Correct count is " + correctCount + "\nTest time is " + testTime / 100 + " seconds\n" + output);
        }
    }
    
    ```
  • 5.5 使用用户确认或者标记值控制循环
    • 一种控制循环的常用技术是在读取和处理一组值时指定一个特殊值。这个特殊的输入值也称为标记值,用以表明循环的结束。如果一个循环使用标记值来控制它的执行,它就称为标记位控制的循环。
    • 给出一个程序:用来读取和计算个数不确定的整数之和,并以输入0表示输入结束。
    ```java
    package chapter05;
    
    import java.util.Scanner;
    
    public class SentinelValue {
        public static void main(String[] args){
            Scanner input = new Scanner(System.in);
            System.out.print("Enter an integer (the input ends if it is 0): ");
            int data = input.nextInt();
            int sum = 0;
            while (data != 0){
                sum += data;
                System.out.print("Enter an integer (the input ends if it is 0): ");
                data = input.nextInt();
            }
            System.out.println("The sum is " + sum);
        }
    }
    
    ```
- 不要比较浮点数值是否相等来进行循环控制。因为浮点值都是近似值,使用他们可能导致不精确的循环次数和不精确的结果。
- 如果要输入大量的数据值,那么从键盘上输入是非常烦琐的。可以将这些数据用空格隔开,保存在一个名为input.txt的文本文件中,然后使用下面的命令运行这个程序:


    ```java
    java SentineValue < input.txt
    ```
- 这个命令称为输入重定向。程序从文件input.txt中读取输入,而不是让用户在运行时从键盘输入数据。
- 类似的,还有输出重定向,输出重定向将输出发送给文件,而不是将他们显示在控制台上。输出重定向的命令为:

    
    ```java
    java ClassName > output.txt
    ```
- 可以在同一个命令中同时使用输入重定向和输出重定向。


    ```java
    java SentineValue < input.txt > output.txt
    ```
  • 5.6 do-while循环
    • do-whil循环和while循环基本一样,不同的是他先执行循环体一次,然后判断循环继续条件。
    • do-while循环是while循环的变体。他的语法是:
    ```java
    do{
        //循环体;
        语句(组);
    }while(循环继续条件);
    ```
[图片上传失败...(image-4d81b4-1602057381370)]

- 首先执行循环体,然后计算循环继续条件。如果计算结果为true,则重复执行循环体;如果是false,则终止do-while循环。
- while循环与do-while循环的不同之处在于:计算循环继续条件和执行循环体的先后顺序不同。在使用do-while循环的情况下,循环体至少执行一次。可以使用while循环或者do-while循环来编写循环。某些情况下选择其中一个会比另一种更加方便。
    ```java
    package chapter05;
    
    import java.util.Scanner;
    
    public class TestDoWhile {
        public static void main(String[] args){
            int data;
            int sum = 0;
            Scanner input = new Scanner(System.in);
            do {
                System.out.print("Enter an integer (the input ends if it is 0): ");
                data = input.nextInt();
                sum += data;
            }while (data != 0);
            System.out.println("The sum is : " + sum);
        } 
    }
    
    ```
- 如果循环中的语句至少要执行一次,建议使用do-while循环。如果使用while循环,那么这些语句必须在循环前和循环内都出现。
  • 5.7 for循环
    • for语句循环的语法如下:

      for(初始操作;循环继续条件;每次迭代后的操作){
      //循环体;
      语句(组);
      }
      
      在这里插入图片描述
    • for循环语句从关键字for开始,然后是用一对圆括号住的循环控制结构体。这个结构体包括初始操作、循环继续条件和每次迭代后的操作。控制结构体后紧跟着花括号括起来的循环体。初始操作、循环继续条件和每次迭代后的操作都要用分号隔开。

    • 一般情况下,for循环使用一个变量来控制循环体的执行次数,以及什么时候循环终止。这个变量称为控制变量。初始操作是指初始控制变量,每次迭代后的操作通常会对控制变量做自增或自减,而循环继续条件检验控制变量是否达到终止值。

    • 控制变量必须在循环控制结构体内或循环前说明。如果循环控制变量只在循环内使用而不再其他地方使用,那么在for循环的初始操作中声明它是一个良好的编程习惯。如果在循环控制结构体内声明变量,那么在循环外不能引用他。

    • for循环中的初始操作可以是0个或者多个以逗号隔开的变量声明语句或赋值表达式。

    • 如果省略for循环中的循环继续条件,则隐含的认为循环继续条件为true。

  • 5.8 采用哪种循环
    • for循环更加简洁,并且相比另外两种循环而言更容易避免错误。
    • while循环和for循环都称为前测循环,因为继续条件是在循环体执行之前检测的,do-while循环称之为后测循环,因为循环条件是在循环体执行之后检测的。
    • 如果已经提前知道重复次数,那就采用for循环。无法确定重复次数,就采用while循环,在检测继续条件前需要执行循环体,就用do-while循环替代while循环。
  • 5.9 嵌套循环
    • for循环打印一个乘法表的程序:
    ```java
    package chapter05;
    
    public class MultiplicationTable {
        public static void main(String[] args){
            System.out.println("            Multiplication Table");
            System.out.print("    ");
            for (int j = 1;j <= 9;j++)
                System.out.print("   " + j);
            System.out.println("\n---------------------------------------------");
            for (int i = 1;i <= 9;i++){
                System.out.print(i + " | ");
                for (int j = 1;j <= 9;j++){
                    System.out.printf("%4d",i * j);
                }
                System.out.println();
            }
        }
    }
    
    ```
  • 5.10 最小化数值错误
```java
package chapter05;

public class TestSum {
    public static void main(String[] args){
        float sum = 0;
        for (float i = 0.01f;i <= 1.0f;i = i + 0.01f)
            sum += i;
        System.out.println("The sum is " + sum);
    }
}

```
  • 5.11 示例学习
    • 5.11.1 求最大公约数
      • 提示用户输入两个正整数,然后找到他们的最大公约数。
        ```java
        package chapter05;
        
        import java.util.Scanner;
        
        public class GreatestCommonDivisor {
            public static void main(String[] args){
                Scanner input = new Scanner(System.in);
                System.out.print("Enter first integer: ");
                int n1 = input.nextInt();
                System.out.print("Enter second integer: ");
                int n2 = input.nextInt();
                int gcd = 1;
                int k = 2;
                while (k <= n1 && n2 % k == 0){
                    if (n1 % k == 0 && n2 % k == 0)
                        gcd = k;
                    k++;
                }
                System.out.println("The greatest common divisor for " + n1 + " and " + n2 + " is " + gcd);
            }
        }
        
        ```
- 5.11.2 预测未来学费
    - 假设某个大学今年的学费是10000美元,而且以每年7%的速度增加。多少年之后学费会翻倍?
    

        ```java
        package chapter05;
        
        public class FutureTuition {
            public static void main(String[] args){
                double tuition = 10000;
                int year = 0;
                while (tuition < 20000){
                    tuition = tuition * 1.07;
                    year++;
                }
                System.out.println("Tuition will be doubled in " + year + " years");
                System.out.printf("Tuition will be $%.2f in %1d years",tuition,year);
            }
        }
        
        ```
- 5.11.3 将十进制数转换为十六进制数
    - 意识用户输入一个十进制数,然后将它转换为一个字符串形式的十六进制数。
        ```java
        package chapter05;
        
        import java.util.Scanner;
        
        public class Dec2Hex {
            public static void main(String[] args){
                Scanner input = new Scanner(System.in);
                System.out.print("Enter a decimal number: ");
                int decimal = input.nextInt();
                String hex = "";
                while (decimal != 0){
                    int hexValue =  decimal % 16;
                    char hexDigit = (0 <= hexValue && hexValue <= 9) ? (char)(hexValue + '0') : (char)(hexValue - 10 + 'A');
                    hex = hexDigit + hex;
                    decimal = decimal / 16;
                }
                System.out.println("The hex number is " + hex);
            }
        }
        
        ```
  • 5.12 关键字break和continue
    • 可以在一个循环中使用break立即终止循环。
      package chapter05;
      
      public class TestBreak {
          public static void main(String[] args){
              int sum = 0;
              int number = 0;
              while (number < 20){
                  number++;
                  sum += number;
                  if (sum >= 100)
                      break;
              }
              System.out.println("The number is " + number);
              System.out.println("The sum is " + sum);
          }
      }
      
      
    • 可以在循环中使用关键字continue。当程序遇到continue时,他会结束当前的迭代。程序控制转向该循环体的末尾。换句话说,continue只是跳出了一次迭代,而关键字break是跳出了整个循环。
      package chapter05;
      
      public class TestContinue {
          public static void main(String[] args){
              int sum = 0;
              int number = 0;
              while (number < 20){
                  number++;
                  if (number == 10 || number == 11)
                      continue;
                  sum += number;
              }
              System.out.println("The sum is " + sum);
          }
      }
      
      
    • continue语句总是在一个循环内。在while和do-while循环中,continue语句之后会马上计算循环继续条件;而在for循环中,continue语句之后会立即先执行每次迭代之后的动作,在计算循环循环条件。
    • 很多程序设计语言都有goto语句。goto语句可以随意的将控制转移到程序中的任意一条语句中,然后执行它。这使程序很容易出错。Java中的break语句和continue语句是不同于goto语句的。他们只能运行在循环中或者switch语句中。break语句跳出整个循环,而continue语句跳出循环的当前迭代。
  • 5.13 示例学习:判断回文
package chapter05;

import java.util.Scanner;

public class Palindrome {
    public static void main(String[] args){
        Scanner input = new Scanner(System.in);
        System.out.print("Enter a string: ");
        String s = input.nextLine();
        int low = 0;
        int high = s.length() - 1;
        boolean isPalindrome = true;
        while (low < high){
            if (s.charAt(low) != s.charAt(high)){
                isPalindrome = false;
                break;
            }
            low++;
            high--;
        }
        if (isPalindrome)
            System.out.println(s + " is a palindrome");
        else 
            System.out.println(s + " is not a palindrome");
    }
}

  • 5.14 示例学习:显示素数
package chapter05;

public class PrimeNumber {
    public static void main(String[] args){
        final int NUMBER_OF_PRIME = 50;
        final int NUMBER_OF_PRIME_PER_LINE = 10;
        int count = 0;
        int number = 2;
        System.out.println("The first 50 prime numbers are \n");
        while (count < NUMBER_OF_PRIME){
            boolean isPrime = true;
            for (int divisor =2; divisor <= number; divisor++){
                if (number % divisor == 0){
                    isPrime = false;
                    break;
                }
            }
            if (isPrime){
                count++;
                if (count % NUMBER_OF_PRIME_PER_LINE == 0){
                    System.out.println(number);
                }
                else 
                    System.out.println(number + " ");
            }
            number++;
        }
    }
}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 221,635评论 6 515
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,543评论 3 399
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 168,083评论 0 360
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,640评论 1 296
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,640评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,262评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,833评论 3 421
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,736评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,280评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,369评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,503评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,185评论 5 350
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,870评论 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,340评论 0 24
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,460评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,909评论 3 376
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,512评论 2 359