ServiceとActivityの間の通信のいくつかの方法

11901 ワード

転載は住所を明記してください.http://blog.csdn.net/xiaanming/article/details/9750689
Androidでは、Activityは主にフロントページの展示を担当しています.Serviceは主に長期運行の任務を担当しています.だから、私達が実際に開発している間に、ActivityとServiceとの通信によく会います.私たちServiceがいくつかの操作を実行した後、UIスレッドを更新したいですが、どうすればいいですか?次に、ServiceとActivityの間の通信問題を実現するための二つの方法を紹介します.
  • Binderオブジェクト
  • を介して
    ActivityがbindServiceを呼び出すことによって、ServiceConnection conn,int flagsを得ることができます.Serviceのオブジェクトの一例を得ることができます.そしてServiceにアクセスできる方法は、例を通して理解してみましょう.
    まず、私たちは新しいプロジェクトのCommunicationを作って、Serviceクラスを作ります.
    package com.example.communication;
    
    import android.app.Service;
    import android.content.Intent;
    import android.os.Binder;
    import android.os.IBinder;
    
    public class MsgService extends Service {
    	/**
    	 *        
    	 */
    	public static final int MAX_PROGRESS = 100;
    	/**
    	 *        
    	 */
    	private int progress = 0;
    
    	/**
    	 *   get()  , Activity  
    	 * @return     
    	 */
    	public int getProgress() {
    		return progress;
    	}
    
    	/**
    	 *       ,       
    	 */
    	public void startDownLoad(){
    		new Thread(new Runnable() {
    			
    			@Override
    			public void run() {
    				while(progress < MAX_PROGRESS){
    					progress += 5;
    					try {
    						Thread.sleep(1000);
    					} catch (InterruptedException e) {
    						e.printStackTrace();
    					}
    					
    				}
    			}
    		}).start();
    	}
    
    
    	/**
    	 *     Binder  
    	 */
    	@Override
    	public IBinder onBind(Intent intent) {
    		return new MsgBinder();
    	}
    	
    	public class MsgBinder extends Binder{
    		/**
    		 *     Service   
    		 * @return
    		 */
    		public MsgService getService(){
    			return MsgService.this;
    		}
    	}
    
    }
    
    上のコードは比較的簡単で、注釈も比較的に詳しくて、一番基本的なServiceの応用は分かります.startDown Load()方法を使ってダウンロードタスクをシミュレーションして、毎秒更新します.しかし、これはバックグラウンドで行われています.私達は見えないので、時には彼がフロントでダウンロードの進捗状況を表示する必要があります.私たちはこれからActivityを使います.
    Intent intent = new Intent("com.example.communication.MSG_ACTION");  
    bindService(intent, conn, Context.BIND_AUTO_CREATE);
    上記のコードを通して、ActivityでServiceをバインドしました.上にServiceConnectionオブジェクトが必要です.これはインターフェイスです.ここでは匿名の内部カテゴリを使っています.
    	ServiceConnection conn = new ServiceConnection() {
    		
    		@Override
    		public void onServiceDisconnected(ComponentName name) {
    			
    		}
    		
    		@Override
    		public void onServiceConnected(ComponentName name, IBinder service) {
    			//    MsgService  
    			msgService = ((MsgService.MsgBinder)service).getService();
    			
    		}
    	};
    OServiceConnect(ComponentName name,IBinder service)フィードバック方法では、MsgServiceのBinderオブジェクトを返しました.get Service()方法でMsgServiceオブジェクトを得ることができます.その後、MsgServiceのいくつかの方法を呼び出すことができます.Activityコードは以下の通りです.
    package com.example.communication;
    
    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.view.View;
    import android.view.View.OnClickListener;
    import android.widget.Button;
    import android.widget.ProgressBar;
    
    public class MainActivity extends Activity {
    	private MsgService msgService;
    	private int progress = 0;
    	private ProgressBar mProgressBar;
    	
    
    	@Override
    	protected void onCreate(Bundle savedInstanceState) {
    		super.onCreate(savedInstanceState);
    		setContentView(R.layout.activity_main);
    		
    		
    		//  Service
    		Intent intent = new Intent("com.example.communication.MSG_ACTION");
    		bindService(intent, conn, Context.BIND_AUTO_CREATE);
    		
    		
    		mProgressBar = (ProgressBar) findViewById(R.id.progressBar1);
    		Button mButton = (Button) findViewById(R.id.button1);
    		mButton.setOnClickListener(new OnClickListener() {
    			
    			@Override
    			public void onClick(View v) {
    				//    
    				msgService.startDownLoad();
    				//    
    				listenProgress();
    			}
    		});
    		
    	}
    	
    
    	/**
    	 *     ,       MsgService getProgress()       ,  UI
    	 */
    	public void listenProgress(){
    		new Thread(new Runnable() {
    			
    			@Override
    			public void run() {
    				while(progress < MsgService.MAX_PROGRESS){
    					progress = msgService.getProgress();
    					mProgressBar.setProgress(progress);
    					try {
    						Thread.sleep(1000);
    					} catch (InterruptedException e) {
    						e.printStackTrace();
    					}
    				}
    				
    			}
    		}).start();
    	}
    	
    	ServiceConnection conn = new ServiceConnection() {
    		@Override
    		public void onServiceDisconnected(ComponentName name) {
    			
    		}
    		
    		@Override
    		public void onServiceConnected(ComponentName name, IBinder service) {
    			//    MsgService  
    			msgService = ((MsgService.MsgBinder)service).getService();
    			
    		}
    	};
    
    	@Override
    	protected void onDestroy() {
    		unbindService(conn);
    		super.onDestroy();
    	}
    
    
    }
    
    実は上のコードはまだ疑問です.つまり、進行状況の変化を監督する方法は直接スレッドの中でUIを更新するものです.他のスレッドでUI操作ができないということではないですか?
    上のコードはServiceでUIを更新する操作を完了しましたが、見つけられましたか?Serviceで進捗が変化したら、Activityに自主的に通知する方法がありますか?答えは肯定的です.私達はコールバックインターフェースを利用してServiceのアクティブなお知らせを実現できます.コールバックの方法が分かりません.確認してみてください.http://blog.csdn.net/xiaanming/article/details/8703708
    新しいコールバックインターフェースを作成します.
    public interface OnProgressListener {
    	void onProgress(int progress);
    }
    MsgServiceのコードはいくつかの小さな変化があります.皆さんが分かりやすいように、コードを全部貼り付けます.
    package com.example.communication;
    
    import android.app.Service;
    import android.content.Intent;
    import android.os.Binder;
    import android.os.IBinder;
    
    public class MsgService extends Service {
    	/**
    	 *        
    	 */
    	public static final int MAX_PROGRESS = 100;
    	/**
    	 *        
    	 */
    	private int progress = 0;
    	
    	/**
    	 *          
    	 */
    	private OnProgressListener onProgressListener;
    	
    	
    	/**
    	 *          ,     
    	 * @param onProgressListener
    	 */
    	public void setOnProgressListener(OnProgressListener onProgressListener) {
    		this.onProgressListener = onProgressListener;
    	}
    
    	/**
    	 *   get()  , Activity  
    	 * @return     
    	 */
    	public int getProgress() {
    		return progress;
    	}
    
    	/**
    	 *       ,       
    	 */
    	public void startDownLoad(){
    		new Thread(new Runnable() {
    			
    			@Override
    			public void run() {
    				while(progress < MAX_PROGRESS){
    					progress += 5;
    					
    					//           
    					if(onProgressListener != null){
    						onProgressListener.onProgress(progress);
    					}
    					
    					try {
    						Thread.sleep(1000);
    					} catch (InterruptedException e) {
    						e.printStackTrace();
    					}
    					
    				}
    			}
    		}).start();
    	}
    
    
    	/**
    	 *     Binder  
    	 */
    	@Override
    	public IBinder onBind(Intent intent) {
    		return new MsgBinder();
    	}
    	
    	public class MsgBinder extends Binder{
    		/**
    		 *     Service   
    		 * @return
    		 */
    		public MsgService getService(){
    			return MsgService.this;
    		}
    	}
    
    }
    
    Activityのコードは以下の通りです.
    package com.example.communication;
    
    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.view.View;
    import android.view.View.OnClickListener;
    import android.widget.Button;
    import android.widget.ProgressBar;
    
    public class MainActivity extends Activity {
    	private MsgService msgService;
    	private ProgressBar mProgressBar;
    	
    
    	@Override
    	protected void onCreate(Bundle savedInstanceState) {
    		super.onCreate(savedInstanceState);
    		setContentView(R.layout.activity_main);
    		
    		
    		//  Service
    		Intent intent = new Intent("com.example.communication.MSG_ACTION");
    		bindService(intent, conn, Context.BIND_AUTO_CREATE);
    		
    		
    		mProgressBar = (ProgressBar) findViewById(R.id.progressBar1);
    		Button mButton = (Button) findViewById(R.id.button1);
    		mButton.setOnClickListener(new OnClickListener() {
    			
    			@Override
    			public void onClick(View v) {
    				//    
    				msgService.startDownLoad();
    			}
    		});
    		
    	}
    	
    
    	ServiceConnection conn = new ServiceConnection() {
    		@Override
    		public void onServiceDisconnected(ComponentName name) {
    			
    		}
    		
    		@Override
    		public void onServiceConnected(ComponentName name, IBinder service) {
    			//    MsgService  
    			msgService = ((MsgService.MsgBinder)service).getService();
    			
    			//                
    			msgService.setOnProgressListener(new OnProgressListener() {
    				
    				@Override
    				public void onProgress(int progress) {
    					mProgressBar.setProgress(progress);
    					
    				}
    			});
    			
    		}
    	};
    
    	@Override
    	protected void onDestroy() {
    		unbindService(conn);
    		super.onDestroy();
    	}
    
    
    }
    
    フィードバックインターフェースを使うのはもっと便利ですか?進捗が変化したら、ServiceからActivityに自動的に通知します.ActivityはUI操作を更新できます.
  • broadcast(ブロードキャスト)による形式
  • 私たちの進捗状況が変化した時に、ラジオを送ります.そしてActivityの登録放送受信機で、放送を受信した後、PrograessBarを更新します.コードは下記の通りです.
    package com.example.communication;
    
    import android.app.Activity;
    import android.content.BroadcastReceiver;
    import android.content.Context;
    import android.content.Intent;
    import android.content.IntentFilter;
    import android.os.Bundle;
    import android.view.View;
    import android.view.View.OnClickListener;
    import android.widget.Button;
    import android.widget.ProgressBar;
    
    public class MainActivity extends Activity {
    	private ProgressBar mProgressBar;
    	private Intent mIntent;
    	private MsgReceiver msgReceiver;
    	
    
    	@Override
    	protected void onCreate(Bundle savedInstanceState) {
    		super.onCreate(savedInstanceState);
    		setContentView(R.layout.activity_main);
    		
    		//         
    		msgReceiver = new MsgReceiver();
    		IntentFilter intentFilter = new IntentFilter();
    		intentFilter.addAction("com.example.communication.RECEIVER");
    		registerReceiver(msgReceiver, intentFilter);
    		
    		
    		mProgressBar = (ProgressBar) findViewById(R.id.progressBar1);
    		Button mButton = (Button) findViewById(R.id.button1);
    		mButton.setOnClickListener(new OnClickListener() {
    			
    			@Override
    			public void onClick(View v) {
    				//    
    				mIntent = new Intent("com.example.communication.MSG_ACTION");
    				startService(mIntent);
    			}
    		});
    		
    	}
    
    	
    	@Override
    	protected void onDestroy() {
    		//    
    		stopService(mIntent);
    		//    
    		unregisterReceiver(msgReceiver);
    		super.onDestroy();
    	}
    
    
    	/**
    	 *      
    	 * @author len
    	 *
    	 */
    	public class MsgReceiver extends BroadcastReceiver{
    
    		@Override
    		public void onReceive(Context context, Intent intent) {
    			//    ,  UI
    			int progress = intent.getIntExtra("progress", 0);
    			mProgressBar.setProgress(progress);
    		}
    		
    	}
    
    }
    
    package com.example.communication;
    
    import android.app.Service;
    import android.content.Intent;
    import android.os.IBinder;
    
    public class MsgService extends Service {
    	/**
    	 *        
    	 */
    	public static final int MAX_PROGRESS = 100;
    	/**
    	 *        
    	 */
    	private int progress = 0;
    	
    	private Intent intent = new Intent("com.example.communication.RECEIVER");
    	
    
    	/**
    	 *       ,       
    	 */
    	public void startDownLoad(){
    		new Thread(new Runnable() {
    			
    			@Override
    			public void run() {
    				while(progress < MAX_PROGRESS){
    					progress += 5;
    					
    					//  Action com.example.communication.RECEIVER   
    					intent.putExtra("progress", progress);
    					sendBroadcast(intent);
    					
    					try {
    						Thread.sleep(1000);
    					} catch (InterruptedException e) {
    						e.printStackTrace();
    					}
    					
    				}
    			}
    		}).start();
    	}
    
    	
    
    	@Override
    	public int onStartCommand(Intent intent, int flags, int startId) {
    		startDownLoad();
    		return super.onStartCommand(intent, flags, startId);
    	}
    
    
    
    	@Override
    	public IBinder onBind(Intent intent) {
    		return null;
    	}
    
    
    }
    
    まとめ:
  • ActivityがbindService(Intent service,ServiceConnection conn,int flags)方法を呼び出し、Serviceオブジェクトの参照を得ることで、Activityが直接Serviceに呼び出すことができます.Activityに能動的に通知するなら、フィードバック方式
  • を利用することができます.
  • ServiceはActivityにメッセージを送信し、ブロードキャストを使用することができ、もちろんActivityは対応する受信機を登録しなければならない.例えば、Serviceが複数のActivityに同じメッセージを送るなら、この方法がいいです.