31

有一种服务可以监听一些声音。如果语音与字符串匹配,则会在服务对象中调用某个方法。

public class SpeechActivationService extends Service {

     public static Intent makeStartServiceIntent(Context pContext){    

         return new Intent(pContext, SpeechActivationService.class);
     }

     //...

     public void onMatch(){
         Log.d(TAG, "voice matches word");
     }

     //...
}

这就是我在活动中启动服务的方式:

Intent i = SpeechActivationService.makeStartServiceIntent(this);
startService(i);

从这个服务方法中,我如何调用驻留在活动对象中的方法?我不想从活动访问到服务,而是从服务到活动。我已经阅读了有关处理程序和广播者的信息,但找不到/理解任何示例。有任何想法吗?

4

4 回答 4

65

假设您的 Service 和 Activity 在同一个包中(即同一个应用程序),您可以使用 LocalBroadcastManager 如下:

在您的服务中:

// Send an Intent with an action named "my-event". 
private void sendMessage() {
  Intent intent = new Intent("my-event");
  // add data
  intent.putExtra("message", "data");
  LocalBroadcastManager.getInstance(this).sendBroadcast(intent);
}

在您的活动中:

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

  // Register mMessageReceiver to receive messages.
  LocalBroadcastManager.getInstance(this).registerReceiver(mMessageReceiver,
      new IntentFilter("my-event"));
}

// handler for received Intents for the "my-event" event 
private BroadcastReceiver mMessageReceiver = new BroadcastReceiver() {
  @Override
  public void onReceive(Context context, Intent intent) {
    // Extract data included in the Intent
    String message = intent.getStringExtra("message");
    Log.d("receiver", "Got message: " + message);
  }
};

@Override
protected void onPause() {
  // Unregister since the activity is not visible
  LocalBroadcastManager.getInstance(this).unregisterReceiver(mMessageReceiver);
  super.onPause();
}

来自@Ascorbin 链接的第 7.3 节:http ://www.vogella.com/tutorials/AndroidBroadcastReceiver/article.html#ownreceiver_localbroadcastmanager

于 2014-08-07T00:42:29.940 回答
11

我会在 Activity 中注册一个 BroadcastReceiver 并从服务向它发送一个 Intent。请参阅本教程:http ://www.vogella.com/articles/AndroidBroadcastReceiver/article.html 它可能看起来有点长,但你还是想学习如何使用它们;)

于 2013-02-15T14:24:00.500 回答
2

有许多不同的方法可以实现这一目标。其中之一使用HandlerMessanger类。该方法的思想是将Handler对象从Activityto 传递给Service。每次Service想要调用它的某个方法时,Activity它只是发送一个MessageActivity以某种方式处理它。

活动:

public class MyActivity extends Activity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        final Handler handler = new Handler() {
            @Override
            public void handleMessage(Message msg) {
                showToast(msg.what);
            }
        };

        final Intent intent = new Intent(this, MyService.class);
        final Messenger messenger = new Messenger(handler);

        intent.putExtra("messenger", messenger);
        startService(intent);
    }

    private void showToast(int messageId) {
        Toast.makeText(this, "Message  " + messageId, Toast.LENGTH_SHORT).show();
    }
}

服务:

public class MyService extends Service {
    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        if (intent != null) {
            final Messenger messenger = (Messenger) intent.getParcelableExtra("messenger");
            final Message message = Message.obtain(null, 1234);

            try {
                messenger.send(message);
            } catch (RemoteException exception) {
                exception.printStackTrace();
            }
        }

        return START_NOT_STICKY;
    }

    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }
}
于 2013-02-15T14:24:41.577 回答
0

经过一些研究,我发现在我的情况下发送和接收广播的时间如下。我在同一过程中有服务。

sendBroadcast(如果两个组件在同一进程中,则不推荐)34 秒

LocalBroadcastManager.getInstance(this).sendBroadcast(intent); 接近 30 秒

使用 AIDL 和 RemoteCallbackList 实现 将适用于相同的进程或不同的进程

为您服务

public final RemoteCallbackList<ICallBackAidl> mDMCallbacks = new RemoteCallbackList<ICallBackAidl>();

public void registerDMCallback(ICallBackAidl cb) {
    Logger.d(LOG_TAG, "registerDMCallback " + cb);
    if (cb != null)
        mDMCallbacks.register(cb);
}

当您需要从服务中调用 Application/Acitvity 中的方法时

public void callMehodsInApplication() {
    final int N = mDMCallbacks.beginBroadcast();
    for (int i = 0; i < N; i++) {
        try {
            mDMCallbacks.getBroadcastItem(i).method1();
        } catch (RemoteException e) {
            e.printStackTrace();
        }
    }
    mDMCallbacks.finishBroadcast();
}

在您的类中扩展应用程序或活动。

private ISyncmlServiceDMCallback mCallback = new ISyncmlServiceDMCallback.Stub() {
 // Implement callback methods here
  public void method1() {
       // Service can call this method
  }
}

 public void onServiceConnected(ComponentName name, IBinder service) {   
        svc.LocalBinder binder = (svc.LocalBinder) service;
        mSvc = binder.getService();
        mSvc.registerDMCallback(mCallback);
 }

以这种方式调用几乎是从同一进程广播和接收的瞬间

于 2016-04-29T22:02:45.273 回答