3

我有一个静态计数器变量的问题。在一个超级类(“卡片”)中,我有一个变量来计算注册的卡片数量(这是一个票务系统)。它是这样写的:

public class Card implements Serializable    {
    private int id;
    public static int nextNr= 000;
    Card next;

    public Card(int t)   {
        id= ++nextNr;
        next= null;
    }
}

该类实现了 Serializable,我使用 ObjectStream 将卡片写入文件。

但是如果我关闭程序并重新启动它,它可以从文件中读取并确认并将文件再次添加到我的卡片注册表中。但是,超级类中的卡计数器变量被重置,我尝试注册的每张新卡都再次从 001 开始。我究竟做错了什么?似乎无法在网上找到有关此特定问题的任何信息。

解决方案: 我使用 DataOutputStream 在退出时保存它,并在启动时使用 DataInputStream 读取它。我不知道这是否是最有效的方法,但它确实有效。非常感谢您的评论,它帮助了我很多!!!!

abstract public class Card implements Serializable  {

private int type;
private int cardNr;
private static int nextNr = readCardNr();
Card next;   //COllections or not.. hmmmm

public Card(int t)   {
    cardNr= ++nextNr;
    next= null;
    type = t;
    writeCardNr();
}

public int getType(){
    return type;
}

public void setCardNr(int i) {
    cardNr= i;
}
public int getCardNr()  {
    return cardNr;
}


public static int readCardNr() {        
    try(DataInputStream inn= new DataInputStream(new FileInputStream("KortNummer")))   {
        nextNr= inn.readInt();
        inn.close();
        return nextNr;
    }
    catch(FileNotFoundException fnfe)   {
        skrivMld("Fant ingen tidligere registrerte kort. Starter nytt kortregister.");
        nextNr= 000;
        return nextNr;
    }
    catch(EOFException eofe)    {
        System.out.println("End of file");
    }
    catch(IOException ioe)  {
        skrivMld("Feilmelding: IO Exception");
    }
    return nextNr;
}

public void writeCardNr()    {
    try(DataOutputStream ut= new DataOutputStream(new FileOutputStream("KortNummer"))){
        ut.writeInt(cardNr);
    }
    catch(IOException ioe)  {
        skrivMld("Problem med skriving til fil.");
    }
}
4

6 回答 6

7

序列化不会持久化静态变量的值。因此,当再次加载该类时,将设置默认值(静态整数为零)。要保留该值,请将变量设为对象级别。

如果您仍想保留静态变量的值,则需要使用private void readObject(ObjectInputStream)和提供自定义序列化private void writeObject(ObjectOutputStream)

注意:用对象序列化静态变量(通过提供自定义序列化)可能会导致问题。

想一想这样的场景:您创建一个 Card 对象并将其序列化。静态计数器将是1。您创建另一个卡片对象并将其序列化。静态计数器将是2。因此,您创建了 10 个对象并序列化了每个对象。静态计数器将是10。因此,除非您反序列化最后一个对象,否则您将无法获得正确的计数器值。

为了避免这个问题并仍然存储卡片的数量,您可以创建一个包装类Cards

public class Cards implements Serializable    {
    private List<Card> cardList = new ArrayList<Card>();
    // getter and setter

}

在开始时首先加载(反序列化)Cards对象。每当您创建Card对象时,将其添加到Cards( cardList) 并序列化该Cards对象。

于 2013-04-23T12:48:40.220 回答
7

计数器是静态的。因此它不是任何 Card实例状态的一部分,并且序列化所有卡片不会保存计数器的值。

要么保存这个计数器值,重新加载它并明确重置它,要么在启动时从所有反序列化卡中获取最大的 ID,然后将计数器重置为这个最大值。

于 2013-04-23T12:49:32.527 回答
2

静态成员没有被序列化。它们属于类,而不是正在序列化的类的实例。

于 2013-04-23T12:50:05.880 回答
2

static fieldsstatic在 ie 范围内,one JVM如果您停止执行,他们将失去其scope and data..

于 2013-04-23T12:58:24.453 回答
1

当程序关闭并重新启动时,静态不会保存/保留该值。static 修饰符用作类级别变量,其值在任何新对象创建过程中都保持不变,并且这些对象也可以通过类级别访问该静态值。例如

public class A{
public static int val = 0;
public int verify = 0;
public A(){
val++;
}
}

A a = new A();      // val = 1
A b = new A();      // val = 2
A c = new A();      // val = 3

a.verify = A.val;   // val = 3
b.verify = A.val;   // val = 3
c.verify = A.val;   // val = 3

现在要保留该静态值,您应该将其保存到某个文件或数据库中,并在应用程序重新启动时,根据需要从那里初始化该静态值。

于 2013-04-23T12:54:13.023 回答
0

根据 Java 规范,静态变量不是实例的一部分,而是类级别的一部分。当你序列化数据时,你只为 Object 而不是 Class,所以静态不会被持久化。

重新启动后重新加载类时,它会加载变量的默认值。

于 2013-04-23T12:58:49.473 回答