Как я могу обновить информацию в Android Activity из фоновой службы - PullRequest
39 голосов
/ 18 марта 2010

Я пытаюсь создать простое приложение для Android, которое имеет список ActivityList, при запуске приложения я планирую запустить службу, которая будет постоянно рассчитывать данные (она будет изменяться), и я хочу, чтобы ActivityList синхронно с данными, которые служба рассчитывает для жизни приложения.

Как мне настроить свою активность на прослушивание Сервиса? Это лучший способ решить эту проблему?

Например, если вы представляете список цен на акции - данные будут регулярно меняться и должны синхронизироваться со службой (в моем случае), которая постоянно вычисляет / извлекает данные.

Заранее спасибо

Ответы [ 5 ]

94 голосов
/ 18 марта 2010

Как я могу настроить свою активность на прослушивание Сервиса?Является ли это лучшим способом решения этой проблемы?

У меня есть три основных варианта, на мой взгляд:

  1. Опрос.Activity периодически запрашивает Service последние данные.ИМХО, эта опция отстой, но она, безусловно, возможна.

  2. Обратные вызовы.Согласно ответу jax, Activity регистрирует объект обратного вызова («наблюдатель») с помощью Service.Service вызывает метод обратного вызова при изменении данных, который, в свою очередь, обновляет пользовательский интерфейс.Вы можете увидеть пример использования этого с Service здесь .

  3. Broadcast Intents.Service передает Intent через sendBroadcast() при изменении данных.Activity регистрирует BroadcastReceiver, используя registerReceiver(), и BroadcastReceiver уведомляется о входящем широковещании.Это вызывает Activity для загрузки последних данных из Service, или, возможно, просто для получения последних данных из дополнительных данных в широковещательной передаче Intent.Вы можете увидеть пример использования этой техники с Service здесь .

4 голосов
/ 18 марта 2010

Звучит как хороший кандидат на Наблюдательский паттерн. По сути, ваша активность (The Observer) будет регистрироваться в фоновом сервисе (The Observable), и вы можете выталкивать или извлекать данные из своей деятельности. В этом случае ваш наблюдатель будет вашей деятельностью, а наблюдаемый - вашей службой.

Если вы ничего не знаете о шаблонах проектирования, купите «Шаблоны проектирования Head First», они легко читаются и содержат много полезной информации.

PS: сейчас читаю.

2 голосов
/ 09 марта 2017

Мне действительно очень интересно, почему никто не упомянул простой подход с использованием EventBus какой-либо библиотекой. Это, конечно, если вы не используете RX. Мой личный фаворит - EventBus от GreenRobot. https://github.com/greenrobot/EventBus

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

1 голос
/ 20 марта 2015

Вам необходимо использовать 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);
  }
}
0 голосов
/ 18 марта 2010

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

Вы можете использовать какой-то ArrayAdapter , чтобы получить данные в ListView. У ArrayAdapter есть метод с именем adpater.notifyDataSetChanged () каждый раз, когда вы вызываете этот метод, адаптер будет видеть, что соответствующие данные изменились, и затем уведомлять просмотр списка, что он должен обновить при следующей возможности.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...