3

我检查了这个问题,但它似乎没有回答我的问题,这涉及的要少得多。我从主流程的菜单项中调用 IntentService。它目前只是一个骨架,它在 onHandleIntent() 中放置一个 Toast,最终应该在处理结束时短暂出现以表示它已完成。然而,它永远保留在屏幕上(即使应用程序停止)。我使用模拟器和我的 Galaxy S2 进行了测试,结果相同。有人可以指出我正确的方向吗?

这是服务代码:

 package com.enborne.spine;

 import android.app.IntentService;
 import android.content.Intent;
 import android.util.Log;
 import android.widget.Toast;

 public class SaveFile extends IntentService {
     private final String TAG = "SaveFile";

     public SaveFile() {
         super("SaveFile");
     }

     @Override
     public void onCreate() {
         super.onCreate();
         Log.d(TAG, "Service Started.. ");
     }

     @Override
     public void onDestroy() {
         super.onDestroy();
         Log.d(TAG, "Service Destroyed.. ");
     }

     @Override
     protected void onHandleIntent(Intent intent) {
         // TODO Auto-generated method stub
         Log.d(TAG, "HandleIntent");
         // File saved
         Toast.makeText(this, "File has been saved", Toast.LENGTH_SHORT).show();
     }
 }

(onCreate 和 onDestroy 覆盖只是暂时用于登录 Eclipse,所以我可以看到发生了什么。)

因此,我从活动中的菜单调用它(与上下文混淆只是因为服务没有启动 - 有些白痴忘记将它包含在清单中!):

@Override
public boolean onOptionsItemSelected(MenuItem item) {
    Intent i;
    switch (item.getItemId()) {
    case R.id.chart:
        i = new Intent(getBaseContext(), Chart.class);
        i.putExtra(KEY_CENTRE_LAT, mCentreLat); // centre coordinate
        i.putExtra(KEY_CENTRE_LONG, mCentreLong); // centre coordinate
        i.putExtra(KEY_RADIUS, mRadius); // effective radius
        startActivity(i);
        break;

    case R.id.save:
        Log.d("SaveFile", "Starting service...");
        //      i = new Intent(getBaseContext(), SaveFile.class);
        i = new Intent(this, SaveFile.class);
        startService(i);
        break;
    }
    return true;
}

编辑:我找到了这篇文章,它可以解释这个问题。引用一句话:

但是,您只能在主 GUI 线程中使用 Toast,否则您会遇到 Toast 消息在一段时间后不会消失的问题(因为主 GUI 上下文不知道在单独的线程上下文中使用的 Toast 消息的任何信息)。

4

2 回答 2

8

在我的编辑中链接到的文章将我带到了这个博客,它解决了我的问题。我没有在其他地方看到您不能直接从服务中发出祝酒词。所以我的 onHandleIntent 现在看起来像这样:

private Handler handler;

@Override
protected void onHandleIntent(Intent intent)
{
    Log.d(TAG, "onHandleIntent");

    /* Stripped out some code here that I added later to keep it similar to
     * my example above
     */

    handler.post(new Runnable()
    {  
//      @Override
        public void run()
        {
            Toast.makeText(getApplicationContext(), "File has been saved", 
                Toast.LENGTH_SHORT).show();
        }
    });    // Display toast and exit
}

(如果有人能解释为什么我必须在 run() 上注释掉 @Override 以避免错误,我将不胜感激。我的应用程序中还有其他几个地方必须这样做。)

编辑:我刚刚发现这篇文章几乎是一回事。我不知道我以前是怎么错过的。

于 2012-09-13T19:39:34.730 回答
2

onHandleIntent 在工作线程中执行。

我会尝试从 UI 线程执行 Toast.show() 。

于 2012-09-13T19:40:30.173 回答