Команды Android Room не работают

#java #android #android-room

Вопрос:

Я работаю над приложением «Задачи». Я хочу сохранить такие данные, как имя задачи, время и приоритет в комнате. Я также создал необходимые классы. Однако всякий раз, когда я пытаюсь удалить какой-либо конкретный элемент из задач, приложение выходит из строя. Я довольно новичок в использовании комнаты, поэтому, пожалуйста, простите меня, если я допустил глупую ошибку. Пожалуйста, сообщите, требуется ли вам код из класса базы данных комнаты или класса dao… Это трассировка стека:

 java.lang.IllegalStateException: Cannot access database on the main thread since it may potentially lock the UI for a long period of time.
        at androidx.room.RoomDatabase.assertNotMainThread(RoomDatabase.java:385)
        at androidx.room.SharedSQLiteStatement.assertNotMainThread(SharedSQLiteStatement.java:60)
        at androidx.room.SharedSQLiteStatement.acquire(SharedSQLiteStatement.java:86)
        at com.example.taskmasterv3.UserDao_Impl.deleteAll(UserDao_Impl.java:120)
        at com.example.taskmasterv3.SubtaskAdapter$1.onClick(SubtaskAdapter.java:85)
        at android.view.View.performClick(View.java:7520)
        at android.view.View.performClickInternal(View.java:7489)
        at android.view.View.access$3600(View.java:826)
        at android.view.View$PerformClick.run(View.java:28555)
        at android.os.Handler.handleCallback(Handler.java:938)
        at android.os.Handler.dispatchMessage(Handler.java:99)
        at android.os.Looper.loop(Looper.java:245)
        at android.app.ActivityThread.main(ActivityThread.java:8004)
        at java.lang.reflect.Method.invoke(Native Method)
        at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:631)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:978)
 

Это код, в котором происходит удаление :

 // full subtask adapter code

package com.example.taskmasterv3;

import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ArrayAdapter;
import android.widget.ImageView;
import android.widget.TextView;
import android.widget.Toast;

import androidx.annotation.NonNull;
import androidx.annotation.Nullable;

import java.util.ArrayList;

public class SubtaskAdapter extends ArrayAdapter<subtask> {


    private final Context context;
    private ArrayList<subtask> values;


    public SubtaskAdapter(Context context, ArrayList<subtask> list) {

        //since your are using custom view,pass zero and inflate the custom view by overriding getview

        super(context, 0 , list);
        this.context = context;
        this.values = list;

    }





    @Override
    public View getView(int position, @Nullable View convertView, @NonNull ViewGroup parent) {


        //check if its null, if so inflate it, else simply reuse it
        if (convertView == null) {
            LayoutInflater inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
            convertView = inflater.inflate(R.layout.subtask_item, parent, false);
        }

        //use convertView to refer the childviews to populate it with data
        TextView tvSubtaskName = convertView.findViewById(R.id.tvsumtaskname);
        ImageView ivPri = convertView.findViewById(R.id.ivsumPri);
        ImageView ivTime = convertView.findViewById(R.id.ivsumTime);
        ImageView ivDelete = convertView.findViewById(R.id.ivDelete);

        String subname = tvSubtaskName.getText().toString().trim();

        ivDelete.setTag(position);


















// This is the part that is giving me the error :


        ivDelete.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
            int position = (Integer) v.getTag();
            remove(values.remove(position));
            AppDatabase.getDatabase(context).userDao().deleteAll();
            notifyDataSetChanged();


            }
        });



        tvSubtaskName.setText(values.get(position).getSubtaskName());

        if (values.get(position).isPriHigh()) {
            ivPri.setImageResource(R.drawable.priority_high);
        } else if (values.get(position).isPriMed()) {
            ivPri.setImageResource(R.drawable.priority_med);
        } else if (values.get(position).isPriLow()) {
            ivPri.setImageResource(R.drawable.priority_low);
        }

        if (values.get(position).isTimeMore()) {
            ivTime.setImageResource(R.drawable.time_symbol_more);
        } else if (values.get(position).isTimeMed()) {
            ivTime.setImageResource(R.drawable.time_symbol_med);
        } else if (values.get(position).isTimeLess()) {
            ivTime.setImageResource(R.drawable.time_symbol_less);
        }





        //return the view you inflated
        return convertView;
    }





    //to keep adding the new subtasks try the following
    public void addANewSubTask(subtask newSubTask){
        ArrayList<subtask> newvalues = new ArrayList<>(this.values);
        newvalues.add(newSubTask);
        this.values = newvalues;
        notifyDataSetChanged();

    }

    @Override
    public int getCount() {
        if (this.values.size()>11)
        {
            Toast.makeText(context, "Max subtask limit reached", Toast.LENGTH_SHORT).show();
            return 10;

        }
        else
        {
            return super.getCount();
        }
    }


}
 

Это код, в котором я пытаюсь сохранить данные в комнате :

 package com.example.taskmasterv3;

import androidx.annotation.Nullable;
import androidx.appcompat.app.AppCompatActivity;
import androidx.room.Dao;
import androidx.room.Database;
import androidx.room.Delete;
import androidx.room.Entity;
import androidx.room.Insert;
import androidx.room.PrimaryKey;
import androidx.room.Query;
import androidx.room.Room;
import androidx.room.RoomDatabase;

import android.content.Intent;
import android.database.SQLException;
import android.os.AsyncTask;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.LinearLayout;
import android.widget.RadioButton;
import android.widget.RadioGroup;
import android.widget.Toast;

import java.util.ArrayList;
import java.util.List;

public class SubtaskActivity extends AppCompatActivity {

