Android开发网

首页|Android开发环境|Android开发教程|Android开发视频|Android游戏开发|Android开发实例|Android开发书籍|鸡啄米博客

Android应用开发教程之十:应用程序通信

  Android 开发中在程序之间通讯的接口做的还是非常丰富的 本例主要向大家介绍程序之间是如何进行沟通,有哪几种沟通方式 如何来实现沟通。

  1.使用handler传递消息

  handler 大家可以把它想象成主线程(UI线程)的一个子线程,它可以给主线程(UI线程)发送数据从而更新主线程(UI线程)的UI与逻辑,handler 是一个子线程所以它的耗时操作不会阻塞主线程,大家都知道在android的开发中如果代码中某个地方阻塞主线程超过5秒的话系统会提示ANR (系统提示强制关闭)所以在耗时操作上我们可以考虑开启一个子线程避免ANR。  handler会向主线程发送消息 会以队列的形式排列着配合等待主线程更新UI 逻辑 等等。

  下面这个例子诠释了这一点 利用handler传递消息来更新主线程的UI显示内容 点击按钮后每过一秒通过handler发送消息更新UI线程显示的时间 直到显示时间更新到10 然后结束这个线程。

Android应用开发教程之十:应用程序通信

Java代码
  1. public class HandlerActivity extends Activity implements Runnable{  
  2.    
  3.     /**更新时间**/  
  4.     public final static int UPDATE_TIME =0;  
  5.     /**更新时间成功**/  
  6.     public final static int UPDATE_COMPLETED =1;  
  7.    
  8.     /**记录显示时间 超过10秒结束线程**/  
  9.     private int mShowNumber = 0;  
  10.    
  11.     /**开始计时按钮**/  
  12.     private Button mButton = null;  
  13.    
  14.     /**计时显示内容**/  
  15.     private TextView mTextView = null;  
  16.    
  17.     /**线程**/  
  18.     private Thread mThread = null;  
  19.    
  20.     /**线程关闭的标志**/  
  21.     private boolean mRunning = false;  
  22.    
  23.     Handler handler = new Handler() {  
  24.     @Override  
  25.     public void handleMessage(Message msg) {  
  26.    
  27.         Bundle bundle= msg.getData();  
  28.         //通过key的名称拿到它的值  
  29.         String  number = bundle.getString("number");  
  30.         //msg.what为handler接收到的消息编号  
  31.         switch(msg.what) {  
  32.         case UPDATE_TIME:  
  33.         mTextView.setText("正在更新时间" + number);  
  34.         break;  
  35.         case UPDATE_COMPLETED:  
  36.         mTextView.setText("更新完毕");  
  37.         break;  
  38.         }  
  39.         super.handleMessage(msg);  
  40.     }  
  41.     };  
  42.    
  43.     @Override  
  44.     protected void onCreate(Bundle savedInstanceState) {  
  45.     setContentView(R.layout.handler);  
  46.    
  47.     /**拿到button 与  TextView 对象**/  
  48.     mButton = (Button)findViewById(R.id.button0);  
  49.     mTextView = (TextView)findViewById(R.id.textView0);  
  50.     mThread = new Thread(this);  
  51.    
  52.     mButton.setOnClickListener(new OnClickListener() {  
  53.         @Override  
  54.         public void onClick(View arg0) {  
  55.         /**点击按钮后开始线程开始计时**/  
  56.         mRunning = true;  
  57.         mThread.start();  
  58.         }  
  59.     });  
  60.    
  61.     mTextView.setText("点击按钮开始更新时间");  
  62.     super.onCreate(savedInstanceState);  
  63.     }  
  64.    
  65.     public void ShowDialog(String string) {  
  66.     AlertDialog.Builder builder = new AlertDialog.Builder(  
  67.         HandlerActivity.this);  
  68.     builder.setIcon(R.drawable.icon);  
  69.     builder.setTitle(string);  
  70.     builder.setPositiveButton("确定"new DialogInterface.OnClickListener() {  
  71.         public void onClick(DialogInterface dialog, int whichButton) {  
  72.         finish();  
  73.         }  
  74.     });  
  75.     builder.show();  
  76.     }  
  77.    
  78.     @Override  
  79.     public void run() {  
  80.    
  81.     while (mRunning) {  
  82.         try {  
  83.         mShowNumber++;  
  84.         /** 把须要的数据放入bandle中 **/  
  85.         Bundle bandle = new Bundle();  
  86.         bandle.putString("number", String.valueOf(mShowNumber));  
  87.    
  88.         /** 设置这条信息的编号为更新时间 **/  
  89.         /** 将bandle写入message中 **/  
  90.         /** 最后将这个message发送出去 **/  
  91.         /** mShowNumber小于10更新时间 否则更新完毕 **/  
  92.         Message msg = new Message();  
  93.         if(mShowNumber <=10) {  
  94.             msg.what = UPDATE_TIME;  
  95.         }else {  
  96.             mRunning = false;  
  97.             msg.what = UPDATE_COMPLETED;  
  98.         }  
  99.         msg.setData(bandle);  
  100.         handler.sendMessage(msg);  
  101.         Thread.sleep(1000);  
  102.         } catch (InterruptedException e) {  
  103.         e.printStackTrace();  
  104.         }  
  105.     }  
  106.     }  
  107. }  

  2.Notifation通知栏信息

  Notifation通知栏会在屏幕上方向用户提示信息 但是不会打断用户正在阅读的内容,除非用户手动将 Notifation通知栏拉下。 Notifation的好处就是在于不会影响用户的操作,比如用户正在阅读非常重要的信息这时候帮他直接打开一个activity会非常不合适 因为直接影响到了他当时的操作行为 所以Notifation就出来了。建议大家在开发中遇到可能打断用户使用的情况下都去使用Notifation通知栏。

       屏幕上方为弹出的Notifation通知栏

