Как обновить информацию в Android-активности из фоновой службы
Я пытаюсь создать простое приложение для Android, которое имеет ActivityList информации, когда приложение запускается, я планирую запустить службу, которая будет постоянно вычислять данные (она будет меняться), и я хочу, чтобы ActivityList был в синхронизации с данными, которые служба вычисляет для жизни приложения.
Как настроить свою активность для прослушивания Сервиса? Это лучший способ подойти к этой проблеме?
Например, если вы представляете список цен на акции - данные будут регулярно меняться и должны быть синхронизированы с (в моем случае) Службой, которая постоянно вычисляет/извлекает данные.
Заранее спасибо
Ответы
Ответ 1
Как я могу настроить свою активность слушая Службу? Является ли это лучший способ подойти к этой проблеме?
У вас есть три основных варианта, как я вижу:
- Опрос
. Activity
периодически запрашивает Service
для последних данных. ИМХО, этот вариант отстой, но это, безусловно, возможно.
-
Callbacks. В ответе на jax Activity
регистрирует объект обратного вызова ( "наблюдатель" ) с помощью Service
. Service
вызывает метод обратного вызова при изменении данных, который, в свою очередь, обновляет пользовательский интерфейс. Вы можете увидеть пример использования этого с Service
здесь.
-
Трансляция Intents
. Service
транслирует Intent
через sendBroadcast()
при изменении данных. Activity
регистрирует BroadcastReceiver
с помощью registerReceiver()
и что BroadcastReceiver
уведомляется о входящей широковещательной передаче. Это заставляет Activity
загружать последние данные из Service
или, возможно, просто для того, чтобы получить последние данные из дополнительных функций в широковещательной передаче Intent
. Вы можете увидеть пример использования этой технологии с Service
здесь.
Ответ 2
Этот звук является хорошим кандидатом на шаблон Observer. В основном ваша деятельность (The Observer) зарегистрируется в фоновом сервисе (Observable), и вы можете выталкивать или извлекать данные из вашей Activity. Ваш наблюдатель в этом случае будет вашей деятельностью, а Наблюдателем будет ваша Служба.
Если вы ничего не знаете о Design Patterns, купите "Head First Design Patterns", его легко прочитать и полон большой информацией.
PS: Я читаю его сейчас.
Ответ 3
У вас будет фоновый поток, который вычисляет изменения в списке. Этот поток теперь нуждается в возможности уведомить GUI о том, что список обновлен.
Вы можете использовать ArrayAdapter, чтобы получить данные в ListView. ArrayAdapter имеет метод adpater.notifyDataSetChanged() каждый раз, когда вы вызываете этот метод, адаптер увидит, что соответствующие данные изменились, а затем уведомили список, который он должен обновить при следующей возможности.
Ответ 4
Вам необходимо использовать bindService(), чтобы связать действие с запущенным сервисом и связаться с ним.
public class BindingActivity extends Activity {
YourService mService;
boolean mBound = false;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
}
@Override
protected void onStart() {
super.onStart();
// Bind to Your Service
Intent intent = new Intent(this, YourService.class);
bindService(intent, mConnection, Context.BIND_AUTO_CREATE);
}
@Override
protected void onStop() {
super.onStop();
// Unbind from the service
if (mBound) {
unbindService(mConnection);
mBound = false;
}
}
/** Called when a button is clicked (the button in the layout file attaches to
* this method with the android:onClick attribute) */
public void onButtonClick(View v) {
if (mBound) {
// Call a method from your Service.
// However, if this call were something that might hang, then this request should
// occur in a separate thread to avoid slowing down the activity performance.
int num = mService.getRandomNumber();
Toast.makeText(this, "number: " + num, Toast.LENGTH_SHORT).show();
}
}
/** Defines callbacks for service binding, passed to bindService() */
private ServiceConnection mConnection = new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName className,
IBinder service) {
// We've bound to the running Service, cast the IBinder and get instance
LocalBinder binder = (LocalBinder) service;
mService = binder.getService();
mBound = true;
}
@Override
public void onServiceDisconnected(ComponentName arg0) {
mBound = false;
}
};
}
и ваш сервис вроде:
public class LocalService extends Service {
// Binder given to clients
private final IBinder mBinder = new LocalBinder();
// Random number generator
private final Random mGenerator = new Random();
/**
* Class used for the client Binder. Because we know this service always
* runs in the same process as its clients, we don't need to deal with IPC.
*/
public class LocalBinder extends Binder {
LocalService getService() {
// Return this instance of LocalService so clients can call public methods
return LocalService.this;
}
}
@Override
public IBinder onBind(Intent intent) {
return mBinder;
}
/** method for clients */
public int getRandomNumber() {
return mGenerator.nextInt(100);
}
}
Ответ 5
Я действительно очень удивляюсь, почему никто не упомянул простой подход с использованием EventBus по любой библиотеке. Это, конечно, если вы не используете RX.
Мой личный фаворит - EventBus от GreenRobot.
https://github.com/greenrobot/EventBus
Только с несколькими строками кода и без интерфейсов. Огоньте событие и слушайте его, где хотите. Он развязан, он потокобезопасен, и он не будет разбивать ваше приложение.