线程/处理程序错误 - 尚未发布指定的消息队列同步屏障标记

时间:2013-04-07 22:31:44

标签: java android multithreading handler

我收到此错误 -

  

java.lang.IllegalStateException:指定的消息队列同步障碍标记尚未发布或已被删除。

作为Java / Android的相对新手,毫无疑问我错过了一些东西,但我正在做的是这个 -

我有一个项目使用Exif数据根据拍摄日期显示照片,目的是在每个舞台上使用类似的模型......

工作人员线程 - > UI线程 - >定制显示适配器。然后单击GridView中的一个“单元格”将触发下一个活动。第一个活动搜索所有照片文件,创建一个“年”列表,然后每个后续活动将其过滤到几个月,几天等。

然后启动第二个活动直接启动上述错误,并通过基本的线程/处理程序设置处理消息。

这是将消息传递给线程的类 -

public class MonthSort {
Handler handler;
int imageWidth;
List<PhotoData> photoList;
public MonthSort(Handler handler2, int width, List<PhotoData> pList) {
    photoList = new ArrayList<PhotoData>();
    photoList = pList;
    imageWidth = width;
    handler = handler2;
}

public void sortFiles()
{
    int month, photoCount;
    File fileName = new File("");
    Message msg = handler.obtainMessage();
    //Message msg = Message.obtain();
    //Bundle bundle = new Bundle();
    try {
        for (int i = 0; i < 12; i++) {
            month = i + 1;
            photoCount = 0;
            for (PhotoData pd : photoList) {
                if(month == pd.month)
                {
                    if(photoCount == 0)
                        fileName = pd.fileName;
                    photoCount++;
                }
            }
            if(photoCount != 0)
            {

                Bundle bundle = new Bundle();
                bundle.putString("filename", fileName.toString());
                bundle.putInt("month", month);
                bundle.putInt("count", photoCount);
                byte[] thumbNail = getThumbnail(fileName, imageWidth);
                bundle.putByteArray("thumbnail", thumbNail);


                msg.setData(bundle);
                handler.sendMessage(msg);

            }
        }
    } catch (Exception e) {
        // TODO Auto-generated catch block
        Log.d("Debug", "handler error occurs in monthSort class");
    }
    /*Bundle bundle = new Bundle();
    bundle.putBoolean("end", true);
    msg.setData(bundle);
    handler.sendMessage(msg);*/
}

...这是在UI线程中接收它的代码。

public class MonthActivity extends Activity {
List<PhotoData> photoList;
static List<MonthData> photos;
int imageWidth;
GridView photoGrid;
static ImageAdapter2 iAdapter;
int year;
Thread monthSortThread;

Handler handler2 = new Handler() {
    @Override
    public void handleMessage(Message msg) 
    {
        Bundle bundle = msg.getData();  // Get the message sent to the Handler.
        boolean ended = bundle.getBoolean("end");
        if(ended)
        {
            //Toast.makeText(getBaseContext(), "FINISHED !!!", Toast.LENGTH_LONG).show();
        } else
        {
            try {
                MonthData md = new MonthData();
                md.monthValue = bundle.getInt("month");
                md.monthString = getMonthString(md.monthValue);
                md.count = bundle.getInt("count");
                byte[] tn = bundle.getByteArray("thumbnail");
                md.thumbnail =  BitmapFactory.decodeByteArray(tn, 0, tn.length);
                photos.add(md);
                iAdapter.notifyDataSetChanged();
            } catch (Exception e) {
                // TODO Auto-generated catch block
                Log.d("Debug", "handler error occurs in UI Handler");
            }
        }
    }
};

请注意,我没有包含所有代码,只是我觉得相关的部分。

之前的活动成功地以相同的方式操作消息,为什么不以第二个活动?

我知道主UI线程已经有一个looper设置,因此你不必创建一个。是否仍然适用于任何后续的活动?

2 个答案:

答案 0 :(得分:7)

使用Handler的dispatchMessage方法而不是sendMessage来解决问题。

答案 1 :(得分:6)

我遇到了和你在这里遇到的问题一样的问题。经过两天的奋斗,我找到了解决问题的方法。 很简单。只需在更新任何UI之前在handlerMessage()中添加this.obtainMessage()。 这样做之后,现在一切都会好起来的。

我想这是因为我们在UI线程和后台线程之间进行了太快的通信,在这种情况下Android系统无法自行调度Msg。当我们强制Android这样做时,问题就解决了。 我不确定,这只是猜测。 我希望它可以帮到你。