Android应用开发教程之十:应用程序通信

  将Notifation通知栏拉下后会出现相应的信息

Android应用开发教程之十:应用程序通信

Java代码
  1. public class NotificationActivity extends Activity {  
  2.     NotificationManager mManager = null;  
  3.     Notification notification =null;  
  4.     @Override  
  5.     protected void onCreate(Bundle savedInstanceState) {  
  6.     setContentView(R.layout.notification);  
  7.    
  8.     // 得到通知消息的管理器对象,负责管理 Notification 的发送与清除消息等  
  9.     mManager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);  
  10.     // 创建Notification对象 参数分别代表 通知栏 中显示的图标 显示的标题 显示的时间  
  11.     notification = new Notification(R.drawable.jay,  
  12.         "Android专业开发群", System.currentTimeMillis());  
  13.    
  14.     // 设置在通知栏中点击后Notification自动消失  
  15.     notification.flags = Notification.FLAG_AUTO_CANCEL;  
  16.    
  17.     //设置点击后转跳的新activity  
  18.     Intent intent = new Intent(this, MyShowActivity.class);  
  19.     intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP¦ Intent.FLAG_ACTIVITY_NEW_TASK);  
  20.    
  21.     //通过bundle可以带一些数据过去 这里将字符串传递了过去  
  22.     Bundle bundle = new Bundle();  
  23.     bundle.putString("name""从Notification转跳过来的");  
  24.     intent.putExtras(bundle);  
  25.    
  26.     //设置通知栏中显示的内容  
  27.     PendingIntent contentIntent = PendingIntent.getActivity(this,  
  28.         R.string.app_name, intent, PendingIntent.FLAG_UPDATE_CURRENT);  
  29.     notification.setLatestEventInfo(this"Android专业开发群",  
  30.         "QQ群号 164257885", contentIntent);  
  31.    
  32.     Button button0 = (Button)findViewById(R.id.button0);  
  33.     button0.setOnClickListener(new OnClickListener() {  
  34.    
  35.         @Override  
  36.         public void onClick(View arg0) {  
  37.         //打开这个Notification通知  
  38.         mManager.notify(0, notification);  
  39.         }  
  40.     });  
  41.    
  42.     Button button1 = (Button)findViewById(R.id.button1);  
  43.     button1.setOnClickListener(new OnClickListener() {  
  44.    
  45.         @Override  
  46.         public void onClick(View arg0) {  
  47.         //关闭这个Notification通知  
  48.         mManager.cancelAll();  
  49.         }  
  50.     });  
  51.    
  52.     super.onCreate(savedInstanceState);  
  53.     }  
  54.    
  55. }  

  3.广播的发送与接收

  Android开发中如果须要对两个完全没关系的程序之间进行通信 就可以使用发送广播与接收广播的机制来实现 ,例如程序A发送了一个广播 程序B接受到 做一些事情 这样就达到了相互的通讯。

Android应用开发教程之十:应用程序通信

  调用sendBroadcast() 传入intent  后 来发送广播

