【发布时间】:2022-02-15 02:56:28
【问题描述】:
我制作了一个应用程序,它从 firebase 实时数据库中检索数据并将其显示在 RecyclerView 中。
MainActivity.java:
package com.example.myapplication;
import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;
import android.widget.Button;
import android.os.Bundle;
import android.view.View;
import android.widget.Toast;
import android.widget.TextView;
import com.google.firebase.FirebaseApp;
import com.google.firebase.database.DataSnapshot;
import com.google.firebase.database.DatabaseError;
import com.google.firebase.database.DatabaseReference;
import com.google.firebase.database.FirebaseDatabase;
import com.google.firebase.database.ValueEventListener;
import java.util.ArrayList;
public class MainActivity extends AppCompatActivity {
RecyclerView recyclerview;
ArrayList<Dishes> list;
DatabaseReference reference;
Food_RecyclerViewAdapter adapter;
Button button;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
recyclerview = findViewById(R.id.recycler);
reference = FirebaseDatabase.getInstance().getReference("Dishes");
recyclerview.setHasFixedSize(true);
recyclerview.setLayoutManager(new LinearLayoutManager(this));
button = findViewById(R.id.button);
list = new ArrayList<>();
adapter = new Food_RecyclerViewAdapter(this, list);
recyclerview.setAdapter(adapter);
reference.addValueEventListener(new ValueEventListener() {
@Override
public void onDataChange(@NonNull DataSnapshot snapshot) {
for (DataSnapshot ds : snapshot.getChildren()){
//Add firebase data to Recyclerview adapter
Dishes dish = ds.getValue(Dishes.class);
list.add(dish);
}
adapter.notifyDataSetChanged();
}
@Override
//when the connection is cancelled call the Display() method
public void onCancelled(@NonNull DatabaseError error) {
Display();
}
});
}
private void Display() {
//Display the toast message "Error"
Toast.makeText(this, "Error", Toast.LENGTH_SHORT).show();
}
}
RecyclerView 的 Model 类:
package com.example.myapplication;
class Dishes {
String name;
int price;
float rating;
public Dishes(String name, float rating, int price) {
this.name = name;
this.price = price;
this.rating = rating;
}
public String getName() {
return name;
}
public int getPrice() {
return price;
}
public float getRating() {
return rating;
}
}
RecyclerView 适配器:
package com.example.myapplication;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Button;
import android.widget.RatingBar;
import android.widget.TextView;
import androidx.annotation.NonNull;
import androidx.recyclerview.widget.RecyclerView;
import java.util.ArrayList;
class Food_RecyclerViewAdapter extends RecyclerView.Adapter<Food_RecyclerViewAdapter.MyViewHolder> {
Context context;
ArrayList<Dishes> models;
public Food_RecyclerViewAdapter(Context context, ArrayList<Dishes> models){
this.context = context;
this.models = models;
}
@NonNull
@Override
public Food_RecyclerViewAdapter.MyViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
LayoutInflater inflater = LayoutInflater.from(context);
View view = inflater.inflate(R.layout.item, parent, false);
return new Food_RecyclerViewAdapter.MyViewHolder(view);
}
@Override
public void onBindViewHolder(@NonNull Food_RecyclerViewAdapter.MyViewHolder holder, int position) {
Dishes dish = models.get(position);
holder.name.setText(dish.getName());
holder.price.setText("Price" + " $ " + dish.getPrice());
holder.rating.setRating(dish.getRating());
}
@Override
public int getItemCount() {
return models.size();
}
public static class MyViewHolder extends RecyclerView.ViewHolder{
TextView name, price;
Button button;
RatingBar rating;
public MyViewHolder(@NonNull View ItemView) {
super(ItemView);
name = ItemView.findViewById(R.id.food_name);
price = ItemView.findViewById(R.id.pri);
button = ItemView.findViewById(R.id.btn);
rating = ItemView.findViewById(R.id.ratingBar);
}
}
}
依赖关系:
dependencies {
implementation 'androidx.appcompat:appcompat:1.3.0'
implementation 'com.google.android.material:material:1.4.0'
implementation 'androidx.constraintlayout:constraintlayout:2.0.4'
implementation 'com.google.firebase:firebase-database:20.0.3'
testImplementation 'junit:junit:4.+'
androidTestImplementation 'androidx.test.ext:junit:1.1.3'
androidTestImplementation 'androidx.test.espresso:espresso-core:3.4.0'
}
但是,当我在我的设备“Samsung”上运行应用程序时,数据不会加载,并且在使用这些规则时会调用 Oncancelled 方法:
{
"rules": {
".read": false,
".write": false
}
}
当 read 和 write 设置为 true 时,应用程序就会崩溃:
{
"rules": {
".read": true,
".write": true
}
}
我猜我做错了一件事或错过了什么,是什么? 我是 firebase 的新手,因此将不胜感激解释和帮助,谢谢。
编辑:logcat 上的堆栈跟踪读取:02-14 16:40:29.800 32134-32142/? W/SQLiteConnectionPool:数据库“+data+data+com_android_vending+databases+resource_data_db”的 SQLiteConnection 对象被泄露!请修复您的应用程序以正确结束正在进行的事务并在不再需要时关闭数据库。
edit2:当应用程序崩溃时,我会收到此错误: 02-14 18:42:33.075 14095-14095/com.example.myapplication E/AndroidRuntime: 致命异常: main com.google.firebase.database.DatabaseException:类 com.example.myapplication.Dishes 未定义无参数构造函数。如果您使用 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.encoding.CustomClassMapper.convertBean(CustomClassMapper.java:433) 在 com.google.firebase.database.core.utilities.encoding.CustomClassMapper.deserializeToClass(CustomClassMapper.java:232) 在 com.google.firebase.database.core.utilities.encoding.CustomClassMapper.convertToCustomClass(CustomClassMapper.java:80) 在 com.google.firebase.database.DataSnapshot.getValue(DataSnapshot.java:203) 在 com.example.myapplication.MainActivity$1.onDataChange(MainActivity.java:54) 在 com.google.firebase.database.core.ValueEventRegistration.fireEvent(ValueEventRegistration.java:75) 在 com.google.firebase.database.core.view.DataEvent.fire(DataEvent.java:63) 在 com.google.firebase.database.core.view.EventRaiser$1.run(EventRaiser.java:55) 在 android.os.Handler.handleCallback(Handler.java:615) 在 android.os.Handler.dispatchMessage(Handler.java:92) 在 android.os.Looper.loop(Looper.java:137) 在 android.app.ActivityThread.main(ActivityThread.java:4947) 在 java.lang.reflect.Method.invokeNative(Native Method) 在 java.lang.reflect.Method.invoke(Method.java:511) 在 com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1038) 在 com.android.internal.os.ZygoteInit.main(ZygoteInit.java:805) 在 dalvik.system.NativeStart.main(Native Method)
编辑3:我在做snachsms回答后解决了这个问题,在菜中添加了另一个空构造函数:
public Dishes(){}
public Dishes(String name, float rating, int price) {
this.name = name;
this.price = price;
this.rating = rating;
}
非常感谢伙计和所有做出贡献的人。
【问题讨论】:
-
如果应用程序崩溃,会有一个堆栈跟踪。请在 logcat 上查找,并将其添加到您的问题中。
-
谢谢,我会这样做的。如果我的应用在设备上运行,当它崩溃时,堆栈跟踪会在 Android Studio 中生成吗?
-
@BelkoDIALLO 如果你还不熟悉logcat,我推荐阅读developer.android.com/studio/debug/am-logcat
-
好的,谢谢,我会在几个小时内编辑它,因为我目前没有 USB 数据线在我的设备上运行应用程序。
标签: android firebase firebase-realtime-database