Редакция 2. Как передать данные из фоновой службы / потока в какое-либо другое действие, кроме MainActivity, которое создало фоновую службу - PullRequest
8 голосов
/ 15 июня 2011

Я создал простое приложение для Android для тестирования того, как использовать обработчик или обработчики для передачи данных из фоновой службы / потока в некоторую другую операцию , отличную от MainActivity, которая создала фоновую службу.У меня есть служба, поток и обработчик, работающий в MainActivity.Последний шаг - получить обработчик для передачи данных в другое действие , отличное от MainActivity.Я могу заставить Службу передавать сообщения обработчику MainActivity, но я не знаю, как заставить его передавать данные в какую-то другую операцию .

Зачем кому-то это делать?Я подумал, что это по сравнению с простым MP3-плеером, но на самом деле это хорошее FM-радио.MainActivity использует фоновую службу, которая позволяет мне выбирать FM-радиостанцию.Когда я запускаю действие Play, оно должно привязываться к одному и тому же фоновому сервису, чтобы я мог продолжать слушать, пока он (приятная часть) отображает графический эквалайзер или анимацию звука.По сути, я не знаю, как привязать к фоновому Сервису несколько операций.

Мой код изначально был основан на странице 304 примера службы в Pro Android 2 и помог. 1015 *

Пожалуйста, ознакомьтесь с моим текущим кодом.Он состоит из трех тщательно прокомментированных файлов, которые описывают то, что я пытаюсь сделать, и трудности, с которыми я сталкиваюсь, передавая данные в некоторые другие действия в дополнение к MainActivity:

/**************************************************************************************************
 * File: MainActivity.java
 * Application: BackgroundService
 * Description: This file contains the main activity that is run when the BackgroundService
 *     application is launched.
 **************************************************************************************************/

package com.marie.mainactivity;

import com.marie.mainactivity.BackgroundService;
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.os.Messenger;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;

/*
 * Class: MainActivity
 * Purpose: Using a button, the MainActivity class starts the backgroundService and
 *     the RcvMessages activity. Using another button MainActivity stops the backgroundService.
 *     NOTE: RcvMessages is only a stub that does nothing but display a simple message.
 * Handler: MainActivity defines and provides a reference to "handler" for the backgroundService.
 */
public class MainActivity extends Activity {
    private static final String TAG = "MainActivity";

    @Override
    public void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        Log.d(TAG, "starting service");

        /*
         * The bind button: bindBtn
         * Clicking this button starts the background Service and launches the
         * RcvMessages activity. NOTE: RcvMessages is only a stub so far.
         */
        Button bindBtn = (Button)findViewById(R.id.bindBtn);
        bindBtn.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View arg0) {

                // Start the background Service for sending canned messages to the handler as a test.
                Intent backgroundService = new Intent(MainActivity.this, com.marie.mainactivity.BackgroundService.class);
                backgroundService.putExtra(BackgroundService.EXTRA_MESSENGER, new Messenger(handler));
                startService(backgroundService);

                // Start the RcvMessages activity to receive messages from the handler. But how???
                Intent messages = new Intent(MainActivity.this, com.marie.mainactivity.RcvMessages.class);
                startActivity(messages);
            }
        });

        /*
         * The unbind button: unbindBtn
         * Clicking this button stops the background Service.
         */
        Button unbindBtn = (Button)findViewById(R.id.unbindBtn);
        unbindBtn.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View arg0) {

                // Stop the background Service
                Intent backgroundService = new Intent(MainActivity.this, BackgroundService.class);
                stopService(backgroundService);
            }
        });
    }

    /*
     * This is the handler to be passed to the background Service via a Messenger.
     * NOTE: I want it to send messages to my RcvMessages activity.
     */
    private Handler handler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            // simple handler test (does not send messages to RcvMessages activity
            String obj = (String) msg.obj;
            Log.i("handleMessge", "obj: " + obj);  
        }
    };
}

/**************************************************************************************************
 * File: BackgroundService.java
 * Application: BackgroundService
 * Description: This file contains the background Service that is launched by the MainActivity's
 *     bind button.
 **************************************************************************************************/

package com.marie.mainactivity;

import android.app.Notification;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.app.Service;
import android.content.Intent;
import android.os.Bundle;
import android.os.IBinder;
import android.os.Message;
import android.os.Messenger;
import android.os.RemoteException;

/*
 * Class: BackgroundService
 * Purpose: Using the onStart() method the BackgroundService gets the reference to the
 *     Messenger instance that was passed to BackgroundService. The messenger is then
 *     used by the ServiceWorker() thread to send messages to the handler that is defined
 *     in the MainActivity class.
 */
