【问题标题】:Why does my app crash when firebase rules are set to true?当 Firebase 规则设置为 true 时,为什么我的应用会崩溃?
【发布时间】: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


【解决方案1】:

com.example.myapplication.Dishes 类没有定义无参数构造函数。

所以Dishes 没有无参数构造函数,这是正确的,因为Dishes 类中只有一个构造函数

public Dishes(String name, float rating, int price) {

所以你需要另一个完全没有参数的,所以

public Dishes() {}

public Dishes(String name, float rating, int price) {
   ... rest of code

为什么会这样?也许 Firebase 正在使用空构造函数来创建 Dishes 对象以用于某些内部目的,但这只是猜测......

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2020-08-15
    • 1970-01-01
    • 1970-01-01
    • 2020-10-18
    • 1970-01-01
    相关资源
    最近更新 更多