2012-05-14 4 views
7

Ich habe diesen einfachen Dienst, der den aktuellen Standort des Benutzers sendet. Ich möchte den Binding-Mechanismus nur dazu verwenden, den Service-Lebenszyklus zu steuern, aber der Service startet gerade nicht.Dienst wird nicht erstellt (oder verbindet) nach bindService()

Was habe ich falsch gemacht?

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> 

EDIT: Fixed dank NickTs Antwort.

Das Manifest Eintrag

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

Und die Absicht habe ich nicht Sie, wenn eine Aktivität Starten ein Intent-Filter oder die richtigen Namen verwenden müssen, wie die, die für die Bindung war haben. die richtige ist:

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

Was Ihr nicht 'öffentliche IBinder onBind (Intent Absicht)' Methode in 'LocationService.java' aussehen? – Jens

+0

@Jens Ich habe die Frage bearbeitet, um die onBind() Methode einzuschließen. – bughi

+0

Hm. Und Sie bekommen nicht einmal den 'Log.i (" Location Service "," onCreate() "); loggen Sie sich in Ihren Logcat ein? – Jens

Antwort

3

Die onStartCommand nur behandelt wird ausgeführt Wenn der Dienst explizit gestartet wird, sieht es so aus, als ob Sie nur daran binden möchten, was in Ordnung ist. Ich sehe jedoch nicht, dass Sie die Dienstverbindung ordnungsgemäß eingerichtet haben. Ich poste mein Stub-Programm, das zeigt, wie man an einen Service bindet und eine Methode im Service durch einen Binder aufruft. Möglicherweise möchten Sie dies ausführen und die Reihenfolge der verschiedenen Protokollmeldungen sehen. Sie müssen Ihren BroadcastReceiver und Ihren onLocationChaged-Code natürlich hinzufügen, um ihn für Sie nützlich zu machen.

Die Aktivität

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(); 
    } 

} 

Das Service

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"); 
    } 
} 

Das 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> 

Hope this

+0

Danke, ich habe meine Antwort mit dem bearbeitet, was ich falsch gemacht habe – bughi

0

hatte ich meine ActivityServiceConnection implementieren und wie folgt gebunden:

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

dann die Rückrufe für onServiceConnected() und onServiceDisconnected() in meinem Activity

+0

Das macht keinen Unterschied. – bughi

0

hilft Wenn Sie anrufen bindService können Sie erhalten th Fehler:

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

die Ausgabe von logcat prüfen.

Dies ist ein Fehler von Android.

Um es zu lösen, verwenden getApplicationContext().bindService(...)

Verwandte Themen