2

Java I/O データの宿題をやっています。問題は、オブジェクトのシリアル化を使用してバイナリ ファイルからデータをロードすることが許可されていないことです。

割り当て要件は次のとおりです。

永続性、ファイルへのオブジェクトの書き込み、ファイルからのオブジェクトの読み取り (テキスト形式)

• All objects should be written to a single file in a text format
• All objects should be read form the same file
• You should use JFileChooser

UnitAssessment、およびの3 つのクラスがありTaskます。

& は具体的なサブクラスですが、Assessmentクラスは抽象的です。IndividualAssessmentGroupAssessment

UnitのコレクションをAssessment持ち、 のコレクションをAssessment持っていTaskます。

すべてのデータを 1 つのテキスト ファイルに保存できますが、テキスト ファイルの各行を適切なクラスFileWriterに読み込む方法がわかりません。Assessment

私が言いたいのは、どの行がIndividualAssessmentまたはGroupAssessmentクラス用であるかをどのように認識するかということです。

これが私が試したコードですが、うまくいきません:

BufferedReader bf = new BufferedReader(file);
While (bf.readLine != null){
    Unit u = new Unit(bf);
    diary.add(u);
    try{
        Assessment a = new IndividualAssessment(bf);
    } catch (IOException ex){
        Assessment a = new GroupAssessment(bf);
            }
    u.add(a);
    Task t = new Task(bf);
    a.add(t);
4

3 に答える 3

1

おそらく、読み取った行の内容を String オブジェクトに格納することから始める必要があります。したがって、次のようになります。

String line = bf.readLine();
while(line != null) {
    // Add code to look at your line to figure out what kind of object it 
    // represents.   For example you could add a one character prefix to each 
    // line when you write it to specify which object it is ('U', 'A' or 'T').
    // Based on that, you can call a constructor of the appropriate object that
    // takes a String as input. Then let the constructor deal with parsing the
    // line for the object it represents. This way you don't end up with some
    // massive parsing routine.

    char code = line.charAt(0);
    if(code == 'T') {
        Task task = new Task();
        task.initFromString(line.sustring(1));
        ... Do something with your task
    }
    else if(code == ...) {
    }

    line = bf.readLine();    // Read the next line
}

すべてのオブジェクトが実装するインターフェイスを定義します。

public interface TextExportable {
    public char getClassIdentifier();
    public void initFromString(String s);
}

あなたのオブジェクトがどのように見えるかはわかりませんが、例えば:

public class Task implements TextExportable {
    private String name;

    public Task() {} // For the pseudo-serialization

    public Task(String name) { // For creating the object by hand
        this.name = name;
    }

    public char getClassIdentifier() {
        return 'T';
    }

    public String toString() {
        return getClassIdentifier()+name;
    }

    public void initFromString(String line) {
        this.name = line;
        // Here, you would need extra parsing if you have more than one attribute
        // and dissect the line
    }
}
于 2012-05-25T16:53:36.067 に答える
0

受け入れられた答えに加えて、解析を行うためのクリーンな方法は、インタープリターパターンに従うことであることを付け加えたいと思います

したがって、次のような bnf を使用した構文を想定します。

<Unit>        ::= <Name>[<Assessments>]
<Assessments> ::= <Assessment> | <Assessment> <Assessments>
<Assessment>  ::= <Name>:<Type>[<Tasks>]
<Tasks>       ::= <Task> | <Task>,<Tasks>
<Name>        ::= <String>
<Type>        ::= Group | Ind

Expressionあなたはそれを解析するためにこのようなものを書くかもしれません - 私はそれを使用するクラスを書く練習として残します。

try {
    BufferedReader br = new BufferedReader(new FileReader("File.txt"));
    try {
        String unit;
        while ((unit = br.readLine()) != null) {
            String unitName = unit.substring(0, unit.indexOf('['));
            String assessments = unit.substring(unit.indexOf('[') + 1,
                            unit.lastIndexOf(']'));
            List<AssessmentExpression> aes = new ArrayList<AssessmentExpression>();
            while (!assessments.equals("")) {
                String assessmentName = assessments.substring(0,
                                assessments.indexOf(':'));
                String assessmentType = assessments.substring(
                                assessments.indexOf(':') + 1,
                                assessments.indexOf('['));
                String tasks = assessments.substring(
                                assessments.indexOf('[') + 1,
                                assessments.indexOf(']'));
                List<String> taskList = Arrays.asList(tasks.split(","));
                aes.add(new AssessmentExpression(assessmentName,
                                assessmentType, taskList));
                assessments = assessments.substring(assessments.indexOf(']')+1);
            }

            Unit u = new UnitExpression(unitName, aes).interpret();
        }
    } finally {
        br.close();
    }
} catch (IOException e) {
    e.printStackTrace();
}
于 2012-05-26T08:46:46.103 に答える
0

BufferedReaderコンストラクターのパラメーターとして aを持つことは、これにアプローチする非常に型破りな方法です。BufferedReaderあなたのインスタンスでは、モデルデータの配信手段です。また、モデル データは、ディスクに保存する方法ではなく、ユース ケースに関するものであるべきであるため、十分に抽象的である必要があります。代わりに、各モデル クラスにデフォルトのコンストラクターを用意し、モデル クラスのすべて/一部のフィールドを初期化するために使用できるコンストラクターを使用することもできます。

ファイル インターフェイスにアプローチする 1 つの方法は、ファイルから/へデータをロード/保存できるファイル リーダー/ライター ユーティリティを作成することです。クラス階層の最上位が次の場合、次のUnitようになります。

Unit loadFromFile(String filename) {
   // read unit, assessments etc in a format you choose, build the model and return it
   ...
}

void saveToFile(String filename) {
   // write you model to a file
   ...
}

私があなたなら、JSONをデータ形式として使用します。読みやすく、デバッグも簡単です。

これが役立つことを願っています。

于 2012-05-25T17:10:33.243 に答える