Serializacaja danych problem z odczytem

0

Mam przykładową klasę Person z dwoma polami imie i nazwisko. Zapisuje do pliku poprzez. Dane w pliku się zapisują. Problem jest odczytem

 String fileName = "person.obj";
FileOutputStream fs = new FileOutputStream(fileName, true);
                ObjectOutputStream os = new ObjectOutputStream(fs);
 List<Person> per = new ArrayList<>();
os.writeObject(new Person("JAGODA ","BRONSKA"));

Przyróbie odczytu danych poprzez


 FileInputStream fis = new FileInputStream(fileName);
                ObjectInputStream ois = new ObjectInputStream(fis);
  List<Person> pera = (List<Person>)ois.readObject();
for (Person el: pera){
    System.out.println(el.getFirstName()+" "+el.getLastName());
}

Mam bład
Exception in thread "main" java.lang.ClassCastException: com.company.Person cannot be cast to java.util.List
at com.company.Main.main(Main.java:58)
Jak sobie z tym poradzić

0

Serializujesz pojedynczy obiekt, a przy deseralizacji oczekujesz listy...

0

Zapisuje kilka obiektów

os.writeObject(new Person("Oleg ","Banan"));
os.writeObject(new Person("jan ","nowak"));

W takim razie jak to odczytac

1
List<Person> per = new ArrayList<>();
os.writeObject(new Person("JAGODA ","BRONSKA"));

Zapisałeś tylko Person, a listę utworzyłeś ale nic z nią nie zrobiłeś. Spróbuj tak:

List<Person> per = new ArrayList<>();
per.add(new Person("JAGODA ","BRONSKA"));
os.writeObject(per);
0

Dziękuję za odpowiedzi. Poprawiłem i działa. Ale jak zapisze np. dwa nazwiska i je wczytam jest ok. Ale jak poźniej dopisze kolejne dwa widzę że w pliku dopisały się te nowe nazwiska ale wczytuje i wyświetla tylko te pierwsze dwa

0

Klasa Person

package com.company;

import java.io.Serializable;

public class Person implements Serializable {
     private static final long serialVersionUID = 3812017177088226528L;
    private String firstName;
    private String lastName;

    public String getFirstName() {
        return firstName;
    }

    public void setFirstName(String firstName) {
        this.firstName = firstName;
    }

    public String getLastName() {
        return lastName;
    }

    public void setLastName(String lastName) {
        this.lastName = lastName;
    }

    public Person(){}

    public Person(String firstName, String lastName) {
        setFirstName(firstName);
        setLastName(lastName);
    }

    public String toString() {
        return firstName + " " + lastName;
    }

}

potem

package com.company;

import java.io.*;
import java.util.ArrayList;
import java.util.List;
import java.util.Scanner;

public class Main {

    public static void main(String[] args) throws IOException, ClassNotFoundException {
        Person person = new Person();
        String fileName = "person.obj";

        FileOutputStream fs = new FileOutputStream(fileName,true);
       ObjectOutputStream os = new ObjectOutputStream(fs);

        List<Person> per = new ArrayList<>();
       per.add(new Person("Jurek", "Nowak"));
        per.add(new Person("Wacek ", "Zielonka"));
       os.writeObject(per);
 os.flush();
        os.close();

        System.out.println("Zapisano obiekt do pliku");

        // ODczyt danych z pliku ///////////////////////////////



        FileInputStream fis = new FileInputStream(fileName);
        ObjectInputStream ois = new ObjectInputStream(fis);


        List<Person> pera = (List<Person>) ois.readObject();


       
        for (Person el : pera) {
            System.out.println(el.getFirstName() + " " + el.getLastName());
            ois.close();
        }


       

    }

}

Wyświetla
Zapisano obiekt do pliku
Jurek Nowak
Wacek Zielonka

Potem zmieniam nazwiska uruchamiam program ( w pliku widzę że nowe nazwiska się dodały) ale wyświetla tylko te dwa zapisane za pierwszym razem/

Zawartość pliku.

¬í sr java.util.ArrayListxÒ™Ça I sizexp   w   sr com.company.Person4çlöôà L 	firstNamet Ljava/lang/String;L lastNameq ~ xpt Jurekt Nowaksq ~ t Wacek t Zielonkax¬í sr java.util.ArrayListxÒ™Ça I sizexp   w   sr com.company.Person4çlöôà L 	firstNamet Ljava/lang/String;L lastNameq ~ xpt Mirekt kaziuksq ~ t Aga t 	kruszonkax
0

Problem jest to, że tak naprawdę nie dopisujesz osób do listy tylko tworzysz nową listę i to ją dopisujesz do pliku. Dlatego za drugim razem jak chcesz wyświetlić listę to pobierasz tylko te pierwszą i ją wyświetlasz, a drugą olewasz.
Tutaj masz rozwiązanie: Zrezygnuj z listy i dopisuj same obiekty Person, a potem czytaj je w pętli jak w tym przykładzie:
https://stackoverflow.com/questions/27409718/java-reading-multiple-objects-from-a-file-as-they-were-in-an-array

0

Dzięki za odpowiedź teraz rozumiem dlaczego tak się dzieje. Niestety nie bardzo mogę połapać się w tym przykładzie.Może masz jakiś inny. Z góry Dziękuję!!!

0

Tu masz esencje rozwiązania:

