2012-05-14 18 views
7

Mam tę prostą usługę, która nadaje bieżącą lokalizację użytkownika. Chcę używać mechanizmu wiążącego tylko do kontrolowania cyklu życia usługi, ale usługa po prostu się nie uruchamia.Usługa nie jest tworzona (lub łączona) po bindService()

Co zrobiłem źle?

public class GPSActivity extends ListActivity { 
... 
protected void onResume() { 
     super.onResume(); 

     Log.i("Service", "Service bound"); 
     Intent intent = new Intent(this, LocationService.class); 
     bindService(intent, service_connection , Context.BIND_AUTO_CREATE); 
    } 

protected void onPause() { 
     if (dataUpdateReceiver!=null) 
      unregisterReceiver(dataUpdateReceiver); 
     unbindService(service_connection); 
     super.onPause(); 
    } 
class LocationServiceConnection implements ServiceConnection{ 
     public void onServiceConnected(ComponentName name, IBinder service) { 
      Log.i("Service", "Service Connected"); 
     } 
       public void onServiceDisconnected(ComponentName name) { 

     } 
    } 
} 

LocalBinder.java

public class LocalBinder<S> extends Binder { 
    private String TAG = "LocalBinder"; 
    private WeakReference<S> mService; 


    public LocalBinder(S service){ 
     mService = new WeakReference<S>(service); 
    } 


    public S getService() { 
     return mService.get(); 
    } 
} 

LocationService.java

public class LocationService extends Service { 
    public void onCreate() { 
     initLocationListener(); 
     Log.i("Location Service","onCreate()"); 
    } 

    @Override 
    public int onStartCommand(Intent intent, int flags, int startId) { 
     Log.i("Location Service", "Received start id " + startId + ": " + intent); 
     return START_NOT_STICKY; 
    } 

    private final IBinder mBinder = new LocalBinder<LocationService>(this); 
    @Override 
    public IBinder onBind(Intent intent) { 
     return mBinder; 
    } 
} 

AndroidManifest.xml

<application 
    android:icon="@drawable/ic_launcher" 
    android:label="@string/app_name" > 
    ... 

    <service android:name=".LocationService"> 
    </service> 
</application> 

Edycja: Naprawiono dzięki odpowiedzi NickT.

Manifest wpis nie miał zamiaru filtr lub poprawną nazwę

<service 
    android:enabled="true" 
    android:name="com.android.gps.services.LocationService"> 
    <intent-filter> 
      <action android:name="com.android.gps.services.LocationService" /> 
    </intent-filter> 
</service> 

a intencje użyłem do wiązania było jak te, które trzeba użyć przy rozpoczynaniu działalności. prawidłowa brzmi:

Intent intent = new Intent("com.android.gps.services.LocationService"); 
+0

Co twoja 'publicznego IBinder onBind (Intent intent)' 'w sposób LocationService.java' wyglądać? – Jens

+0

@Jens Edytowałem pytanie, aby uwzględnić metodę onBind(). – bughi

+0

Hm. I nawet nie dostajesz 'Log.i (" Usługa lokalizacji "," onCreate() ");' zaloguj się do swojego logcat? – Jens

Odpowiedz

3

The onStartCommand tylko wykonać jeśli usługa jest jawnie uruchomiona, wygląda na to, że chcesz ją związać, co jest w porządku. Nie widzę jednak, żebyś poprawnie skonfigurował połączenie serwisowe. Publikuję mój program pośredniczący, który pokazuje jak powiązać usługę i wywołać metodę w usłudze za pomocą spinacza. Możesz go uruchomić i zobaczyć sekwencję różnych komunikatów dziennika. Oczywiście będziesz musiał dodać kod BroadcastReceiver i onLocationChaged, aby był dla Ciebie przydatny.

Activity

package com.servtest.test; 

import com.servtest.test.LocationService.LocalBinder; 
import android.app.Activity; 
import android.content.ComponentName; 
import android.content.Context; 
import android.content.Intent; 
import android.content.ServiceConnection; 
import android.os.Bundle; 
import android.os.IBinder; 
import android.util.Log; 

public class ServiceTestActivity extends Activity { 

    boolean mServiceConnected = false; 
    boolean mBound = false; 
    private LocationService mLocnServ; 

