Как я могу обновить информацию в активности Android из фоновой службы
-
20-09-2019 - |
Вопрос
Я пытаюсь создать простое приложение для Android, у которого есть ActivityList информации, при запуске приложения я планирую запустить Службу, которая будет постоянно вычислять данные (они будут меняться), и я хочу, чтобы ActivityList был синхронизирован с данными, которые служба вычисляет в течение срока службы приложения.
Как я могу настроить свою Активность на прослушивание Сервиса?Это лучший способ подойти к этой проблеме?
Например, если вы представите список цен на акции - данные будут регулярно меняться и должны быть синхронизированы с (в моем случае) Сервисом, который постоянно вычисляет / извлекает данные.
Заранее спасибо
Решение
Как я могу настроить свою деятельность таким образом, чтобы она заключалась в прослушивании сервиса?Это лучший способ подойти к этой проблеме?
На мой взгляд, у вас есть три основных варианта:
Опрос.В
Activity
периодически спрашиваетService
для получения последних данных.ИМХО, этот вариант отстой, но это, безусловно, возможно.Обратные вызовы.Согласно ответу Джакса,
Activity
регистрирует объект обратного вызова ("observer") с помощьюService
.ВService
вызывает метод при обратном вызове при изменении данных, который, в свою очередь, обновляет пользовательский интерфейс.Вы можете увидеть пример использования этого сService
здесь.Трансляция
Intents
.ВService
транслируетIntent
черезsendBroadcast()
при изменении данных.ВActivity
регистрируетBroadcastReceiver
используяregisterReceiver()
, и чтоBroadcastReceiver
получает уведомление о входящей широковещательной передаче.Это запускаетActivity
чтобы загрузить последние данные изService
, или , возможно , просто для того , чтобы получить последние данные от статистов в трансляцииIntent
.Вы можете увидеть пример использования этой техники сService
здесь.
Другие советы
Это звучит как хороший кандидат для шаблона Observer .По сути, ваша активность (Наблюдатель) будет регистрироваться в фоновом сервисе (Observable), и вы сможете передавать или извлекать данные из своей активности.Вашим Наблюдателем в этом случае будет ваша Деятельность, а Наблюдаемым - ваше Служение.
Если вы ничего не знаете о шаблонах проектирования, купите "Head First Design Patterns", она легко читается и полна отличной информации.
PS:Я читаю это сейчас.
Мне действительно, очень интересно, почему никто не упомянул простой подход с использованием EventBus из любой библиотеки.Это, конечно, если вы не используете RX.Мой личный фаворит - EventBus от GreenRobot.https://github.com/greenrobot/EventBus
Всего с парой строк кода и без каких-либо интерфейсов.Запустите событие и слушайте его везде, где захотите.Он развязан, потокобезопасен и не приведет к сбою вашего приложения.
Вам нужно использовать 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);
}
}
У вас будет запущен фоновый поток, который вычисляет изменения в списке.Теперь этому потоку нужна возможность уведомлять графический интерфейс пользователя о том, что список был обновлен.
Вы можете использовать какой-то Адаптер массива чтобы получить данные в ListView.У ArrayAdapter есть метод, который называется adpater.notifyDataSetChanged() каждый раз, когда вы вызываете этот метод, адаптер будет видеть, что соответствующие данные изменились, а затем уведомлять listview о том, что он должен обновиться при следующей возможности.