FileInputStream fis = new FileInputStream("outputFile");
ArrayList<Object> objectsList = new ArrayList<Object>();
boolean cont = true;
try{
   ObjectInputStream input = new ObjectInputStream(fis);
   while(cont){
      Object obj = input.readObject();
      if(obj != null)
         objectsList.add(obj);
      else
         cont = false;
   }
}catch(Exception e){
   //System.out.println(e.printStackTrace());
}

Jak widzisz obiekt pobierany jest ze streama za pomocą:

Object obj = input.readObject();

I działa to w pętli while tak długo aż readObject nie zwróci null. Za każdym razem gdy zwróci wartość inną niż null to ten obiekt dopisywany jest do listy. Powiedz czego konkretnie nie rozumiesz to spróbuję Ci wyjaśnić

0

Ale co z zapisywaniem. Czy muszę zmienić kod? Jak? Bo nadal wczytuje mi tylko te elementy które zapisałem za pierwszym razem do pliku.

0

Chyba że coś źle wyświetlam

for (Object el : objectsList) {
            System.out.println(el.toString());

        }
1

@mprofitw: Nie wiem co Ci mam doradzić. Musisz pokombinować i to zrobić samemu. Jedyne co mogę więcej zrobić to napisać Ci gotowe rozwiązanie ale to nie o to chodzi.

  1. Zrezygnuj z ArrayList przy zapisywaniu
  2. Zapisuj po prostu obiekty do pliku
  3. Wczytuj obiekty z pliku w pętli while jeden po drugim jak wklejałem wcześniej
  4. Każdy odczytany obiekt wrzucaj do ArrayListy
0

Mój kod wygląda tak pomimo że zrezygnowałem z listy przy zapisie za każdym razem wczytują się tylko te nazwiska z pliku, które były zapisane za pierwszym razem pomimo że w pliku są dopisane kolejne. Naprawdę nie wiem jak sobie z tym poradzić. Utknąłem w tym pkt. mojej aplikacji!

package com.company;

import java.io.*;
import java.util.ArrayList;


public class Main {

    public static void main(String[] args) throws IOException, ClassNotFoundException {
        Person person = new Person();
        String fileName = "person.obj";

        FileOutputStream fs = new FileOutputStream(fileName, true);
        ObjectOutputStream os = new ObjectOutputStream(fs);

        os.writeObject(new Person("Gargamel", "benek"));
        os.writeObject(new Person("Franek", "Paluszka"));
        os.close();

        System.out.println("Zapisano obiekt do pliku");

        // ODczyt danych z pliku ///////////////////////////////

        FileInputStream fis = new FileInputStream("person.obj");
        ArrayList<Object> objectsList = new ArrayList<Object>();
        boolean cont = true;
        try {
            ObjectInputStream input = new ObjectInputStream(fis);
            while (cont) {
                Object obj = input.readObject();
                if (obj != null)
                    objectsList.add(obj);
                else
                    cont = false;
            }
        } catch (Exception e) {
            //System.out.println(e.printStackTrace());
        }
        for (Object el : objectsList) {
            System.out.println(el.toString());

        }

    }

}
1
  1. Nie zjadaj wyjątków w taki sposób. W kodzie produkcyjnym to nie jest najlepszy pomysł ale jak szukasz błędu i zjadasz wyjątki to sam rzucasz sobie kłody pod nogi. Już lepiej zrobić pritnStackTrace. Jeśli chciałeś uniknąć brudzenia poprzez drukowanie java.io.EOFException (co jest normalne w Twoim wypadku i nie świadczy o błędzie) to mogłeś zrobić catch na EOFException tylko.

  2. Przed dopisaniem do pliku powinno się zawołać metodę reset() na streamie. Przeczytaj jdoca tej metody choć może niewiele Ci on powie, w internecie znajdziesz więcej info na ten temat.

Tu masz działający kod:

public static void main(String[] args) throws IOException, ClassNotFoundException {
        String fileName = "person.obj";
        Path path= Paths.get(fileName);
        
        ObjectOutputStream os;
        if(!Files.exists(path)) {
        	os = new ObjectOutputStream(new FileOutputStream(fileName));
        }
        else {
        	os = new ObjectOutputStream(new FileOutputStream(fileName, true)) {
        		protected void writeStreamHeader() throws IOException {
                    reset();
                }
        	};
        }
 
        os.writeObject(new Person("Gargamel", "benek"));
        os.writeObject(new Person("Franek", "Paluszka"));
        os.close();
 
        System.out.println("Zapisano obiekt do pliku");
 
        // ODczyt danych z pliku ///////////////////////////////
 
        FileInputStream fis = new FileInputStream("person.obj");
        ArrayList<Object> objectsList = new ArrayList<Object>();
        boolean cont = true;
        try {
			ObjectInputStream input = new ObjectInputStream(fis);
            while (cont) {
                Object obj = input.readObject();
                if (obj != null)
                    objectsList.add(obj);
                else
                    cont = false;
            }
        } catch (EOFException e) {
//            e.printStackTrace(); //tego wyjątku niekoniecznie chcemy printowac
        }
        for (Object el : objectsList) {
            System.out.println(el.toString());
 
        }
 
    }
0

Bardzo dziękuję za całą pomoc i poświęcony czas. Teraz wszystko działa tak jak należy. Jeszcze jedno małe pytanko. Dlaczego nie mogę dodać tej listy do tavleview. Wcześniej nie miałem problemu z tym.

tableViewPerson.getItems().addAll(objectsList);
0

Już działa poprawiłem na

@FXML
    private TableView<Object> tableViewPerson;

1 użytkowników online, w tym zalogowanych: 0, gości: 1