领域从不正确的线程Android访问

时间:2015-05-19 08:46:13

标签: java android multithreading realm

我在Android上处理这个问题:

Realm从错误的线程访问。 Realm对象只能在创建它们的线程上访问。

我想在我的RemoteViewsFactory中使用Realm和

public class RemoteViewsX implements RemoteViewsFactory {

    public RemoteViews getViewAt(int paramInt) {
    if (this.results != null && this.results.size() != 0 && this.results.get(paramInt) != null) {
    //FAILED HERE
    }

}

... 这个电话失败了!为什么?

我在班上取这样的数据:

public void onDataSetChanged() {
        Realm realm = Realm.getInstance(RemoteViewsX.this.ctx);
        this.results =  realm.where(Model.class).findAll();
}

我这样调用了我的remoteFactory:

public class ScrollWidgetService extends RemoteViewsService {
    @Override
    public RemoteViewsFactory onGetViewFactory(Intent intent) {
        return new RemoteViewsX (getApplicationContext());
    }
}

有什么想法吗?

2 个答案:

答案 0 :(得分:2)

如果问题是通过从另一个线程调用onDataSetChangedgetViewAt引起的,您可以强制他们使用相同的线程,创建您自己的HandlerThread,如下所示:

public class Lock {
    private boolean isLocked;

    public synchronized void lock() throws InterruptedException {
        isLocked = true;
        while (isLocked) {
            wait();
        }
    }

    public synchronized void unlock() {
        isLocked = false;
        notify();
    }
}

public class MyHandlerThread extends HandlerThread {
    private Handler mHandler;

    public MyHandlerThread() {
        super("MY_HANDLER_THREAD");
        start();
        mHandler = new Handler(getLooper());
    }

    public Handler getHandler() {
        return mHandler;
    }
}

public class RemoteViewsX implements RemoteViewsFactory {
    private MyHandlerThread mHandlerThread;
    ...
}

public void onDataSetChanged() {
    Lock lock = new Lock();
    mHandlerThread.getHandler().post(new Runnable() {
        @Override
        public void run() {
            Realm realm = Realm.getInstance(ctx);
            results = realm.where(Model.class).findAll();
            lock.unlock();
        }
    });
    lock.lock();
}

public RemoteViews getViewAt(int paramInt) {
    Lock lock = new Lock();
    final RemoteViews[] result = {null};
    mHandlerThread.getHandler().post(new Runnable() {
        @Override
        public void run() {
            // You can safely access results here.
            result[0] = new RemoteViews();
            lock.unlock();
        }
    });
    lock.lock();
    return result[0];
}

我从此页面复制了Lock课程:http://tutorials.jenkov.com/java-concurrency/locks.html
完成任务后,不要忘记quit处理程序线程。

答案 1 :(得分:0)

我基于RxJava,所以我在一个领域做到这一点。我克隆每个项目,因为它们是主线程的一部分,当我在另一个线程如小部件主屏幕工作时,这会搞砸。

myRealm.where( Dog.class ).findAllAsync().subscribe( mainThreadDogs->{
       thisThreadDogs.clear();

       for( Dog dog: mainThreadDogs ){
            thisThreadDogs.add( ModelUtil.cloneDog( dog ) ); 
       }    
});