我正在尝试创建一个具有 ActivityList 信息的简单 Android 应用程序,当应用程序启动时,我计划启动一个服务,该服务将不断计算数据(它将不断变化)并且我希望 ActivityList 与服务在应用程序生命周期内计算的数据。
如何设置我的活动以收听服务?这是解决这个问题的最佳方法吗?
例如,如果您想象一个股票价格列表 - 数据将定期更改,并且需要与(在我的情况下)不断计算/获取数据的服务同步。
提前致谢
我正在尝试创建一个具有 ActivityList 信息的简单 Android 应用程序,当应用程序启动时,我计划启动一个服务,该服务将不断计算数据(它将不断变化)并且我希望 ActivityList 与服务在应用程序生命周期内计算的数据。
如何设置我的活动以收听服务?这是解决这个问题的最佳方法吗?
例如,如果您想象一个股票价格列表 - 数据将定期更改,并且需要与(在我的情况下)不断计算/获取数据的服务同步。
提前致谢
如何设置我的活动以收听服务?这是解决这个问题的最佳方法吗?
在我看来,您有三个主要选择:
轮询。Activity
定期询问最新Service
数据。恕我直言,这个选项很糟糕,但它肯定是可能的。
回调。根据 jax 的回答,Activity
注册一个回调对象(“观察者”)与Service
. 当Service
数据发生变化时,它会调用回调上的一个方法,从而更新 UI。Service
您可以在此处查看将其与 a 一起使用的示例。
广播Intents
。Service
广播一个数据变化的Intent
通路sendBroadcast()
。Activity
注册一个BroadcastReceiver
using ,registerReceiver()
并BroadcastReceiver
通知传入广播。这会触发从广播Activity
中加载最新数据Service
,或者可能只是从广播中的附加内容中获取最新数据Intent
。Service
您可以在此处查看使用该技术的示例。
这听起来像是观察者模式的一个很好的候选者。基本上,您的活动(The Observer)将在后台服务(The Observable)中注册自己,您可以从您的活动中推送或拉取数据。在这种情况下,您的 Observer 将是您的 Activity,而 Observable 将是您的 Service。
如果您对设计模式一无所知,请购买“Head First Design Patterns”,它易于阅读并且信息量很大。
PS:我现在正在读。
我真的很想知道为什么没有人提到任何库使用 EventBus 的简单方法。这当然是在您不使用 RX 的情况下。我个人最喜欢的是 GreenRobot 的 EventBus。 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);
}
}
您将运行一个后台线程来计算列表中的更改。该线程现在需要通知 GUI 列表已更新。
您可以使用某种ArrayAdapter将数据放入 ListView。ArrayAdapter 有一个名为adpater.notifyDataSetChanged()的方法,每次调用此方法时,适配器都会看到相应的数据已更改,然后通知列表视图它应该在下一次可能更新时进行更新。