while循环使用hasNext();没有完全循环

时间:2015-01-08 16:21:57

标签: java for-loop while-loop eof

我正在创建一个从外部文件读取数据的程序,将其与文件中的其他数据进行比较,然后将结果打印到新的外部文件。我的代码的while循环部分有问题。我不确定它是while循环本身还是嵌套在其中的for循环。这是代码:

public class WageCalculator {

  public static void main(String[] args) throws FileNotFoundException {

    Scanner input = new Scanner(new FileReader("TestData.txt")); //Scanner for external file
    PrintWriter output = new PrintWriter("wagedaily.txt");

    float RecommendedMaximum;
    RecommendedMaximum = Float.parseFloat(JOptionPane.showInputDialog(null, "Enter the recommended maximum journey cost:"));

    String ShipID, JourneyID; //Variables
    int JourneyLength, Crew;
    double RateOfPay, CrewCost, TotalCost;

    while (input.hasNext()) { //EOF-Controlled While Loop
      ShipID = input.nextLine();
      JourneyID = input.nextLine();
      JourneyLength = input.nextInt();
      Crew = input.nextInt();

      CrewCost = 0; //Default Values Set
      TotalCost = 0;
      for (int x = Crew; x > 0; x--) { //For Loop updates the above values
        RateOfPay = input.nextDouble();
        CrewCost = RateOfPay * JourneyLength;
        TotalCost = TotalCost + CrewCost;
      }

      if (TotalCost < RecommendedMaximum) { //if-else statements to compare values
        System.out.println("The total cost of...");
        output.println("The total cost of...");
      } else if (TotalCost == RecommendedMaximum) {
        System.out.println("The total cost of...");
        output.println("The total cost of...");
      } else {
        System.out.println("The total cost of...");
      }
    }

    output.close(); //Close both Scanner and Printwriter
    input.close();
  }

}

我得到的错误是:

Exception in thread "main" java.util.InputMismatchException  
at java.util.Scanner.throwFor(Unknown Source)  
at java.util.Scanner.next(Unknown Source)  
at java.util.Scanner.nextInt(Unknown Source)  
at java.util.Scanner.nextInt(Unknown Source)  
at (package).WageCalculator.main(WageCalculator.java:30)

错误说我的代码中的第30行是问题,但我不太确定。

任何人都需要查看TestData.txt文件:

Monarch  //ShipID
M141     //JourneyID
16       //JourneyLength
6        //Crew
10.5     //RateOfPay -
10.5
20
20
20
30       //- RateOfPay
Princess //ShipID
P103     //JourneyID
18       //JourneyLength
5        //Crew
40       //RateOfPay -
45
45
60
80       //- RateOfPay

任何帮助将不胜感激:)

2 个答案:

答案 0 :(得分:4)

在循环顶部检查input.nextXXX()一次后,您在循环中进行了大量input.hasNext()次调用。不要这样做,因为这非常不安全并且必然会失败,因为 check get之间始终存在一对一的对应关系。例如,如果您想获得下一行,则应在调用input.HasNextLine()之前调用一个input.nextLine()input.next()input.nextInt()也是如此。

我自己,我会逐行检查hasNextLine()然后一次读取nextLine(),然后操纵收到的字符串。

while (input.hasNextLine()) {
   String line = input.nextLine();

   // now do not use input further within the loop
}

然后你可以在循环中使用收到的行使用第二个扫描仪,或者通过String#split(String regex)拆分行或者做你需要做的任何事情。

或者您可以使用字符串replaceAll(...)和正则表达式来删除所有空格,后跟“//”后跟任何字符。如,

  while (input.hasNextLine()) {
     String line = input.nextLine();

     // get rid of white space followed by "//" followed by anything
     line = line.replaceAll("\\s+//.*", "");
     System.out.println(line);
  }

修改
我对你的问题和你的数据有了更多了解,我将修改我的答案。 如果您完全确定数据文件的完整性,则可以考虑每次获取整艘船的数据时检查下一行。例如:

public static void main(String[] args) {
  InputStream inStream = GetData.class.getResourceAsStream(DATA_FILE);
  List<Cruise> cruiseList = new ArrayList<>();
  Scanner input = new Scanner(inStream);
  while (input.hasNext()) {
     String name = getLine(input);
     String id = getLine(input);
     int length = Integer.parseInt(getLine(input));
     int crew  = Integer.parseInt(getLine(input));

     // assuming a class called Cruise
     Cruise cruise = new Cruise(name, id, length, crew);

     for (int i = 0; i < crew; i++) {
        cruise.addPayRate(Double.parseDouble(getLine(input)));
     }

     cruiseList.add(cruise);
  }

  input.close();
}

private static String getLine(Scanner input) {
  String line = input.nextLine();

  // get rid of white space followed by "//" followed by anything
  line = line.replaceAll("\\s+//.*", "");
  return line;
}

答案 1 :(得分:2)

它给你带来麻烦的原因是因为当用户输入一个整数然后点击回车时,刚刚输入了两件事 - 整数和&#34;换行符&#34;这是\ n。您正在调用的方法&#34; nextInt&#34;只读取整数,从而在输入流中保留换行符。但是调用nextLine()会读取换行符,这就是为什么在代码工作之前必须调用nextLine()的原因。您也可以调用next(),它也可以在换行符中读取。

另请阅读Scanner类的文档以进一步了解:

以下是更正码:

import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.PrintWriter;
import java.util.Scanner;

import javax.swing.JOptionPane;

public class WageCalculator {

      public static void main(String[] args) throws FileNotFoundException {

        Scanner input = new Scanner(new FileReader("TestData.txt")); //Scanner for external file
        PrintWriter output = new PrintWriter("wagedaily.txt");

        float RecommendedMaximum;
        RecommendedMaximum = Float.parseFloat(JOptionPane.showInputDialog(null, 
                                                "Enter the recommended maximum journey cost:"));

        String ShipID, JourneyID; //Variables
        int JourneyLength, Crew;
        double RateOfPay, CrewCost, TotalCost;

        while (input.hasNext()) { //EOF-Controlled While Loop
            System.out.println("While Enter"); // For debugging purpose
            ShipID = input.nextLine();
            JourneyID = input.nextLine();
            JourneyLength = input.nextInt();
            input.nextLine();     // Enter this to read the data of skipped Line
            Crew = input.nextInt();
            input.nextLine();
            CrewCost = 0; //Default Values Set
            TotalCost = 0;
            for (int x = Crew; x > 0; x--) { //For Loop updates the above values
                System.out.println("While Under if Enter");// For debugging purpose
                RateOfPay = input.nextDouble();
                input.nextLine();
                CrewCost = RateOfPay * JourneyLength;
                TotalCost = TotalCost + CrewCost;
                System.out.println("While Under if Exit");// For debugging purpose
            }

            if (TotalCost < RecommendedMaximum) { //if-else statements to compare values
                output.println("The total cost of...");
            } else if (TotalCost == RecommendedMaximum) {
                System.out.println("The total cost of...");
                output.println("The total cost of...");
            } else {
                System.out.println("The total cost of...");
            }
            System.out.println("While Exit"); // For debugging purpose
        }    
        output.close(); //Close both Scanner and Printwriter
        input.close();
    }

}