Java代码
  1. public class BroadcastActivity extends Activity {  
  2.    
  3.     Button mButton0 = null;  
  4.     Button mButton1 = null;  
  5.    
  6.     @Override  
  7.     protected void onCreate(Bundle savedInstanceState) {  
  8.     setContentView(R.layout.broadcast);  
  9.    
  10.     mButton0 = (Button)findViewById(R.id.button0);  
  11.     mButton0.setOnClickListener(new OnClickListener() {  
  12.    
  13.         @Override  
  14.         public void onClick(View arg0) {  
  15.                 Intent intent = new Intent(MyService.SEND_OK_MESSAGE);  
  16.                 intent.putExtra("name""您发送了OK这条广播哦");  
  17.                 sendBroadcast(intent);  
  18.         }  
  19.     });  
  20.    
  21.     mButton1 = (Button)findViewById(R.id.button1);  
  22.     mButton1.setOnClickListener(new OnClickListener() {  
  23.    
  24.         @Override  
  25.         public void onClick(View arg0) {  
  26.                 Intent intent = new Intent(MyService.SEND_CANCLE_MESSAGE);  
  27.                 intent.putExtra("name""您发送了Cancle这条广播哦");  
  28.                 sendBroadcast(intent);  
  29.         }  
  30.     });  
  31.    
  32.     //启动Service  
  33.     Intent i = new Intent(this, MyService.class);  
  34.     startService(i);  
  35.     super.onCreate(savedInstanceState);  
  36.     }  
  37. }  

  接收广播的话 我们开启一个service 在service中通过BroadcastReceiver 来接收广播 前提是须要接收的广播须要在onStart()中注册一下 在AndroidManifest.xml中可以过滤只接收须要接收的广播。

XML/HTML代码
  1. <service android:name=".MyService">  
  2.     <intent-filter>  
  3.         <action android:name="cn.m15.xys.MyService"></action>  
  4.     </intent-filter>  
  5.     <intent-filter>  
  6.         <action android:name="send.ok.message" />  
  7.         <action android:name="send.cancle.message" />  
  8.     </intent-filter>  
  9. </service>  

  在onStart()中注册了程序中所需要的两个广播

Java代码
  1. public class MyService extends Service {  
  2.    
  3.     public final static String SEND_OK_MESSAGE = "send.ok.message";  
  4.     public final static String SEND_CANCLE_MESSAGE = "send.cancle.message";  
  5.    
  6.     private BroadcastReceiver myBroadCast = new BroadcastReceiver() {  
  7.    
  8.     @Override  
  9.     public void onReceive(Context context, Intent intent) {  
  10.         String action = intent.getAction();  
  11.         if (action.equals(SEND_OK_MESSAGE)) {  
  12.         Toast.makeText(context, "接收到了一条广播为" + SEND_OK_MESSAGE, Toast.LENGTH_LONG).show();  
  13.         }else if(action.equals(SEND_CANCLE_MESSAGE)) {  
  14.         Toast.makeText(context, "接收到了一条广播为" + SEND_CANCLE_MESSAGE, Toast.LENGTH_LONG).show();  
  15.         }  
  16.     }  
  17.    
  18.     };  
  19.    
  20.     @Override  
  21.     public void onCreate() {  
  22.     super.onCreate();  
  23.     }  
  24.    
  25.     @Override  
  26.     public void onStart(Intent intent, int startId) {  
  27.     //注册这两个广播  
  28.     IntentFilter myFilter = new IntentFilter();  
  29.     myFilter.addAction(SEND_OK_MESSAGE);  
  30.     myFilter.addAction(SEND_CANCLE_MESSAGE);  
  31.     this.registerReceiver(myBroadCast, myFilter);  
  32.         super.onStart(intent, startId);  
  33.     }  
  34.     @Override  
  35.     public IBinder onBind(Intent arg0) {  
  36.     return null;  
  37.     }  
  38.    
  39. }  

  这里注意一下 service如果没有起来 我们是接收不到广播的 所以一定要保证接收的时候service是开启的,上例中的service是在打开activity时开启的 但是如果用户把手机关掉然后在开机 , 这样的话service就不是打开状态 这样就非常危险了因为这时scrvice就接收不到任何消息了除非用户再次进activity 才会帮他打开scrvice 所以我们可以在用户开机后就直接将scrvice打开,具体的实现方式如下

  在AndroidManifest.xml中注册一个开机广播  这个广播系统只会在开机发出而且只会发出一次 所以我们接收这个广播就可以知道手机是否为开机状态

XML/HTML代码
  1. <receiver android:name=".MyBootReceiver" >  
  2.     <intent-filter>  
  3.        <action android:name="android.intent.action.BOOT_COMPLETED" />  
  4.    </intent-filter>  
  5. </receiver>  

  注意加入权限

