Как отображать 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();
}
});