public class BackgroundService extends Service {
    private NotificationManager notificationMgr;

    public static final String EXTRA_MESSENGER = "com.marie.mainactivity.EXTRA_MESSENGER";
    private Messenger messenger;

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

        notificationMgr = (NotificationManager)getSystemService(NOTIFICATION_SERVICE);

        displayNotificationMessage("starting Background Service");

        Thread thr = new Thread(null, new ServiceWorker(), "BackgroundService");
        thr.start();
    }   

    /*
     * This is the ServiceWorker thread that passes messages to the handler defined in
     * the MainActivity class.
     * NOTE: Instead of passing messages to a handler in MainActivity I would like
     * it to pass messages to a handler defined in the RcvMessages activity.
     */
    class ServiceWorker implements Runnable
    {
        public void run() {
            // do background processing here... something simple

            // send a message to the handler defined in the MainActivity class
            try {
                Message msg1 = Message.obtain();
                msg1.obj = "Hello 1";
                messenger.send(msg1);
            } catch (RemoteException e) {
                e.printStackTrace();
            }

            // stop the service when done...
            // BackgroundService.this.stopSelf();
            // Or use the unbindBtn in the MainActivity class.
        }
    }

    @Override
    public void onDestroy()
    {
        displayNotificationMessage("stopping Background Service");
        super.onDestroy();
    }

    /*
     * onStart is where we get our reference the Messenger that allows
     * us to send messages to the handler defined in the MainActivity class.
     */
    @Override
    public void onStart(Intent intent, int startId) {
        super.onStart(intent, startId);

        Bundle extras = intent.getExtras();
        messenger = (Messenger)extras.get(EXTRA_MESSENGER);
    }

    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }

    private void displayNotificationMessage(String message)
    {
        Notification notification = new Notification(R.drawable.note, message, System.currentTimeMillis());

        PendingIntent contentIntent = PendingIntent.getActivity(this, 0, new Intent(this, MainActivity.class), 0);

        notification.setLatestEventInfo(this, "Background Service", message, contentIntent);

        notificationMgr.notify(R.id.app_notification_id, notification);
    }
}

/**************************************************************************************************
 * File: RcvMessages.java
 * Application: BackgroundService
 * Description: This file contains stub code that displays a test message in an EditText.
 **************************************************************************************************/

package com.marie.mainactivity;

import android.app.Activity;
import android.os.Bundle;
import android.text.InputType;
import android.widget.EditText;

/*
 * Class: RcvMessages
 * Purpose: RcvMessages is stub code that I want to extend in some way to receive messages from
 *     the background Service.
 *     NOTE: I don't know who to do this.
 */
public class RcvMessages extends Activity {

    EditText myText;

    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.messages);

        myText = (EditText)findViewById(R.id.my_text);

        myText.setSingleLine();
        myText.setInputType(InputType.TYPE_NULL);

        // Display a simple test message for now.
        myText.setText("RcvMessages here");
    }
}

Любая помощьиспользование фоновой службы, потока и обработчика (ей) для передачи данных в некоторую другую деятельность в дополнение к MainActivity, которая создала фоновую службу, будет весьма приветствоваться.

1 Ответ

9 голосов
/ 18 июня 2011

Handler связан с конкретным потоком, поэтому до тех пор, пока вы создаете его в потоке пользовательского интерфейса, все, что вам нужно, это делиться обработчиком. Я бы поставил твой обработчик в мой Application класс. И вы можете положить туда целую Messenger, если хотите. Затем вы можете получить доступ к обработчику в любой деятельности через ((MyApplication)getApplication()).getHandler(). Когда действие начинается или приостанавливается, вы можете зарегистрировать его как обратный вызов.

Так что-то вроде этого

public class MyApplication extends Application {
   Handler h = new Handler() {
      public void handleMessage(Message m) {
        if (realCallback!=null) {
           realCallback.handleMessage(m);
        }
      }
   };
   Handler.Callback realCallback=null;
   ......
   public Handler getHandler() {
      return h;
   }

   public void setCallback(Handler.Callback c) {
      realCallback = c;
   }
}

В любой деятельности, требующей получения запросов через мессенджер

public class MyActivity extends Activity implements Handler.Callback

......

public void onStart() {
   ((MyApplication)getApplication()).setCallback(this);
}

public void onPause() {
   ((MyApplication)getApplication()).setCallback(null);
}

public void handleMessage(Message msg) {
  //.... Do stuff ....
}

}

Это всего лишь идея. Возможно, вам придется настроить его под свои нужды.

И не забудьте установить MyApplication имя в AndroidManifest.xml

...