如何将服务绑定到正确的活动实例?

时间:2012-07-21 23:24:30

标签: android android-service android-handler

我无法让我的应用程序正常运行以进行各种重新计算。

设计概述: 我的目标是一个Activity,它有两个按钮(Connect,Disconnect)和一个textview,显示从连接接收的数据。我编写了一个处理WebSocket连接的“MySocket”服务(Gottox的Java实现),当服务器从服务器接收内容时,该服务会向活动发布消息,然后活动会将此数据附加到textview上。该服务必须从活动接收连接/断开连接提示,并能够将消息传递给活动。 MySocket还需要作为前台服务运行,因此它会无限期地保持连接,并为用户提供粘性通知。

我在各个地方遇到了很多重大问题。首先,我注意到我可以启动另一个活动实例,而原始版本仍在更新中。我通过在清单中指定android:launchMode="singleInstance">来解决此问题。我遇到的问题是,当活动被销毁并创建一个新活动时,服务继续发布更新,但重点活动没有收到它们。如果原始活动被销毁并且服务处理程序没有更新,那么肯定会在活动被销毁后抛出空指针异常,但应用程序仍在运行。然后我尝试在我的onDestroy方法中运行unbindService(mConnection),但是它抛出了java.lang.IllegalArgumentException: Service not registered:该服务是消息某些应该是重点活动,但似乎没有收到它。我怀疑我的前台服务实施不好是罪魁祸首,但我真的不知道。

有问题的两个类是LONG,所以我删除了Logcat行,全局变量以及一些我确定无关的方法。我意识到这个体系结构是可怕的,但这与我开始工作的时间差不多。如果代码太难以阅读,我会很高兴有人建议一个替代结构,只要它达到了预期的结果。

活动类:

@Override
public void onCreate(Bundle savedInstanceState)
{
      ...
      mMessenger = new Messenger(mHandler);
      connect.setOnClickListener(new View.OnClickListener()
    {
        @Override
        public void onClick(View v)
        {
            messageSocket("ACTION", "CONNECT");
        }
    });

     @Override
    protected void onDestroy()
    {
      super.onDestroy();
      doUnbindService();
    }

    private ServiceConnection conn = new ServiceConnection() 
    {
    public void onServiceConnected(ComponentName className, IBinder binder) 
    {
        Log.d(debug, "Service Connected");
        messenger = new Messenger(binder);
        boundToService = true;
    }

    public void onServiceDisconnected(ComponentName className) 
    {
        Log.d(debug, "Service Disconnected");
        messenger = null;
        boundToService = false;
    }
    };

    void doBindService() 
    {
    final Intent intent = new Intent(this, MySocket.class);
    intent.putExtra("MESSENGER", messenger);
    Thread t = new Thread()
    {
        public void run()
        {
        boundToService =
                getApplicationContext().bindService(
                intent,
                conn,
                Context.BIND_AUTO_CREATE
            );
        }
    };
    t.start();
}

和MySocket类:

class IncomingHandler extends Handler 
{
    @Override
    public void onCreate() 
{...}

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        startForeground( id, showNotification() );
    connect();
        return START_STICKY;
}

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        startForeground( id, showNotification() );
        connect();
        return START_STICKY;
}


private Notification showNotification()
{
    nm = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
    CharSequence text = "WebSocket currently active";
    Notification notification = new Notification(R.drawable.ic_launcher,
            text, System.currentTimeMillis());
    PendingIntent contentIntent = PendingIntent.getActivity(this, 0,
            new Intent(this, SocketTester.class), 0);
    notification.setLatestEventInfo(this, "WebSocket Test", text,
            contentIntent);
    notification.flags = Notification.FLAG_FOREGROUND_SERVICE;
    return notification;
}
        @Override
    public IBinder onBind(Intent intent)
    {
        Bundle extras = intent.getExtras();
        Log.d(debug, "onBind called");
        if (intent.getExtras() != null)
        {   
            Log.d(debug, "Setting messenger");
            outMessenger = (Messenger) extras.get("MESSENGER");
        }
        return inMessenger.getBinder();
        }   

    @Override
    public void handleMessage(Message msg) 
            {           
        Bundle data = msg.getData();            
        Message backMsg = Message.obtain();
        Bundle bundle = new Bundle();
        backMsg.setData(bundle);

        if ("CONNECT".equals(data.getString(ACTION)))
        {   
            Log.d(debug, "Connecting");
            startService(new Intent(MySocket.this, MySocket.class));
            /*startService now calls connect();*/
        }
        else if ("DISCONNECT".equals(data.getString(ACTION)))
        {
            if (socket != null)
            {   
                socket.disconnect();
            }
            stopForeground( true );
            stopSelf();
        }
    }
}

1 个答案:

答案 0 :(得分:-1)

“如果你用startService(..)启动一个android服务,那么服务将一直运行,直到你明确地调用stopService(..)。系统可以运行服务有两个原因。如果有人调用Context。 startService()然后系统将检索服务(创建它并在需要时调用其onCreate()方法),然后使用客户端提供的参数调用其onStartCommand(Intent,int,int)方法。此时服务将继续运行,直到调用Context.stopService()或stopSelf()。注意,对Context.startService()的多次调用不会嵌套(尽管它们会导致多次对onStartCommand()的相应调用),所以无论多少次它启动后,一旦调用了Context.stopService()或stopSelf(),服务就会停止;但是,服务可以使用它们的stopSelf(int)方法来确保在处理完启动意图之前不停止服务。

客户端还可以使用Context.bindService()来获取与服务的持久连接。如果服务尚未运行(在执行此操作时调用onCreate()),这同样会创建服务,但不会调用onStartCommand()。客户端将接收服务从其onBind(Intent)方法返回的IBinder对象,允许客户端然后回调该服务。只要建立连接,服务将保持运行(客户端是否保留对服务的IBinder的引用)。通常,IBinder返回的是一个用aidl编写的复杂接口。

服务既可以启动,也可以绑定连接。在这种情况下,只要系统启动或者使用Context.BIND_AUTO_CREATE标志有一个或多个连接,系统就会保持服务运行。一旦这些情况都不成立,就会调用服务的onDestroy()方法并有效终止服务。从onDestroy()返回时,应完成所有清理(停止线程,取消注册接收器)。“

我希望这有助于完成工作。

谢谢:)