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++;
        }
    }
}

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。