24

私はJava チュートリアルに従って初心者の Java プログラマーです。

Java チュートリアルData Streams Pageから単純な Java プログラムを使用していますが、実行時に表示され続けますEOFException。リーダーは最終的にファイルの最後に到達する必要があるため、これが正常かどうか疑問に思っていました。

import java.io.*;

public class DataStreams {
    static final String dataFile = "F://Java//DataStreams//invoicedata.txt";

    static final double[] prices = { 19.99, 9.99, 15.99, 3.99, 4.99 };
    static final int[] units = { 12, 8, 13, 29, 50 };
    static final String[] descs = {
        "Java T-shirt",
        "Java Mug",
        "Duke Juggling Dolls",
        "Java Pin",
        "Java Key Chain"
    };
    public static void main(String args[]) {
        try {
            DataOutputStream out = new DataOutputStream(new BufferedOutputStream(new FileOutputStream(dataFile)));

            for (int i = 0; i < prices.length; i ++) {
                out.writeDouble(prices[i]);
                out.writeInt(units[i]);
                out.writeUTF(descs[i]);
            }

            out.close(); 

        } catch(IOException e){
            e.printStackTrace(); // used to be System.err.println();
        }

        double price;
        int unit;
        String desc;
        double total = 0.0;

        try {
            DataInputStream in = new DataInputStream(new BufferedInputStream(new FileInputStream(dataFile)));

            while (true) {
                price = in.readDouble();
                unit = in.readInt();
                desc = in.readUTF();
                System.out.format("You ordered %d" + " units of %s at $%.2f%n",
                unit, desc, price);
                total += unit * price;
            }
        } catch(IOException e) {
            e.printStackTrace(); 
        }

        System.out.format("Your total is %f.%n" , total);
    }
}

正常にコンパイルされますが、出力は次のとおりです。

You ordered 12 units of Java T-shirt at $19.99
You ordered 8 units of Java Mug at $9.99
You ordered 13 units of Duke Juggling Dolls at $15.99
You ordered 29 units of Java Pin at $3.99
You ordered 50 units of Java Key Chain at $4.99
java.io.EOFException
        at java.io.DataInputStream.readFully(Unknown Source)
        at java.io.DataInputStream.readLong(Unknown Source)
        at java.io.DataInputStream.readDouble(Unknown Source)
        at DataStreams.main(DataStreams.java:39)
Your total is 892.880000.

Java チュートリアルData Streams Pageには、次のように書かれています。

DataStreams は、無効な戻り値をテストする代わりに、EOFExceptionをキャッチすることによってファイルの終わりの状態を検出することに注意してください。DataInput メソッドのすべての実装は、戻り値の代わりに EOFException を使用します。

では、これは、キャッチするEOFExceptionのが正常なので、キャッチして処理しなくても問題ない、つまりファイルの終わりに到達したということですか?

それを処理する必要があるということであれば、どうすればよいかアドバイスをお願いします。

編集

in.available() > 0提案から、 forwhileループ条件を使用して修正しました。

または、それで問題ないので、例外を処理するために何もできませんでした。

4

8 に答える 8

24

ファイルからの読み取り中、ループを終了していません。そのため、すべての値が読み取られ、次の行の読み取りの次の反復で EOFException が正しくスローされます。

 price = in.readDouble();

ドキュメントを読むと、次のように書かれています。

スロー:

EOFException - この入力ストリームが 8 バイトを読み取る前に最後に到達した場合。

IOException - ストリームが閉じられており、含まれている入力ストリームが閉じた後の読み取りをサポートしていないか、別の I/O エラーが発生した。

while ループに適切な終了条件を入れて、問題を解決します。たとえば、以下のとおりです。

     while(in.available() > 0)  <--- if there are still bytes to read
于 2013-08-26T19:13:29.320 に答える
7

これを処理する最善の方法は、適切な条件で無限ループを終了することです。

しかし、例外処理を要求したので:

2つのキャッチを使用してみてください。あなたの EOFException は予期されているので、発生しても問題はないようです。その他の例外は処理する必要があります。

...
} catch (EOFException e) {
   // ... this is fine
} catch(IOException e) {
    // handle exception which is not expected
    e.printStackTrace(); 
}
于 2013-08-26T19:17:54.573 に答える
2

while(in.available() != 0)の代わりに使用できますwhile(true)

于 2013-08-26T19:14:40.680 に答える
2

または、次を使用して、最初に (ヘッダーとして) 要素の数を書き出すことができます。

out.writeInt(prices.length);

ファイルを読み取るときは、最初にヘッダー (要素数) を読み取ります。

int elementCount = in.readInt();

for (int i = 0; i < elementCount; i++) {
     // read elements
}
于 2013-08-26T19:24:04.313 に答える
0

継承されているため、IOExceptionwhich もcatchをキャッチします。チュートリアルEOFExceptionの例を見ると、キャッチする必要があることに下線が引かれています。これが彼らが行うことです。問題を解決するには、前にキャッチします:EOFExceptionEOFExceptionIOException

try
{
    //...
}
catch(EOFException e) {
    //eof - no error in this case
}
catch(IOException e) {
    //something went wrong
    e.printStackTrace(); 
}

それに加えて、例外を使用したデータ フロー制御は好きではありません。これは意図した例外の使用法ではないため、(私の意見では) 非常に悪いスタイルです。

于 2013-08-26T19:22:05.697 に答える
0

EOFException は IOException の子であり、以下のように好む ==>

try {
        .
        .
        .
    } catch (IOException e) {
        if (!(e instanceof EOFException)) {
            throw new RuntimeException(e);
        }
    }
于 2021-06-13T15:40:27.747 に答える