Как отображать Toast из сервиса после завершения основной операции?

ОБНОВЛЕНИЕ: Я не согласен с тем, что это дубликат, потому что я ищу способ выйти из основного приложения и по-прежнему показывать Toast из службы.

В очень простое тестовое приложение У меня есть 2 кнопки:

screenshot

При нажатии любой из кнопок будет запущена служба с соответствующей строкой действия ( "open" или "flash" ) -

OpenActivity.java:

public class OpenActivity extends Activity {
    private Intent mServiceIntent;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_open);
        mServiceIntent = new Intent(this, RegionService.class);
   }

    public void openCar(View v) {
        mServiceIntent.setAction("open");
        startService(mServiceIntent);
    }

RegionService.java:

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

    @Override
    protected void onHandleIntent(Intent intent) {
        Log.d(TAG, "Received an intent: " + intent);
        String action = intent.getAction();
        Log.d(TAG, "Received an action: " + action);

        if(action.equals("open")) {
            Toast.makeText(this, 
                    getString(R.string.car_opened), 
                    Toast.LENGTH_SHORT).show();
        } 

К сожалению, мое приложение вылетает с:

D/RegionService(24506): Received an intent: Intent { act=open cmp=de.afarber.mynotification/.RegionService }

D/RegionService(24506): Received an action: open

W/MessageQueue(24506): Handler (android.os.Handler) {422768a8} sending message to a Handler on a dead thread
W/MessageQueue(24506): java.lang.RuntimeException: Handler (android.os.Handler) {422768a8} sending message to a Handler on a dead thread
W/MessageQueue(24506):  at android.os.MessageQueue.enqueueMessage(MessageQueue.java:320)
W/MessageQueue(24506):  at android.os.Handler.enqueueMessage(Handler.java:626)
W/MessageQueue(24506):  at android.os.Handler.sendMessageAtTime(Handler.java:595)
W/MessageQueue(24506):  at android.os.Handler.sendMessageDelayed(Handler.java:566)
W/MessageQueue(24506):  at android.os.Handler.post(Handler.java:326)
W/MessageQueue(24506):  at android.widget.Toast$TN.hide(Toast.java:370)
W/MessageQueue(24506):  at android.app.ITransientNotification$Stub.onTransact(ITransientNotification.java:54)
W/MessageQueue(24506):  at android.os.Binder.execTransact(Binder.java:412)
W/MessageQueue(24506):  at dalvik.system.NativeStart.run(Native Method)

Будучи новичком по программированию на Android, я задаюсь вопросом Как правильно отобразить Toast из службы?

Я думаю, что я уже видел Тосты в Android Home (т.е. на экране устройства не было активности и все еще были тосты).

Мой фон: Я хотел бы отслеживать маяковое устройство с моей службы и показывать текстовые тосты - даже когда мое приложение было закрыто.

Ответ 1

OnHandleIntent будет выполняться в разном Thread поэтому вы показываете Toast в потоке, который не разрешен в android

поэтому измените свой код следующим образом

Handler handler = new Handler(Looper.getMainLooper());
handler.post(new Runnable() {

    @Override
    public void run() {
         Toast.makeText(getApplicationContext(), 
                       getString(R.string.car_opened), 
                       Toast.LENGTH_SHORT).show();              
    }
});

Из мертвой нити в сервисе

IntentService создаст поток для обработки нового намерения и немедленно прекратит его сразу после выполнения задачи. Таким образом, тост не будет контролироваться мертвой нитью.

Вы должны увидеть некоторые исключения в консоли, когда на экране появится тост.

Ответ 2

У IntentService есть несколько ограничений:

Он не может напрямую взаимодействовать с вашим пользовательским интерфейсом. Положить результаты в пользовательском интерфейсе, вы должны отправить их на мероприятие.

Все происходит в фоновом потоке, а не в потоке пользовательского интерфейса, поэтому вам нужен другой способ, как показано ниже:

@Override 
public void onCreate() { 
    super.onCreate(); 
    mHandler = new Handler(); 
} 

@Override 
protected void onHandleIntent(Intent intent) {
    mHandler.post(new Runnable() {            
        @Override 
        public void run() { 
            Toast.makeText(MyIntentService.this, "Hello Toast!", Toast.LENGTH_LONG).show();                
        } 
    }); 
} 

Источник: Тост, созданный в IntentService, никогда не уходит

Ответ 3

OnHandleIntent вызывается в фоновом потоке, и любая попытка коснуться пользовательского интерфейса приведет к сбою. Используйте Handler для публикации Runnable в потоке пользовательского интерфейса, чтобы показать ваш тост.

private class MyRunnable implements Runnable {

   final int mTextId = -1; 
   final Context mContext;
   public MyRunnable(Context c, int textId) {
       mTextId = textId;
       mContext = c;
   }

   @Override
   public void run() {
       Toast.makeText(mContext, 
           getString(mTextId), 
           Toast.LENGTH_SHORT).show();             
    }
}

   Handler handler = new Handler();
   handler.post(new MyRunnable(this, R.string.car_opened));

Ответ 4

используйте следующий код:

  runOnUiThread(new Runnable(){
     public void run() {
          // UI code goes here
     }
    });

Ответ 5

Эта проблема из-за того, что не запускается Toast из main_thread, и чтобы преодолеть это, при создании Activity сохраняйте контекст из метода onCreate():

public static Context ctx;

// the method responsible for running the MainActivity
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        ctx = this;
}

то в службе добавьте обработчик и запустите поток из него (поскольку обработчик выполняется через главный поток):

    Handler handler = new Handler(Looper.getMainLooper());
    handler.post(new Runnable() {
        @Override
        public void run() {
            Toast.makeText(OpenActivity.ctx, getString(R.string.car_opened),
                    Toast.LENGTH_SHORT).show();
        }
    });