    EditText etSubtaskName;
    EditText etTaskName;
    Button btnDone, btnCancel;
    RadioGroup radgrpPri, radgrpTime;
    RadioButton radbtnPriHigh, radbtnPriMed, radbtnPriLow, radbtnTimeMore, radbtnTimeMed, radbtnTimeLess;
    boolean priHigh, priMed, priLow, timeMore, timeMed, timeLess;
    String subtaskName;
    String pri;
    String time;


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

        btnDone = findViewById(R.id.btnDone);
        radgrpPri = findViewById(R.id.radgrpPri);
        radgrpTime = findViewById(R.id.radgrpTime);
        radbtnPriHigh = findViewById(R.id.radbtnPriHigh);
        radbtnPriMed = findViewById(R.id.radbtnPriMed);
        radbtnPriLow = findViewById(R.id.radbtnPriLow);
        radbtnTimeMore = findViewById(R.id.radbtnTimeMore);
        radbtnTimeMed = findViewById(R.id.radbtnTimeMed);
        radbtnTimeLess = findViewById(R.id.radbtnTimeLess);
        etSubtaskName = findViewById(R.id.etSubtaskName);
        btnCancel = findViewById(R.id.btnCancel);





        radgrpPri.setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {
            @Override
            public void onCheckedChanged(RadioGroup group, int checkedId) {

                if (radbtnPriHigh.isChecked())
                {
                    priHigh = true;
                    priLow = false;
                    priMed = false;
                    pri = "h";




                }
                else if (radbtnPriMed.isChecked())
                {
                    priHigh = false;
                    priLow = false;
                    priMed = true;
                    pri = "m";


                }
                else if (radbtnPriLow.isChecked())
                {
                    priHigh = false;
                    priLow = true;
                    priMed = false;
                    pri = "l";
                }
                else {
                    priLow = true;
                    pri = "l";

                }

            }
        });

        radgrpTime.setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {
            @Override
            public void onCheckedChanged(RadioGroup group, int checkedId) {

                if (radbtnTimeMore.isChecked())
                {
                    timeMore = true;
                    timeMed = false;
                    timeLess = false;

                    time = "more";
                }
               else if (radbtnTimeMed.isChecked())
                {
                    timeMore = false;
                    timeMed = true;
                    timeLess = false;
                    time = "med";
                }
               else if (radbtnTimeLess.isChecked())
                {
                    timeMore = false;
                    timeMed = false;
                    timeLess = true;
                    time = "less";
                }
               else
                {
                    timeLess = true;
                    time = "less";
                }

            }
        });






        btnDone.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {




               String name = etSubtaskName.getText().toString().trim();

               Intent intent = new Intent(SubtaskActivity.this, TaskInfo.class);
               intent.putExtra("subtaskName", name);
               intent.putExtra("priHigh", priHigh);
               intent.putExtra("priMed", priMed);
               intent.putExtra("priLow", priLow);
               intent.putExtra("timeMore", timeMore);
               intent.putExtra("timeMed", timeMed);
               intent.putExtra("timeLess", timeLess);
               setResult(RESULT_OK, intent);
               SubtaskActivity.this.finish();
               
               // THIS IS WHERE I SAVE THE DATA IN ROOM

                AsyncTask.execute(new Runnable() {
                    @Override
                    public void run() {
                        SubtaskDetails subtaskDetails = new SubtaskDetails(etSubtaskName.toString().trim(), pri, time);
                        AppDatabase.getDatabase(getApplicationContext()).userDao().insertAll(subtaskDetails);
                    }
                });



            }
        });



        btnCancel.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {

                etSubtaskName.setText("");
                radgrpPri.clearCheck();
                radgrpTime.clearCheck();
                finish();



            }
        });






    }

}
 

Комментарии:

1. Создание AsyncTasls для всех операций поможет

2. Используйте RxJava для сохранения данных в базе данных комнат. Вы не можете сохранить данные в пространство в основном потоке, вам нужно выполнить эту задачу в фоновом потоке. Если вы используете kotlin , используйте сопрограмму.

3. Я использовал asynctask, где у меня был код удаления. Работает и сейчас..

Ответ №1:

Во-первых, AsyncTask устарел на уровне API 30

В документах добавляется, что AsyncTasks в идеале следует использовать для коротких операций (не более нескольких секунд). Если вам необходимо поддерживать работу потоков в течение длительного периода времени, настоятельно рекомендуется использовать различные API, предоставляемые java.util.concurrent пакетом, такие как Executor, ThreadPoolExecutor и FutureTask.

Я бы посоветовал вам работать с исполнителем, который создает новый поток для каждой задачи, освобождая тем самым основной поток.

Существует идеальный учебник, с которым вы можете ознакомиться здесь

Ответ №2:

Как сказал вам исключение:

java.lang.Исключение IllegalStateException: Не удается получить доступ к базе данных в основном потоке, так как это потенциально может заблокировать пользовательский интерфейс на длительный период времени.

Вам нужно получить доступ к базе данных из сопрограммы, а не из основного потока

Сопрограммы для Android

Ответ №3:

 Thread(Runnable {
        Handler(Looper.getMainLooper()).post(Runnable {
            SubtaskDetails subtaskDetails = new SubtaskDetails(etSubtaskName.toString().trim(), pri, time);
            AppDatabase.getDatabase(getApplicationContext()).userDao().insertAll(subtaskDetails);
        })
    }).start()
 

Попробуйте это, ваша проблема заключалась в том, что вы пробовали функцию в основном потоке, которая создает проблему, попробуйте в другом потоке!