Android--進程間通信的幾種實現方式
一、概述
由于應用程序之間不能共享內存。在不同應用程序之間交互數據(跨進程通訊),在AndroidSDK中提供了4種用于跨進程通訊的方式。這4種方式正好對應于android系統中4種應用程序組件:Activity、Content Provider、Broadcast和Service。其中Activity可以跨進程調用其他應用程序的Activity;Content Provider可以跨進程訪問其他應用程序中的數據(以Cursor對象形式返回),當然,也可以對其他應用程序的數據進行增、刪、改操 作;Broadcast可以向android系統中所有應用程序發送廣播,而需要跨進程通訊的應用程序可以監聽這些廣播;Service和Content Provider類似,也可以訪問其他應用程序中的數據,但不同的是,Content Provider返回的是Cursor對象,而Service返回的是Java對象,這種可以跨進程通訊的服務叫AIDL服務。
Activity
Activity的跨進程訪問與進程內訪問略有不同。雖然它們都需要Intent對象,但跨進程訪問并不需要指定Context對象和Activity的 Class對象,而需要指定的是要訪問的Activity所對應的Action(一個字符串)。有些Activity還需要指定一個Uri(通過 Intent構造方法的第2個參數指定)。
在android系統中有很多應用程序提供了可以跨進程訪問的Activity,例如,下面的代碼可以直接調用撥打電話的Activity。
- Intent callIntent = new Intent(Intent.ACTION_CALL, Uri.parse( "tel:12345678" );
- startActivity(callIntent);
Content Provider
Android應用程序可以使用文件或SqlLite數據庫來存儲數據。Content Provider提供了一種在多個應用程序之間數據共享的方式(跨進程共享數據)。應用程序可以利用Content Provider完成下面的工作
1. 查詢數據
2. 修改數據
3. 添加數據
4. 刪除數據
雖然Content Provider也可以在同一個應用程序中被訪問,但這么做并沒有什么意義。Content Provider存在的目的向其他應用程序共享數據和允許其他應用程序對數據進行增、刪、改操作。
Android系統本身提供了很多Content Provider,例如,音頻、視頻、聯系人信息等等。我們可以通過這些Content Provider獲得相關信息的列表。這些列表數據將以Cursor對象返回。因此,從Content Provider返回的數據是二維表的形式。
廣播(Broadcast)
廣播是一種被動跨進程通訊的方式。當某個程序向系統發送廣播時,其他的應用程序只能被動地接收廣播數據。這就象電臺進行廣播一樣,聽眾只能被動地收聽,而不能主動與電臺進行溝通。
在應用程序中發送廣播比較簡單。只需要調用sendBroadcast方法即可。該方法需要一個Intent對象。通過Intent對象可以發送需要廣播的數據。
Service
1.利用AIDL Service實現跨進程通信
這是我個人比較推崇的方式,因為它相比Broadcast而言,雖然實現上稍微麻煩了一點,但是它的優勢就是不會像廣播那樣在手機中的廣播較多時會有明顯的時延,甚至有廣播發送不成功的情況出現。
注意普通的Service并不能實現跨進程操作,實際上普通的Service和它所在的應用處于同一個進程中,而且它也不會專門開一條新的線程,因此如果在普通的Service中實現在耗時的任務,需要新開線程。
要實現跨進程通信,需要借助AIDL(Android Interface Definition Language)。Android中的跨進程服務其實是采用C/S的 架構 ,因而AIDL的目的就是實現通信接口。
首先舉一個簡單的栗子。
服務端 代碼如下:
首先是aidl的代碼:
package com.android.service;
interface IData
{
int getRoomNum();
}
RoomService的代碼如下:
package com.android.service;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.os.RemoteException;
public class RoomService extends Service{
private IData.Stub mBinder=new IData.Stub() {
@Override
public int getRoomNum() throws RemoteException {
return 3008;
}
};
@Override
public IBinder onBind(Intent intent) {
return mBinder;
}
}
AndroidManifest如下:
<?<span id="19_nwp" style="padding: 0px; width: auto; height: auto; float: none;"><a target=_blank id="19_nwl" ?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.android.aidlsampleservice"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk
android:minSdkVersion="8"
android:targetSdkVersion="17" />
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@<span id="20_nwp" style="padding: 0px; width: auto; height: auto; float: none;"><a target=_blank id="20_nwl" >
<service android:name="com.android.service.RoomService">
<intent-filter>
<action android:name="com.aidl.service.room"/>
</intent-filter>
</service>
</application>
</manifest>
然后運行該Service所在的Project即可。
客戶端代碼如下:
注意客戶端也要有aidl文件,所以最簡單的辦法就是將Service端中aidl所在的包直接復制過去。另外要注意的是在onDestroy中要解除和Service的綁定。
MainActivity.java的代碼如下:
package com.example.aidlsampleclient;
import com.android.service.IData;
import android.os.Bundle;
import android.os.IBinder;
import android.os.RemoteException;
import android.app.Activity;
import android.content.ComponentName;
import android.content.Intent;
import android.content.ServiceConnection;
import android.util.Log;
import android.view.Menu;
import android.widget.Button;
import android.widget.Toast;
import android.view.View;
public class MainActivity extends Activity implements View.OnClickListener{
private static final String TAG="MainActivity";
private static final String ROOM_SERVICE_ACTION="com.aidl.service.room";
private Button bindServiceButton;
private Button getServiceButton;
IData mData;
private ServiceConnection conn=new ServiceConnection()
{
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
Log.i(TAG,"----------------onServiceConnected--------");
mData=IData.Stub.asInterface(service);
}
@Override
public void onServiceDisconnected(ComponentName name) {
Log.i(TAG,"----------------onServiceDisconnected-------------");
mData=null;
}
};
private void initView()
{
bindServiceButton=(Button)findViewById(R.id.bindServiceButton);
getServiceButton=(Button)findViewById(R.id.getServiceButton);
bindServiceButton.setOnClickListener(this);
getServiceButton.setOnClickListener(this);
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initView();
}
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
<span id="17_nwp" style="padding: 0px; width: auto; height: auto; float: none;"><a target=_blank id="17_nwl" >switch</span></a></span>(v.getId())
{
case R.id.bindServiceButton:
bindService();
break;
case R.id.getServiceButton:
getService();
break;
default:
break;
}
}
private void bindService()
{
Intent intent=new Intent();
intent.setAction(ROOM_SERVICE_ACTION);
bindService(intent,conn,BIND_AUTO_CREATE);
}
private void getService()
{
try
{
if(mData!=null)
{
int roomNum=mData.getRoomNum();
showLongToast("RoomNum:"+roomNum);
}
}
catch(RemoteException ex)
{
ex.printStackTrace();
}
}
private void showLongToast(String info)
{
Toast.makeText(getBaseContext(), info, Toast.LENGTH_LONG).show();
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is p<span id="18_nwp" style="padding: 0px; width: auto; height: auto; float: none;"><a target=_blank id="18_nwl" >res</span></a></span>ent.
getMenuInflater().inflate(R.menu.main, menu);
return true;
}
@Override
protected void onDestroy() {
// TODO Auto-generated method stub
super.onDestroy();
unbindService(conn);
}
}
activity_main.xml的代碼如下:
<RelativeLayout <span id="14_nwp" style="padding: 0px; width: auto; height: auto; float: none;"><a target=_blank id="14_nwl"
xmlns:<span id="16_nwp" style="padding: 0px; width: auto; height: auto; float: none;"><a target=_blank id="16_nwl"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context=".MainActivity" >
<Button
android:id="@+id/bindServiceButton"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="BindService"
/>
<Button
android:id="@+id/getServiceButton"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="GetService"
android:layout_below="@id/bindServiceButton"
/>
</RelativeLayout>
運行結果如下:
然后舉一個稍微復雜一點的栗子,注意如果*.aidl文件中含有自定義的對象,那么該對象的類要實現Parcelable接口,并且要新建一個該類的aidl文件,否則會出現could not find import for class com.android.service.XX的錯誤,其中XX為類名。還是上面的栗子,但是aidl文件中添加了一些新的方法。仍以上面的RoomService為例,
Service端的代碼如下:
Room類的代碼為:
package com.android.service;
import android.os.Parcel;
import android.os.Parcelable;
public class Room implements Parcelable{
//房間號
private int roomNum;
//房間大小
private float roomSpace;
//是否有空調
private boolean hasAirConditioner;
//是否有Wifi
private boolean hasWifi;
//房間內的裝飾風格
private String decorativeStyle;
public static final Parcelable.Creator<Room>CREATOR=new Parcelable.Creator<Room>() {
@Override
public Room createFromParcel(Parcel source) {
return new Room(source);
}
@Override
public Room[] newArray(int size) {
// TODO Auto-generated method stub
return null;
}
};
public Room(int roomNum,float roomSpace,boolean hasAirConditioner,boolean hasWifi,String decorativeStyle)
{
this.roomNum=roomNum;
this.roomSpace=roomSpace;
this.hasAirConditioner=hasAirConditioner;
this.hasWifi=hasWifi;
this.decorativeStyle=decorativeStyle;
}
private Room(Parcel source)
{
roomNum=source.readInt();
roomSpace=source.readFloat();
boolean[]tempArray=new boolean[2];
source.readBooleanArray(tempArray);
hasAirConditioner=tempArray[0];
hasWifi=tempArray[1];
decorativeStyle=source.readString();
}
@Override
public String toString()
{
StringBuilder sb=new StringBuilder();
sb.append("Basic info of room is as follows:\n");
sb.append("RoomNum:"+roomNum+"\n");
sb.append("RoomSpace:"+roomSpace+"\n");
sb.append("HasAirConditioner:"+hasAirConditioner+"\n");
sb.append("HasWifi:"+hasWifi+"\n");
sb.append("Decorative Style:"+decorativeStyle);
return sb.toString();
}
@Override
public int describeContents() {
// TODO Auto-generated method stub
return 0;
}
@Override
public void writeToParcel(Parcel dest,int flags) {
dest.writeInt(roomNum);
dest.writeFloat(roomSpace);
dest.writeBooleanArray(new boolean[]{hasAirConditioner,hasWifi});
dest.writeString(decorativeStyle);
}
}
Room的聲明為:
package com.android.service;
parcelable Room;
IRoom.aidl的代碼為:
package com.android.service;
import com.android.service.Room;
interface IRoom
{
Room getRoom();
}
RoomService的代碼為:
package com.android.service;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.os.RemoteException;
public class RoomService extends Service{
private IRoom.Stub mBinder=new IRoom.Stub() {
@Override
public Room getRoom() throws RemoteException {
Room room=new Room(3008,23.5f,true,true,"IKEA");
return room;
}
};
@Override
public IBinder onBind(Intent intent) {
return mBinder;
}
}
由于AndroidManifest. xml
的代碼不變,因而此處不再貼出。下面是客戶端的代碼:
package com.example.aidlsampleclient;
import com.android.service.IRoom;
import android.os.Bundle;
import android.os.IBinder;
import android.os.RemoteException;
import android.app.Activity;
import android.content.ComponentName;
import android.content.Intent;
import android.content.ServiceConnection;
import android.util.Log;
import android.view.Menu;
import android.widget.Button;
import android.widget.Toast;
import android.view.View;
public class MainActivity extends Activity implements View.OnClickListener{
private static final String TAG="MainActivity";
//private static final String SERVICE_ACTION="com.aidl.service.data";
private static final String ROOM_SERVICE_ACTION="com.aidl.service.room";
private Button bindServiceButton;
private Button getServiceButton;
IRoom mRoom;
private ServiceConnection conn=new ServiceConnection()
{
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
Log.i(TAG,"----------------onServiceConnected--------");
showLongToast("onServiceConnected");
mRoom=IRoom.Stub.asInterface(service);
}
@Override
public void onServiceDisconnected(ComponentName name) {
Log.i(TAG,"----------------onServiceDisconnected-------------");
mRoom=null;
}
};
private void initView()
{
bindServiceButton=(Button)findViewById(R.id.bindServiceButton);
getServiceButton=(Button)findViewById(R.id.getServiceButton);
bindServiceButton.setOnClickListener(this);
getServiceButton.setOnClickListener(this);
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initView();
}
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
<span id="12_nwp" style="padding: 0px; width: auto; height: auto; float: none;"><a target=_blank id="12_nwl" >switch</span></a></span>(v.getId())
{
case R.id.bindServiceButton:
bindService();
break;
case R.id.getServiceButton:
getService();
break;
default:
break;
}
}
private void bindService()
{
Intent intent=new Intent();
intent.setAction(ROOM_SERVICE_ACTION);
bindService(intent,conn,BIND_AUTO_CREATE);
}
private void getService()
{
if(mRoom!=null)
{
try
{
showLongToast(mRoom.getRoom().toString());
}
catch (RemoteException e)
{
e.printStackTrace();
}
}
}
private void showLongToast(String info)
{
Toast.makeText(getBaseContext(), info, Toast.LENGTH_LONG).show();
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is p<span id="13_nwp" style="padding: 0px; width: auto; height: auto; float: none;"><a target=_blank id="13_nwl" >res</span></a></span>ent.
getMenuInflater().inflate(R.menu.main, menu);
return true;
}
@Override
protected void onDestroy() {
// TODO Auto-generated method stub
super.onDestroy();
unbindService(conn);
}
}
注意首先仍然是要將Room,IRoom的代碼復制過去,否則會出錯。
運行結果如下:
顯然,客戶端已經成功讀取到服務信息。
注意,上面的所舉的栗子其實不只是跨進程,還是跨應用。要注意的是,跨應用一定跨進程,但是跨進程不一定是跨應用。對于跨應用的情況,利用AIDL基本上是較好的解決了問題,但也只是“較好”而已,實際上并不完美,比如,如果要增加一個服務,如果利用AIDL的話,那么又要改寫aidl文件,如果是涉及自定義對象,則還要增加自定義對象的聲明,而且這種改變不只是Service端的改變,客戶端也要跟著改變,顯然這種解決方案不夠優雅。
那么,有沒有更優雅的方法呢?
當然有,那就是利用Service的onStartCommand(Intent intent, int flags, int startId)方法。
服務端 代碼如下:
package com.android.service;
import android.app.Service;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.os.IBinder;
import android.os.RemoteException;
import android.util.Log;
import android.widget.Toast;
public class RoomService extends Service{
private static final String TAG="RoomService";
private static final int CLEAN_SERVICE=0x1;
private static final int ORDER_SERVICE=0x2;
private static final int PACKAGE_SERVICE=0x3;
private static final String SERVICE_KEY="ServiceName";
@Override
public void onStart(Intent intent, int startId) {
showLog("onStart");
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
//String action=intent.getAction();
Log.i(TAG,"onStartCommand");
int actionFlag=intent.getIntExtra(SERVICE_KEY, -1);
<span id="11_nwp" style="padding: 0px; width: auto; height: auto; float: none;"><a target=_blank id="11_nwl" >switch</span></a></span>(actionFlag)
{
case CLEAN_SERVICE:
showShortToast("Start Clean Service Right Now");
break;
case ORDER_SERVICE:
showShortToast("Start Order Service Right Now");
break;
case PACKAGE_SERVICE:
showShortToast("Start Package Service Right Now");
break;
default:
break;
}
return super.onStartCommand(intent, flags, startId);
}
private void showLog(String info)
{
Log.i(TAG,info);
}
private void showShortToast(String info)
{
Toast.makeText(getBaseContext(), info, Toast.LENGTH_SHORT).show();
}
@Override
public void onDestroy() {
// TODO Auto-generated method stub
showLog("onDestroy");
super.onDestroy();
}
@Override
public void onCreate() {
// TODO Auto-generated method stub
showLog("onCreate");
super.onCreate();
}
@Override
public IBinder onBind(Intent intent) {
showLog("onBind");
return null;
}
@Override
public boolean onUnbind(Intent intent) {
showLog("onUnbind");
return super.onUnbind(intent);
}
}
客戶端代碼如下:
package com.example.aidlsampleclient;
import com.android.service.IRoom;
import com.android.service.RoomService;
import android.os.Bundle;
import android.os.IBinder;
import android.os.RemoteException;
import android.app.Activity;
import android.content.ComponentName;
import android.content.Intent;
import android.content.ServiceConnection;
import android.util.Log;
import android.view.Menu;
import android.widget.Button;
import android.widget.Toast;
import android.view.View;
public class MainActivity extends Activity implements View.OnClickListener{
private static final String TAG="MainActivity";
private static final String ROOM_SERVICE_ACTION="com.aidl.service.room";
private static final int CLEAN_SERVICE=0x1;
private static final int ORDER_SERVICE=0x2;
private static final int PACKAGE_SERVICE=0x3;
private static final String SERVICE_KEY="ServiceName";
private Button cleanButton;
private Button orderButton;
private Button packageButton;
private void initView()
{
cleanButton=(Button)findViewById(R.id.cleanButton);
orderButton=(Button)findViewById(R.id.orderButton);
packageButton=(Button)findViewById(R.id.packageButton);
cleanButton.setOnClickListener(this);
orderButton.setOnClickListener(this);
packageButton.setOnClickListener(this);
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initView();
}
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
<span id="9_nwp" style="padding: 0px; width: auto; height: auto; float: none;"><a target=_blank id="9_nwl" >switch</span></a></span>(v.getId())
{
case R.id.cleanButton:
cleanAction();
break;
case R.id.orderButton:
orderAction();
break;
case R.id.packageButton:
packageAction();
break;
default:
break;
}
}
private void cleanAction()
{
startAction(ROOM_SERVICE_ACTION,CLEAN_SERVICE);
}
private void orderAction()
{
startAction(ROOM_SERVICE_ACTION,ORDER_SERVICE);
}
private void packageAction()
{
startAction(ROOM_SERVICE_ACTION,PACKAGE_SERVICE);
}
private void startAction(String actionName,int serviceFlag)
{
//Intent intent=new Intent(this,RoomService.class);
Intent intent=new Intent();
intent.setAction(actionName);
intent.putExtra(SERVICE_KEY, serviceFlag);
this.startService(intent);
}
private void showLongToast(String info)
{
Toast.makeText(getBaseContext(), info, Toast.LENGTH_LONG).show();
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is p<span id="10_nwp" style="padding: 0px; width: auto; height: auto; float: none;"><a target=_blank id="10_nwl" >res</span></a></span>ent.
getMenuInflater().inflate(R.menu.main, menu);
return true;
}
@Override
protected void onDestroy() {
// TODO Auto-generated method stub
super.onDestroy();
}
}
運行結果如下:
顯然,此時客戶端順利獲取了服務。
上面舉的是跨應用的例子,如果是在同一個應用的不同進程的話,則有更簡單的實現方法。
RoomService的代碼如下:
<<span id="6_nwp" style="padding: 0px; width: auto; height: auto; float: none;"><a target=_blank id="6_nwl" >package com.android.service;
import com.android.actions.Actions;
import android.app.Service;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.os.IBinder;
import android.os.RemoteException;
import android.util.Log;
import android.widget.Toast;
public class RoomService extends Service{
private static final String TAG="RoomService";
@Override
public void onStart(Intent intent, int startId) {
showLog("onStart");
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
//String action=intent.getAction();
Log.i(TAG,"onStartCommand");
String action=intent.getAction();
if(Actions.CLEAN_ACTION.equals(action))
{
showShortToast("Start Clean Service Right Now");
}
else if(Actions.ORDER_ACTION.equals(action))
{
showShortToast("Start Order Service Right Now");
}
else if(Actions.PACKAGE_ACTION.equals(action))
{
showShortToast("Start Package Service Right Now");
}
else
{
showShortToast("Wrong action");
}
return super.onStartCommand(intent, flags, startId);
}
private void showLog(String info)
{
Log.i(TAG,info);
}
private void showShortToast(String info)
{
Toast.makeText(getBaseContext(), info, Toast.LENGTH_SHORT).show();
}
@Override
public void onDestroy() {
// TODO Auto-generated method stub
showLog("onDestroy");
super.onDestroy();
}
@Override
public void onCreate() {
// TODO Auto-generated method stub
showLog("onCreate");
super.onCreate();
}
@Override
public IBinder onBind(Intent intent) {
showLog("onBind");
return null;
}
@Override
public boolean onUnbind(Intent intent) {
showLog("onUnbind");
return super.onUnbind(intent);
}
}</<span id="8_nwp" style="padding: 0px; width: auto; height: auto; float: none;"><a target=_blank id="8_nwl" >span</span></a></span>>
MainActivity的代碼如下:
package com.android.activity;
import com.android.activity.R;
import com.android.service.RoomService;
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.Menu;
import android.widget.Button;
import android.widget.Toast;
import android.view.View;
import com.android.actions.Actions;
public class MainActivity extends Activity implements View.OnClickListener{
private static final String TAG="MainActivity";
private static final String SERVICE_KEY="ServiceName";
private Button cleanButton;
private Button orderButton;
private Button packageButton;
private void initView()
{
cleanButton=(Button)findViewById(R.id.cleanButton);
orderButton=(Button)findViewById(R.id.orderButton);
packageButton=(Button)findViewById(R.id.packageButton);
cleanButton.setOnClickListener(this);
orderButton.setOnClickListener(this);
packageButton.setOnClickListener(this);
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initView();
}
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
<span id="4_nwp" style="padding: 0px; width: auto; height: auto; float: none;"><a target=_blank id="4_nwl" >switch</span></a></span>(v.getId())
{
case R.id.cleanButton:
cleanAction();
break;
case R.id.orderButton:
orderAction();
break;
case R.id.packageButton:
packageAction();
break;
default:
break;
}
}
private void cleanAction()
{
startAction(Actions.CLEAN_ACTION);
}
private void orderAction()
{
startAction(Actions.ORDER_ACTION);
}
private void packageAction()
{
startAction(Actions.PACKAGE_ACTION);
}
private void startAction(String actionName)
{
Intent intent=new Intent(this,RoomService.class);
intent.setAction(actionName);
this.startService(intent);
}
private void showLongToast(String info)
{
Toast.makeText(getBaseContext(), info, Toast.LENGTH_LONG).show();
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is p<span id="5_nwp" style="padding: 0px; width: auto; height: auto; float: none;"><a target=_blank id="5_nwl" >res</span></a></span>ent.
getMenuInflater().inflate(R.menu.main, menu);
return true;
}
@Override
protected void onDestroy() {
// TODO Auto-generated method stub
super.onDestroy();
}
}
從打印的log可看出,客戶端每調用一次Context.startService(Intent),Service就會重新執行一次onStartCommand---->onStart;但是使用AIDL的話,綁定服務之后,不會重復執行onStart,顯然后者的代價更小。
Service: 前臺 Service,像我們經常用的天氣、音樂其實都利用了前臺Service來實現
來自:http://blog.csdn.net/chaoyu168/article/details/56478837