1

我正在尝试从 Java Android 项目的 Firebase 实时数据库中读取数据。我正在使用 Android Studio 4.2.2。

我可以很好地获取 JSON 字符串。当我尝试反序列化数据时,问题就开始了。

那时我得到一个例外:“没有定义无参数构造函数。如果您使用 ProGuard,请确保这些构造函数没有被剥离。”

关键是我已经定义了一个无参数构造函数。我还检查了这里的文档https://firebase.google.com/docs/database/android/read-and-write

我的问题是:

  1. 我做错了什么?
  2. 如何解析具有多个级别的 JSON?
  3. 变量的名称是否必须与 JSON 的键名称相同?

提前致谢!

JSON 看起来像这样:

数据树

在 Logcat 中,我可以看到 JSON 安全到达应用程序:

{ key = 1, value = {phone=123456789, name=Mike} }

这里有代码:

public class DataBaseHelper {

private DatabaseReference mDatabase;
public TestData result;

public DataBaseHelper() {
    mDatabase = FirebaseDatabase.getInstance().getReference();

    // Creamos query
    Query query = mDatabase.child("TestData");


    query.addValueEventListener(new ValueEventListener() {
        @Override
        public void onDataChange(DataSnapshot dataSnapshot) {
            for (DataSnapshot userSnapshot : dataSnapshot.getChildren()) {
                try {
                    Log.w("datosdb", userSnapshot.toString());
                    result = userSnapshot.getValue(TestData.class);

                } catch (Exception e) {
                    Log.w("datosdb", "Parse failed", e);
                }
            }
        }

        @Override
        public void onCancelled(DatabaseError databaseError) {
            // Getting Post failed, log a message
            Log.w("datosdb", "loadPost:onCancelled", databaseError.toException());
            // ...
        }
    });


}


public class TestData {
    public TestData() {
    }

    public String name;
    public int phone;

    public String getName() {
        return this.name;
    }

    public int getPhone() {
        return this.phone;
    }
}

}

这是一个例外:

2021-07-12 00:58:01.631 25262-25262/com.example.purinasa W/datosdb: DataSnapshot { key = 1, value = {phone=123456789, name=Mike} } 2021-07-12 00:58: 01.637 25262-25262/com.example.purinasa W/datosdb:解析失败 com.google.firebase.database.DatabaseException:com.example.purinasa.application.dataBase.DataBaseHelper$TestData 类未定义无参数构造函数。如果您使用 ProGuard,请确保未剥离这些构造函数。在 com.google.firebase.database.core.utilities.encoding.CustomClassMapper$BeanMapper.deserialize(CustomClassMapper.java:570) 在 com.google.firebase.database.core.utilities.encoding.CustomClassMapper$BeanMapper.deserialize(CustomClassMapper. java:563) 在 com.google.firebase.database.core.utilities 的 com.google.firebase.database.core.utilities.encoding.CustomClassMapper.convertBean(CustomClassMapper.java:433)。

更新:

在使用@Ticherhaz-FreePalestine 进行一些测试后,我们设法使用以下行单独反序列化数据:

 String name = userSnapshot.child("name").getValue(String.class);
                    Log.w("datosdb", "String is here!!!" + name);

                    long phone = userSnapshot.child("phone").getValue(Long.class);
                    Log.w("datosdb", "Long is here!!!" + phone);

顺便说一句,我找不到在不引发异常的情况下使用另一个代码的方法。

4

2 回答 2

1

最后,它通过使用 TestData 作为独立类而不是内部类来工作。我创建了一个新文件 TestData.java 并将其代码移到那里。

所以最后 DatabaseHelper 像这样丢失:

public class DatabaseHelper {

private DatabaseReference mDatabase;
TestData result;

public DatabaseHelper() {
    mDatabase = FirebaseDatabase.getInstance().getReference();

    // We create a query
    Query query = mDatabase.child("TestData");

    //We recive query
    query.addValueEventListener(new ValueEventListener() {
        @Override
        public void onDataChange(DataSnapshot dataSnapshot) {
            for (DataSnapshot userSnapshot : dataSnapshot.getChildren()) {
                Log.w("datosdb", "Tenemos el resultado de la query.");
                try {
                    Log.w("datosdb", userSnapshot.toString());

                    // Deserializing only name
                    String name = userSnapshot.child("name").getValue(String.class);
                    Log.w("datosdb", "String is here!!!" + name);

                    // Deserializing only phone
                    long phone = userSnapshot.child("phone").getValue(Long.class);
                    Log.w("datosdb", "Long is here!!!" + phone);

                    // Deserializing into a TestData object
                    TestData result = userSnapshot.getValue(TestData.class);
                    Log.w("datosdb", "Parse success!!!");

                } catch (Exception e) {
                    Log.w("datosdb", "Parse failed", e);
                }
            }
        }

        @Override
        public void onCancelled(DatabaseError databaseError) {
            // Getting Post failed, log a message
            Log.w("datosdb", "loadPost:onCancelled", databaseError.toException());
            // ...
        }
    });


}

}

TestData 看起来像这样(现在在它自己的 .java 文件中):

public class TestData {
private String name;
private long phone;

public TestData(){}

public TestData(String name, long phone){
    this.name = name;
    this.phone = phone;
}

public String getName() {
    return this.name;
}

public void setName(String name) {
    this.name = name;
}

public long getPhone() {
    return this.phone;
}

public void setPhone(long phone) {
    this.phone = phone;
}

}

于 2021-07-12T02:03:29.753 回答
0

您已经有一个空的构造函数。所以需要另一个具有所有属性的构造函数。

@Keep
public class TestData {
    private String name;
    private int phone;

    public TestData() {
    }

    public TestData(String name, int phone) {
        this.name = name;
        this.phone = phone;
    }

    public String getName() {
        return this.name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getPhone() {
        return this.phone;
    }

    public void setPhone(int phone) {
        this.phone = phone;
    }
}
于 2021-07-11T23:14:07.993 回答