Android не может привязать к сервису - PullRequest
3 голосов
/ 11 февраля 2011

Новичок на андроид, пытаюсь выяснить сервисы.Я пытаюсь привязать сервис к деятельности, я следую примерам в документации, но получаю исключение NullPointerException в строке, отмеченной ниже (appService.playSong (title)).Проверка его в отладчике показывает, что appService действительно имеет значение null.

public class Song extends Activity implements OnClickListener,Runnable {
protected static int currentPosition;
private ProgressBar progress;
private TextView songTitle;
private MPService appService;

private ServiceConnection onService = new ServiceConnection() {
    public void onServiceConnected(ComponentName className,
            IBinder rawBinder) {
        appService = ((MPService.LocalBinder)rawBinder).getService();
    }

    public void onServiceDisconnected(ComponentName classname) {
        appService = null;
    }
};


public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.song);

    Intent bindIntent = new Intent(Song.this,MPService.class);
    bindService(bindIntent,onService,
            Context.BIND_AUTO_CREATE);

    Bundle b = getIntent().getBundleExtra("songdata");
    String title = b.getString("song title");

    // ... 

    appService.playSong(title); // nullpointerexception

    // ...

}

Вот соответствующая часть службы:

package org.example.music;

// imports

public class MPService extends Service {
private MediaPlayer mp;
public static int currentPosition = 0;
public List<String> songs = new ArrayList<String>();
public static String songTitle;
private static final String MEDIA_PATH = new String("/mnt/sdcard/");

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

    mp = new MediaPlayer();
    songs = Music.songs;
}

@Override
public int onStartCommand(Intent intent, int flags, int startId) {
    return Service.START_STICKY;
}

public class LocalBinder extends Binder {
    MPService getService() {
        return MPService.this;
    }
}

private final IBinder binder = new LocalBinder();

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

public void playSong(String songPath) {
try {
    mp.reset();
    mp.setDataSource(songPath);
    mp.prepare();
    mp.start();

    mp.setOnCompletionListener(new OnCompletionListener() {
        public void onCompletion(MediaPlayer arg0) {
            nextSong();
        }
    });

    songTitle = songPath.substring(12,songPath.length()-4);

} catch (IOException e) {
    Log.v(getString(R.string.app_name),e.getMessage());
}
}

public void nextSong() {
if (++currentPosition >= songs.size()) {
    currentPosition = 0;
}
String song = MEDIA_PATH+songs.get(currentPosition);
playSong(song);
} 

public void prevSong() {
if (--currentPosition<0) {
    currentPosition=songs.size()-1;
}
String song = Music.MEDIA_PATH+songs.get(currentPosition);
playSong(song);
}

public int getSongPosition() {
return mp.getCurrentPosition();
}

public MediaPlayer getMP() {
return mp;
}
}

Я зарегистрировал службу в AndroidManifest.xml и установил Android: enabled = "true".Вы видите здесь очевидные ошибки?

Ответы [ 3 ]

1 голос
/ 14 февраля 2011

Существует два вида привязок, которые вы можете сделать локальными и удаленными. Локальный предназначен только для использования вашим приложением, а удаленный - для использования любым приложением, реализующим определенный интерфейс. Вы должны начать с локальной привязки.

Учебник по локальной привязке .
Учебник по дистанционному связыванию .

Мое решение без привязки:

public class MyActivity extends Activity{

  @Override
  public void onCreate(Bundle savedInstanceState){
   ...
   Intent it = new Intent(MyService.ACTIVITY_START_APP);
   it.setClass(getApplicationContext(), MyService.class);
   startService(it);
}

  ...

@Override
    protected void onResume() {
        super.onResume();
        registerBroadcastReceiver();
    }

@Override
    protected void onPause() {
        super.onPause();
        this.unregisterReceiver(this.receiver);
    }

  ...

private BroadcastReceiver receiver = new BroadcastReceiver(){

        @Override
        public void onReceive(Context context, Intent intent) {
            if (intent.getAction().equals(MyService.BROADCAST_INIT)) {
                //do your stuff here after init
            }
        }
    };

private void registerBroadcastReceiver(){
         IntentFilter filter = new IntentFilter();
         filter.addAction(HMyService.BROADCAST_INIT);
         this.registerReceiver(receiver, filter);
    }
}

Ваш сервис:

public class MyService extends Service{

public static final String BROADCAST_INITIAL_DATA = "org.myapp.BROADCAST_INIT";
public static final String ACTIVITY_START_APP = "org.myapp.ACTIVITY_START_APP";

  @Override
  public int onStartCommand (Intent intent, int flags, int startId){
    super.onStartCommand(intent, flags, startId);
    if(intent.getAction().equals(ACTIVITY_START_APP)){
      //do your initialization
      //inform the client/GUI
      Intent i = new Intent();
      i.setAction(BROADCAST_INIT);
      sendBroadcast(i);
    }else{
      //some other stuff like handle buttons
    }              
  }
}

удачи.

1 голос
/ 20 февраля 2011

Предполагается, что bindService () будет подключаться к службе синхронно, но подключение будет доступно только после завершения onCreate ().

Фреймворк запускается onCreate () в потоке пользовательского интерфейса, а bindService () просто делает примечание для подключения к сервису позже. Подключение к сервису всегда будет выполняться в потоке пользовательского интерфейса, поэтому это может произойти только после выполнения onCreate. Вы даже не можете рассчитывать на подключение, устанавливаемое сразу после onCreate (). Это произойдет когда-нибудь после этого :). Кроме того, инфраструктура может отключить службу по своему желанию, хотя это должно происходить только в условиях нехватки памяти.

Итак, переместите код, который работает с appService, из onCreate () в onServiceConnected (), и он заработает.

0 голосов
/ 11 февраля 2011

На первый взгляд кажется, что вы пытаетесь получить доступ к своему сервису до завершения привязки. Вы должны убедиться, что onServiceConnected запущен, прежде чем пытаться вызвать какие-либо методы в вашем сервисе.

Пример:

Intent bindIntent = new Intent(Song.this,MPService.class);
bindService(bindIntent,onService, Context.BIND_AUTO_CREATE);

//Wait until service has bound
while(appService == null){
     Thread.sleep(100);
}
appService.playSong(title);

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

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