房间数据库、RecyclerView和并发问题



我在开发一款Android应用程序时遇到了麻烦,该应用程序必须将数据存储在SQLite数据库(Room database(中。我想做的是从SQLite表中检索数据,并将其显示在RecyclerView上。然而,根据我如何修改代码,我会得到三种不同的错误输出:

1.-如果我把数据库访问放在主线程中,错误消息类似于:"数据库访问不能从主线程完成,因为应用程序可能会冻结"。

2.-如果我在另一个线程中编写数据库访问代码,它会告诉我只有创建视图的线程才能访问这些视图。

3.-如果我把数据库访问和Recycler-Adapter声明都放在另一个线程中,我会得到令人担忧的"空指针异常",因为它找不到发送到Adapter对象的ArrayList。

这是我的代码:

menuPrincipal.java:

public class menuPrincipal extends AppCompatActivity {
public BaseDatosApp basedatos;
public List<Lista> listas;
public RecyclerView rcv;
public RecyclerView.LayoutManager gestorLayouts;
private Intent actividadPrincipal;
private Button btnNuevaLista, btnAbrirLista, btnGuardarLista, btnBorrarLista;
private ManejaBotonesMnuPrinc manejador;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_menu_principal);
basedatos = Room.databaseBuilder(getApplicationContext(), BaseDatosApp.class, "lista-compra").build();
Thread hilo=new Thread(new Runnable(){
@Override
public void run() {
listas = new ArrayList<>();
RecyclerAdapter rca;
basedatos.listaDao().insert(new Lista("Otra lista nueva"));
listas = basedatos.listaDao().getAll();
rca = new RecyclerAdapter(listas);
rcv = findViewById(R.id.recycler_listcom);
rcv.setLayoutManager(new LinearLayoutManager(getApplicationContext()));
rcv.setAdapter(rca);
}
});
btnNuevaLista = findViewById(R.id.btnNuevaLista);
btnAbrirLista = findViewById(R.id.btnAbrirLista);
btnGuardarLista = findViewById(R.id.btnGuardarLista);
btnBorrarLista = findViewById(R.id.btnBorrarLista);
btnNuevaLista.setOnTouchListener(manejador);
btnAbrirLista.setOnTouchListener(manejador);
btnGuardarLista.setOnTouchListener(manejador);
btnBorrarLista.setOnTouchListener(manejador);
manejador = new ManejaBotonesMnuPrinc();
actividadPrincipal = new Intent(this, MainActivity.class);
}

RecyclerAdapter.java

public class RecyclerAdapter extends RecyclerView.Adapter<RecyclerAdapter.ListasViewHolder>{
public List<Lista> listasElem;
public RecyclerAdapter(List<Lista> listas){
listasElem=listas;
}
@NonNull
@Override
public ListasViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
Context miContexto = parent.getContext();
int layoutIdParaListItem = R.layout.elemento_texto;
LayoutInflater inflador=LayoutInflater.from(miContexto);
boolean attachToParentRapido=false;
View vista = inflador.inflate(layoutIdParaListItem,parent,attachToParentRapido);
ListasViewHolder viewHolder = new ListasViewHolder(vista);
return viewHolder;
}
@Override
public void onBindViewHolder(@NonNull ListasViewHolder holder, int position){
holder.elementoListaView.setText(listasElem.get(position).getNombre());
}
@Override
public int getItemCount() {
return listasElem.size();
}
class ListasViewHolder extends RecyclerView.ViewHolder{
TextView elementoListaView;
public ListasViewHolder(@NonNull View itemView) {
super(itemView);
elementoListaView = itemView.findViewById(R.id.elementoLista_tv);
}
}
}

如果你有什么建议,请告诉我。

我建议使用rxJava 2执行此任务https://medium.com/androiddevelopers/room-rxjava-acb0cd4f3757在你的情况下,代码看起来像这个

db.employeeDao().getAll()
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new Observer<List<Employee>>() {
@Override
public void accept(List<Employee> employees) throws Exception {
// set data to adapter
}
});

最新更新