我目前正在使用Android Room在Android上存储一些小数据。从理论上讲,使用 allowMainThreadQuery(( 应该没有问题,因为我的所有数据都很小,但为了将来证明我的程序,我正在尝试将所有调用移动到单独的线程中。截至目前,我在 AppConfig 中有一个静态处理程序,该类在应用程序初始启动时初始化,并且在应用程序范围内可见:
HandlerThread thread = new HandlerThread("dbHandlerThread");
thread.start();
serviceHandler = new DbHandler(thread.getLooper(),lApp.getBaseContext(),dbName);
DbHandler 是一个自定义类,它扩展了 Handler,并具有以下构造函数:
DbHandler(Looper looper, Context context, String dbName) {
super(looper);
AppDatabase db = Room.databaseBuilder(context, AppDatabase.class, dbName)
.fallbackToDestructiveMigration().build();
coDao = db.countOrderDao();
plDao = db.productListDao();
pDao = db.productDao();
}
它覆盖handleMessage(Message msg(,并根据msg.what执行各种插入/更新/删除操作,所有这些都是无效的。
在某些活动的 UI 线程中,我调用此代码:
Message msg = Message.obtain();
msg.what = AppConfig.SAVE;
//insert some data and args
AppConfig.serviceHandler.dispatchMessage(msg);
我的理解是,由于处理程序在新的线程/循环器上运行,并且由于 UI 线程没有等待任何输出,因此应该没有任何问题。但是,每当我调用此代码时,我都会收到与直接在 ui 线程中访问数据库相同的错误:
java.lang.IllegalStateException: Cannot access database on the main thread since it may potentially lock the UI for a long period of time.
我做错了什么?
我发现了错误:
dispatchMessage(msg( 重写任何线程并手动调用 handleMessage 方法,而发送消息遵循处理程序的 looper/thread。
溶液:
将 dispatchMessage(msg(替换为 sendMessage(msg(