Пользовательский интерфейс активности обновления Android от службы

У меня есть служба, которая постоянно проверяет новую задачу. Если есть новая задача, я хочу обновить пользовательский интерфейс активности, чтобы показать эту информацию. Я нашел https://github.com/commonsguy/cw-andtutorials/tree/master/18-LocalService/ этот пример. Это хороший подход? Любые другие примеры?

Спасибо.

Ответ 1

См. ниже мой первоначальный ответ - этот шаблон работал хорошо, но в последнее время я начал использовать другой подход к обмену службами/действиями:

  • Используйте связанный сервис, который позволяет Управлению получить прямой ссылку на Службу, что позволяет прямое обращение к ней, а чем использование намерений.
  • Используйте RxJava для выполнения асинхронных операций.

  • Если службе необходимо продолжить фоновые операции, даже если нет Активность запущена, также запустите службу из приложения класс, чтобы он не останавливался при несвязании.

Преимущества, которые я нашел в этом подходе по сравнению с методом startService()/LocalBroadcast,

  • Нет необходимости в объектах данных для реализации Parcelable - это особенно важно для меня, поскольку теперь я использую код между Android и iOS (используя RoboVM).
  • RxJava предоставляет готовое (и кросс-платформенное) планирование и легкий состав последовательных асинхронных операций.
  • Это должно быть более эффективным, чем использование LocalBroadcast, хотя накладные расходы на использование RxJava могут перевесить это.

Пример кода примера. Сначала служба:

public class AndroidBmService extends Service implements BmService {

    private static final int PRESSURE_RATE = 500000;   // microseconds between pressure updates
    private SensorManager sensorManager;
    private SensorEventListener pressureListener;
    private ObservableEmitter<Float> pressureObserver;
    private Observable<Float> pressureObservable;

    public class LocalBinder extends Binder {
        public AndroidBmService getService() {
            return AndroidBmService.this;
        }
    }

    private IBinder binder = new LocalBinder();

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        logMsg("Service bound");
        return binder;
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        return START_NOT_STICKY;
    }

    @Override
    public void onCreate() {
        super.onCreate();

        sensorManager = (SensorManager)getSystemService(SENSOR_SERVICE);
        Sensor pressureSensor = sensorManager.getDefaultSensor(Sensor.TYPE_PRESSURE);
        if(pressureSensor != null)
            sensorManager.registerListener(pressureListener = new SensorEventListener() {
                @Override
                public void onSensorChanged(SensorEvent event) {
                    if(pressureObserver != null) {
                        float lastPressure = event.values[0];
                        float lastPressureAltitude = (float)((1 - Math.pow(lastPressure / 1013.25, 0.190284)) * 145366.45);
                        pressureObserver.onNext(lastPressureAltitude);
                    }
                }

                @Override
                public void onAccuracyChanged(Sensor sensor, int accuracy) {

                }
            }, pressureSensor, PRESSURE_RATE);
    }

    @Override
    public Observable<Float> observePressure() {
        if(pressureObservable == null) {
            pressureObservable = Observable.create(emitter -> pressureObserver = emitter);
            pressureObservable = pressureObservable.share();
        }
         return pressureObservable;
    }

    @Override
    public void onDestroy() {
        if(pressureListener != null)
            sensorManager.unregisterListener(pressureListener);
    }
} 

И активность, которая связывается с сервисом и получает обновления абсолютной высоты:

public class TestActivity extends AppCompatActivity {

    private ContentTestBinding binding;
    private ServiceConnection serviceConnection;
    private AndroidBmService service;
    private Disposable disposable;

    @Override
    protected void onDestroy() {
        if(disposable != null)
            disposable.dispose();
        unbindService(serviceConnection);
        super.onDestroy();
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        binding = DataBindingUtil.setContentView(this, R.layout.content_test);
        serviceConnection = new ServiceConnection() {
            @Override
            public void onServiceConnected(ComponentName componentName, IBinder iBinder) {
                logMsg("BlueMAX service bound");
                service = ((AndroidBmService.LocalBinder)iBinder).getService();
                disposable = service.observePressure()
                    .observeOn(AndroidSchedulers.mainThread())
                    .subscribe(altitude ->
                        binding.altitude.setText(
                            String.format(Locale.US,
                                "Pressure Altitude %d feet",
                                altitude.intValue())));
            }

            @Override
            public void onServiceDisconnected(ComponentName componentName) {
                logMsg("Service disconnected");
            }
        };
        bindService(new Intent(
            this, AndroidBmService.class),
            serviceConnection, BIND_AUTO_CREATE);
    }
}

Макет для этой операции:

<?xml version="1.0" encoding="utf-8"?>
<layout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    >
    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        tools:context="com.controlj.mfgtest.TestActivity">

        <TextView
            tools:text="Pressure"
            android:id="@+id/altitude"
            android:gravity="center_horizontal"
            android:layout_gravity="center_vertical"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"/>

    </LinearLayout>
</layout>

Если служба должна запускаться в фоновом режиме без связанного действия, ее можно запустить из класса Application, а также в OnCreate(), используя Context#startService().


Мой оригинальный ответ (с 2013 года):

В вашей службе: (используя COPA как услугу в примере ниже).

Используйте LocalBroadCastManager. В своем сервисе onCreate настройте вещатель:

broadcaster = LocalBroadcastManager.getInstance(this);

Если вы хотите уведомить пользовательский интерфейс:

static final public String COPA_RESULT = "com.controlj.copame.backend.COPAService.REQUEST_PROCESSED";