    ServiceConnection mServconn = new ServiceConnection() { 
     @Override 
     public void onServiceConnected(ComponentName name, IBinder service) { 
      Log.d("SVTEST", "Activity service connected"); 
      LocalBinder binder = (LocalBinder) service; 
      mLocnServ = binder.getService(); 
      // Can't call this methodInTheService UNTIL IT'S BOUND! 
      mLocnServ.methodInTheService(); 
      mBound = true; 
     } 

     @Override 
     public void onServiceDisconnected(ComponentName name) { 
      Log.d("SVTEST", "Activity service disconnected"); 
      mBound = false; 
     } 
    }; 
    @Override 
    public void onCreate(Bundle savedInstanceState) { 
     super.onCreate(savedInstanceState); 
     setContentView(R.layout.main); 
    } 
    @Override 
    public void onStart() { 
     super.onStart(); 
     Log.d("SVTEST", "Activity onStart"); 
     mServiceConnected = bindService(new Intent(
       "com.servtest.test.LOCATIONSERVICE"), mServconn, 
       Context.BIND_AUTO_CREATE); 
    } 
    @Override 
    protected void onResume() { 
     super.onResume(); 
     Log.d("SVTEST", "Activity onResume"); 
    } 
    @Override 
    public void onPause() { 
     Log.d("SVTEST", "Activity onPause"); 
     super.onPause(); 
    } 
    @Override 
    public void onStop() { 
     Log.d("SVTEST", "Activity onStop"); 
     if (mBound) { 
      unbindService(mServconn); 
      mBound = false; 
     } 
     super.onStop(); 
    } 

} 

Usługa

package com.servtest.test; 

import android.app.Service; 
import android.content.Intent; 
import android.location.Location; 
import android.location.LocationListener; 
import android.os.Binder; 
import android.os.Bundle; 
import android.os.IBinder; 
import android.util.Log; 

public class LocationService extends Service implements LocationListener { 

    private final IBinder mBinder = new LocalBinder(); 

    @Override 
    public void onLocationChanged(Location arg0) {} 
    @Override 
    public void onProviderDisabled(String arg0) {} 
    @Override 
    public void onProviderEnabled(String arg0) {} 
    @Override 
    public void onStatusChanged(String arg0, int arg1, Bundle arg2) {} 

    @Override 
    public IBinder onBind(Intent intent) { 
     Log.d("SVTEST", "Loc service ONBIND"); 
     return mBinder; 
    } 
    @Override 
    public boolean onUnbind(Intent intent) { 
     Log.d("SVTEST", "Loc service ONUNBIND"); 
     return super.onUnbind(intent); 
    } 
    @Override 
    public int onStartCommand(Intent intent, int flags, int startId) { 
     // Won't run unless it's EXPLICITLY STARTED 
     Log.d("SVTEST", "Loc service ONSTARTCOMMAND"); 
     return super.onStartCommand(intent, flags, startId); 
    } 
    @Override 
    public void onDestroy() { 
     super.onDestroy(); 
     Log.d("SVTEST", "Loc service ONDESTROY"); 
    } 

    public class LocalBinder extends Binder { 
     LocationService getService() { 
      // Return this instance of LocalService so clients can call public methods 
      return LocationService.this; 
     } 
    } 

    public void methodInTheService() { 
     // A method you can call in the service 
     Log.d("SVTEST", "Loc service EXECUTING THE METHOD"); 
    } 
} 

Manifest

<?xml version="1.0" encoding="utf-8"?> 
<manifest xmlns:android="http://schemas.android.com/apk/res/android" 
    package="com.servtest.test" 
    android:versionCode="1" 
    android:versionName="1.0" > 
    <application 
     android:icon="@drawable/ic_launcher" 
     android:label="@string/app_name" > 
     <activity 
      android:name=".ServiceTestActivity" 
      android:label="@string/app_name" > 
      <intent-filter> 
       <action android:name="android.intent.action.MAIN" /> 
       <category android:name="android.intent.category.LAUNCHER" /> 
      </intent-filter> 
     </activity> 
     <service 
      android:enabled="true" 
      android:name="LocationService"> 
      <intent-filter> 
       <action android:name="com.servtest.test.LOCATIONSERVICE" /> 
      </intent-filter> 
     </service> 
    </application> 
</manifest> 

Nadzieja to pomaga

+0

Dzięki redagowałem moją odpowiedź z tym, co zrobiłem źle – bughi

0

miałem Activity wdrożyć ServiceConnection i związany tak:

bindService(new Intent(this, Service.class), this, Context.BIND_AUTO_CREATE); 

Następnie zajmował się zwrotnych dla onServiceConnected() i onServiceDisconnected() w moim Activity

+0

To nie robi żadnej różnicy. – bughi

0

Po wywołaniu bindService można uzyskać th jest błąd:

ActivityManager java.lang.ClassCastException: android.os.BinderProxy cannot be cast to com.android.server.am.ActivityRecord$Token

Sprawdź wyjście LogCat.

To jest błąd Androida.

go rozwiązać, należy getApplicationContext().bindService(...)

Powiązane problemy