3
public class ExternalizableClass implements Externalizable
{
  public static ExternalizableClass CACHE = new ExternalizableClass(-1);

  int id;

  public ExternalizableClass()
  {
    id = (int)(Math.random() * 1000);
  }

  public ExternalizableClass(int i)
  {
    id = i;
  }


  @Override
  public void writeExternal(ObjectOutput out) throws IOException
  {
    out.writeInt(id);
  }

  @Override
  public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException
  {
    //id = in.readInt();
    id = in.readInt();
  }


  public Object writeReplace() throws ObjectStreamException
  {
    return new Write(0);
  }

  private class Write extends ExternalizableClass
  {
    int value;

    public Write()
    {
    }

    public Write(int i)
    {
      value = i;
    }

    public Object readResolve() throws ObjectStreamException
    {
      return ExternalizableClass.CACHE;
    }


  }

  @Override
  public String toString()
  {
    return "id: " + id;
  }

  public static void main(String[] argv)
  {
    try
    {
      ExternalizableClass ex = ExternalizableClass.CACHE;

      System.out.println(ex);

      ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("temp.txt"));
      oos.writeObject(ex);

      oos.close();
    }
    catch (Exception e)
    {
      e.printStackTrace();
    }

    ExternalizableClass ex;
    try
    {
      ObjectInputStream ois = new ObjectInputStream(new FileInputStream("temp.txt"));
      ex = (ExternalizableClass) ois.readObject();

      System.out.println(ex);

      ois.close();
    }
    catch (Exception e)
    {
      e.printStackTrace();
    }

  }
}

ExternalizableClassとclass の両方に、ExternalizableClass.Writeデフォルトの (引数以外の) コンストラクターがあります。しかし、それをデシリアライズするとき、Java は不平を言います:

java.io.InvalidClassException: SeralizableTest.ExternalizableClass$Write; no valid constructor
    at java.io.ObjectStreamClass$ExceptionInfo.newInvalidClassException(ObjectStreamClass.java:150)
    at java.io.ObjectStreamClass.checkDeserialize(ObjectStreamClass.java:768)
    at java.io.ObjectInputStream.readOrdinaryObject(ObjectInputStream.java:1772)
    at java.io.ObjectInputStream.readObject0(ObjectInputStream.java:1350)
    at java.io.ObjectInputStream.readObject(ObjectInputStream.java:370)
    at SeralizableTest.ExternalizableClass.main(ExternalizableClass.java:124)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:606)
    at com.intellij.rt.execution.application.AppMain.main(AppMain.java:134)

誰が問題が何であるかを説明できますか?

4

1 に答える 1

6

問題はそれが内部クラスであることにあると思われます...したがって、実際にはパラメーターなしのコンストラクターがありません。代わりに、2 つのコンストラクターがあり、そのうちの 1 つは囲んでいるクラスのインスタンスへの参照を受け取り、もう 1 つは囲んでいるクラスのインスタンスへの参照とint.

クラス宣言 ( for ) に修飾子を追加して入れ子になったクラスにすれば、各コンストラクターに暗黙的な余分なパラメーターがないため、問題ないと思います。staticWrite

于 2014-12-27T20:47:35.023 に答える