static final public String COPA_MESSAGE = "com.controlj.copame.backend.COPAService.COPA_MSG";

public void sendResult(String message) {
    Intent intent = new Intent(COPA_RESULT);
    if(message != null)
        intent.putExtra(COPA_MESSAGE, message);
    broadcaster.sendBroadcast(intent);
}

В вашей деятельности:

Создать прослушиватель onCreate:

public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    super.setContentView(R.layout.copa);
    receiver = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {
            String s = intent.getStringExtra(COPAService.COPA_MESSAGE);
            // do something here.
        }
    };
}

и зарегистрируйте его в onStart:

@Override
protected void onStart() {
    super.onStart();
    LocalBroadcastManager.getInstance(this).registerReceiver((receiver), 
        new IntentFilter(COPAService.COPA_RESULT)
    );
}

@Override
protected void onStop() {
    LocalBroadcastManager.getInstance(this).unregisterReceiver(receiver);
    super.onStop();
}

Ответ 2

для меня самым простым решением было отправить широковещательную рассылку в активную запись oncreate я и определить такую ​​широковещательную передачу (updateUIReciver определяется как экземпляр класса):

 IntentFilter filter = new IntentFilter();

 filter.addAction("com.hello.action"); 

 updateUIReciver = new BroadcastReceiver() {

            @Override
            public void onReceive(Context context, Intent intent) {
                //UI update here

            }
        };
 registerReceiver(updateUIReciver,filter);

И из службы вы отправляете намерение следующим образом:

Intent local = new Intent();

local.setAction("com.hello.action");

this.sendBroadcast(local);

не забывайте отменить регистрацию в операции по уничтожению:

unregisterReceiver(updateUIReciver);

Ответ 3

Я бы рекомендовал проверить Otto, EventBus, специально предназначенный для Android. Ваша активность/пользовательский интерфейс может прослушивать события, отправленные на шине из Сервиса, и отделяться от бэкэнда.

Ответ 4

Я бы использовал связанный сервис, чтобы сделать это и связался с ним, выполнив прослушиватель в моей деятельности. Поэтому, если ваше приложение реализует myServiceListener, вы можете зарегистрировать его как слушателя в своей службе после того, как вы связались с ним, вызовите listener.onUpdateUI из вашей связанной службы и обновите свой интерфейс там!

Ответ 5

Решение Clyde работает, но это широковещательная передача, которая, я уверен, будет менее эффективной, чем вызов метода напрямую. Я мог ошибаться, но я думаю, что трансляции предназначены для обмена сообщениями между приложениями.

Я предполагаю, что вы уже знаете, как связать службу с Activity. Я делаю что-то вроде кода ниже, чтобы справиться с такой проблемой:

class MyService extends Service {
    MyFragment mMyFragment = null;
    MyFragment mMyOtherFragment = null;

    private void networkLoop() {
        ...

        //received new data for list.
        if(myFragment != null)
            myFragment.updateList();
        }

        ...

        //received new data for textView
        if(myFragment !=null)
            myFragment.updateText();

        ...

        //received new data for textView
        if(myOtherFragment !=null)
            myOtherFragment.updateSomething();

        ...
    }
}


class MyFragment extends Fragment {

    public void onResume() {
        super.onResume()
        //Assuming your activity bound to your service
        getActivity().mMyService.mMyFragment=this;
    }

    public void onPause() {
        super.onPause()
        //Assuming your activity bound to your service
        getActivity().mMyService.mMyFragment=null;
    }

    public void updateList() {
        runOnUiThread(new Runnable() {
            public void run() {
                //Update the list.
            }
        });
    }

    public void updateText() {
       //as above
    }
}

class MyOtherFragment extends Fragment {
             public void onResume() {
        super.onResume()
        //Assuming your activity bound to your service
        getActivity().mMyService.mMyOtherFragment=this;
    }

    public void onPause() {
        super.onPause()
        //Assuming your activity bound to your service
        getActivity().mMyService.mMyOtherFragment=null;
    }

    public void updateSomething() {//etc... }
}

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

Ответ 6

Callback from service to activity to update UI.
ResultReceiver receiver = new ResultReceiver(new Handler()) {
    protected void onReceiveResult(int resultCode, Bundle resultData) {
        //process results or update UI
    }
}

Intent instructionServiceIntent = new Intent(context, InstructionService.class);
instructionServiceIntent.putExtra("receiver", receiver);
context.startService(instructionServiceIntent);

Ответ 7

Мое решение может быть не самым чистым, но оно должно работать без проблем. Логика состоит в том, чтобы создать статическую переменную для хранения ваших данных на Service и обновить ваш просмотр каждую секунду на Activity.

Скажем, что у вас есть String на Service, который вы хотите отправить в TextView на Activity. Это должно выглядеть так.

Ваша услуга:

public class TestService extends Service {
    public static String myString = "";
    // Do some stuff with myString

Ваша активность:

public class TestActivity extends Activity {
    TextView tv;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        tv = new TextView(this);
        setContentView(tv);
        update();
        Thread t = new Thread() {
            @Override
            public void run() {
                try {
                    while (!isInterrupted()) {
                        Thread.sleep(1000);
                        runOnUiThread(new Runnable() {
                            @Override
                            public void run() {
                                update();
                            }
                        });
                    }
                } catch (InterruptedException ignored) {}
            }
        };
        t.start();
        startService(new Intent(this, TestService.class));
    }
    private void update() {
        // update your interface here
        tv.setText(TestService.myString);
    }
}