从 ROOM 数据库中少获取一个对象

Getting one object less from the ROOM database

我正在 android studio 中使用 ROOM 实现本地数据库。我有问题。我成功地将数据条目插入和删除到数据库中。但是当我从数据库中获取所有数据对象时,每次我都少获取一个对象。 当 activity 开始时就没有问题,但是当我在将新数据对象插入数据库后尝试更新回收器视图时,该对象在创建新数据对象后一步被更新到回收器视图中进入,同样的问题也会发生。

下面是我的完整代码:-

package com.example.eventus.DashBoard;

import androidx.appcompat.app.AppCompatActivity;
import androidx.fragment.app.DialogFragment;
import androidx.recyclerview.widget.GridLayoutManager;
import androidx.recyclerview.widget.RecyclerView;

import android.app.AlertDialog;
import android.app.DatePickerDialog;
import android.os.Bundle;
import android.os.Handler;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.DatePicker;
import android.widget.EditText;
import android.widget.Toast;

import com.example.eventus.ApplicationDatabase.Events.Event;
import com.example.eventus.ApplicationDatabase.Events.EventsAdapter;
import com.example.eventus.ApplicationDatabase.Events.EventsDao;
import com.example.eventus.R;
import com.example.eventus.Singletons.DatabaseSingleton;
import com.google.android.material.floatingactionbutton.FloatingActionButton;

import java.text.DateFormat;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.List;

public class Dashboard extends AppCompatActivity implements View.OnClickListener, DatePickerDialog.OnDateSetListener {

    //Declaring the views
    private FloatingActionButton fbAddEvent;
    private Button btnOKNewEvent, btnCancelNewEvent;
    private EditText etDeadlineNewEvent, etTitleNewEvent, etInfoNewEvent;

    //For recycler view
    private RecyclerView rvEvents;
    private RecyclerView.LayoutManager layoutManager;
    private List<Event> events;
    private EventsAdapter eventsAdapter;

    //For alert box
    private AlertDialog.Builder alertDialogBuilder;
    private AlertDialog alertDialog;
    private View alertDialogView;

    //For database
    private DatabaseSingleton databaseSingleton;
    private EventsDao eventsDao;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_dashboard);

        //Initializing and setting onclicklisteners on views
        init();
        //RecyclerView stuff
        recyclerViewStuff();
        //new event stuff
        newEventDialog();
        //database stuff
        database();
        //loadEvents
        loadEvents();
    }

    @Override
    public void onClick(View v) {
        switch (v.getId()){
            case R.id.fb_add_event_dashboard:
                alertDialog.show();
        }
    }

    @Override
    public void onDateSet(DatePicker view, int year, int month, int dayOfMonth) {
        Calendar mCalendar = Calendar.getInstance();
        mCalendar.set(Calendar.YEAR, year);
        mCalendar.set(Calendar.MONTH, month);
        mCalendar.set(Calendar.DAY_OF_MONTH, dayOfMonth);
        String selectedDate = DateFormat.getDateInstance(DateFormat.FULL).format(mCalendar.getTime());
        etDeadlineNewEvent.setText(selectedDate);
    }

    private void init(){
        //Initializing the views
        rvEvents = findViewById(R.id.rv_events_dashboard);
        fbAddEvent = findViewById(R.id.fb_add_event_dashboard);

        //Setting onclicklisteners on views
        fbAddEvent.setOnClickListener(this);
    }

    private void recyclerViewStuff() {

        //Initializing the recycler view stuffs
        layoutManager = new GridLayoutManager(this, 2);
        rvEvents.setLayoutManager(layoutManager);

    }

    private void newEventDialog() {
        alertDialogBuilder = new AlertDialog.Builder(Dashboard.this);
        alertDialogView = getLayoutInflater().inflate(R.layout.new_event, null);
        alertDialogBuilder.setView(alertDialogView);
        alertDialog = alertDialogBuilder.create();
        alertDialog.setCanceledOnTouchOutside(false);

        //Instantiating the view
        btnOKNewEvent = alertDialogView.findViewById(R.id.btn_ok_new_event);
        etDeadlineNewEvent = alertDialogView.findViewById(R.id.et_deadline_new_event);
        btnCancelNewEvent = alertDialogView.findViewById(R.id.btn_cancel_new_event);
        etTitleNewEvent = alertDialogView.findViewById(R.id.et_title_new_event);
        etInfoNewEvent = alertDialogView.findViewById(R.id.et_info_new_event);

        //setting on click listeners for the views in dialog box
        btnOKNewEvent.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                addEvent();
                //loadEvents();
                refreshEvents();
                alertDialog.dismiss();
            }
        });

        btnCancelNewEvent.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                alertDialog.dismiss();
            }
        });

        etDeadlineNewEvent.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                DialogFragment datePicker = new com.example.eventus.DateAndTime.DatePicker();
                datePicker.show(getSupportFragmentManager(), null);
            }
        });
    }

    private void database(){
        //getting the database instance
        databaseSingleton = DatabaseSingleton.getInstance();
        //getting the events dao
        eventsDao = databaseSingleton.getEventsDao(getApplicationContext());

        //deleting all previously saved events
        eventsDao.deleteAllEvent();
    }

    private void addEvent() {
        String eventTitle, eventInfo, eventDeadline;
        eventTitle = etTitleNewEvent.getText().toString();
        eventInfo = etInfoNewEvent.getText().toString();
        eventDeadline = etDeadlineNewEvent.getText().toString();

        //creating an event
        Event event = new Event(eventTitle, eventInfo, eventDeadline);

        Handler handler = new Handler();

        Runnable runnable = new Runnable() {
            @Override
            public void run() {
                eventsDao.createEvent(event);
                Toast.makeText(Dashboard.this, "Event Created", Toast.LENGTH_SHORT).show();
            }
        };

        handler.post(runnable);
    }

    private void loadEvents(){

        events = eventsDao.getEvents();

        eventsAdapter = new EventsAdapter(events);

        rvEvents.setAdapter(eventsAdapter);
    }
    
    private void refreshEvents(){
        events.clear();

        List<Event> tempEventsList = eventsDao.getEvents();
        for(int i=0;i<tempEventsList.size();i++){
            events.add(tempEventsList.get(i));
        }

        eventsAdapter.notifyDataSetChanged();
    }
}

您遇到的问题与并发有关。如果您检查您的代码,您首先调用添加事件方法并将插入放入 Runnable 中,这会告诉应用程序在可能的情况下稍后执行,但它不会立即执行;因此代码继续执行,您获取事件,然后在稍后添加新事件。

要解决您的问题,请尝试将方法 refreshEvents 的调用放在 runnable 中。

Runnable runnable = new Runnable() {
        @Override
        public void run() {
            eventsDao.createEvent(event);
            Toast.makeText(Dashboard.this, "Event Created", Toast.LENGTH_SHORT).show();
            refreshEvents();
        }
    };

handler.post(runnable);

编辑:正如评论中正确所说的那样,不是多线程问题,而是单线程内代码片段的执行时间。