XML/HTML代码
  1. <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />  

  在BroadcastRecevier中接收开机广播  然后打开service 就可以实现开机启动service。

Java代码
  1. public class MyBootReceiver extends BroadcastReceiver {  
  2.    /**开机广播**/  
  3.     static final String BOOT_COMPLETED = "android.intent.action.BOOT_COMPLETED";  
  4.    
  5.     @Override  
  6.     public void onReceive(Context context, Intent intent) {  
  7.     /**如果为开机广播则开启service**/  
  8.     if (intent.getAction().equals(BOOT_COMPLETED)) {  
  9.         Intent i = new Intent(context, MyService.class);  
  10.         context.startService(i);  
  11.     }  
  12.    
  13.     }  
  14. }  

  3.Activity与Activity之间的转跳

  在软件应用的开发中肯定会有多个Activity     这样它们之间就会存在相互转跳的关系      转跳的实现方式还是使用Intent  然后startActivity  ,当然转跳的话是可以带数据过去的。比如从A跳到B 可以把A中的一些数据通过Intent传递给B 。

Android应用开发教程之十:应用程序通信

  读下面这段代码 大家会发现intent与bandle 传递数值的方式基本一样为什么还要分成两个呢? 确实他们两个传递的数值的方式非常类似, 他们两个的区别就是Intent属于把零散的数据传递过去 而bundle则是把零散的数据先放入bundle 然后在传递过去。我举一个例子 比如我们现在有3个activity  A.B.C  须要把A的数据穿给B然后在穿给C ,如果使用intent一个一个传递 须要在A类中一个一个传递给B 然后B类中获取到所有数值  然后在一个一个传递给C  这样很麻烦 但是 如果是bundle的话 B类中直接将bundler传递给C 不用一个一个获得具体的值  然后在C类中直接取得解析数值。

  传递

Java代码
  1. /**Activity之间传递值**/  
  2.         Button botton3 = (Button)findViewById(R.id.button3);  
  3.         botton3.setOnClickListener(new OnClickListener() {  
  4.    
  5.         @Override  
  6.         public void onClick(View arg0) {  
  7.          Intent intent = new Intent(mContext,ShowActivity.class);  
  8.          //使用intent.putExtra()直接传递  
  9.          intent.putExtra("name""雨松MOMO");  
  10.          intent.putExtra("age"25);  
  11.          intent.putExtra("boy"true);  
  12.    
  13.          //把数值放进bundle 然后在把整个bundle通过intent.putExtra()传递  
  14.          Bundle bundle = new Bundle();  
  15.          bundle.putString("b_name""小可爱");  
  16.          bundle.putInt("b_age"23);  
  17.          bundle.putBoolean("b_boy"false);  
  18.          //在这里把整个bundle 放进intent中  
  19.          intent.putExtras(bundle);  
  20.          //开启一个新的 activity 将intent传递过去  
  21.          startActivity(intent);  
  22.         }  
  23.     });  

  接收

Java代码
  1. public class ShowActivity extends Activity {  
  2.    
  3.     @Override  
  4.     protected void onCreate(Bundle savedInstanceState) {  
  5.     setContentView(R.layout.my);  
  6.    
  7.     Intent intent = getIntent();  
  8.    
  9.     String name = intent.getStringExtra("name");  
  10.     //第二个参数为默认值 意思就是如果在intent中拿不到的话  
  11.     //就用默认值  
  12.     int age  = intent.getIntExtra("age"0);  
  13.     boolean isboy = intent.getBooleanExtra("boy"false);  
  14.     TextView textView0 = (TextView)findViewById(R.id.text0);  
  15.    
  16.     textView0.setText("姓名  " + name + "年龄 " + age + "男孩?  " + isboy);  
  17.    
  18.     Bundle bundle = intent.getExtras();  
  19.     name = bundle.getString("b_name");  
  20.     //第二个参数为默认值 意思就是如果在bundle中拿不到的话  
  21.     //就用默认值  
  22.     age = bundle.getInt("b_age",0);  
  23.     isboy = bundle.getBoolean("b_boy"false);  
  24.    
  25.     TextView textView1 = (TextView)findViewById(R.id.text1);  
  26.    
  27.     textView1.setText("姓名  " + name + "年龄 " + age + "男孩?  " + isboy);  
  28.    
  29.     super.onCreate(savedInstanceState);  
  30.     }  
  31.    
  32. }  

  源码下载地址:http://vdisk.weibo.com/s/aa3fI

Tags:Handler,BroadcastReceiver,Notification | 2016/4/20 | 发表评论

相关文章: