Android中进程间通信的几种方式(学习中的笔记)

  1. Activity (借助Intent调用其他APP的Activity实现跨进程通信)
    • Android四大组件之一
    • 被调用方代码段
        Intent i = getIntent();
        if(null != i){
            System.out.println("被调用的Activity中获取数据:"+i.getIntExtra("id", 0));
        }

清单文件中的Activity中添加如下属性(如果有intentfilter应该默认就是true)
android:exported="true"
- 调用方的代码段

  // 参数1:包名;参数2:包名+类名
        ComponentName componentName= new ComponentName("cn.cc.testandroid", "cn.cc.testandroid.TesttwoActivity");
        Intent intent = new Intent();
        intent.putExtra("id", 9999);
        intent.setComponent(componentName);
        startActivity(intent);
  1. 广播接收者(BroadcastReceiver)
    • Android四大组件之一,有广播接收者,与之对应要有发送者。
    • 广播:存在1个数据的发送方,和若干个接收方。(理解为:某个电台发送广播,在此接收频率上的收音机都可以接得收到)
    • 在Android系统中,把符合“广播”的特性的数据传递方式称之为“广播”,“广播”是一种在Android系统中全设备通信的机制。
    • 广播分类:普通广播(无序广播)、有序广播、粘滞普通(无序)广播、粘滞有序广播。
    • 广播发送:在Activity或Service中调用sendBroadcast(Intent)方法即可发普通广播,或调用sendOrderedBroadcast(Intent, String)发送有序广播。
    • 广播接收者注册方式:静态注册(清单文件)、动态注册(registerReceiver,请在合适时机调用unregisterReceiver取消注册)。
      接收者:
// 新建类继承BroadcastReceiver ,重写onReceive
public class CustomReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Bundle bundle = intent.getExtras();
        Log.i("CustomReceiver", "onReceive:"+bundle.getInt("id"));
    }
}

清单文件注册:

        <!--  注册广播接收者 action可以理解为频道号  priority为优先级数,
                值越大优先级别越高,取值范围:-1000到1000,
                如需要提高级别请使用动态注册 -->
        <receiver 
            android:name="cn.cc.testandroid.CustomReceiver">
            <intent-filter android:priority="999">
                <action android:name="cctv"/>
            </intent-filter>
        </receiver>

发送者:

        Intent intent = new Intent();
        intent.setAction("cctv");
        Bundle bundle = new Bundle();
        bundle.putInt("id", 8888);
        intent.putExtras(bundle);
        sendBroadcast(intent);
  1. 内容提供者(ContentProvider)
    • Android四大组件之一,有提供者,与之对应也要有调用者。
    • ContentProvider是一种设备内部共享数据的机制,APP可以通过ContentProvider将自身应用的数据对外提供共享,使得其它应用可以对这些数据实现增删改查的访问功能。
    • Android系统使用了许多ContentProvider将系统中的数据对外提供共享,任何APP都可以通过访问这些ContentProvider对相关的数据进行操作。
      内容提供者:
public class CustomProvider extends ContentProvider {// 新建类继承ContentProvider并实现需实现的方法
    private static UriMatcher MATCHER;// UriMatcher对象
    private static final int MATCH_ROOT = 1;
    static {
        MATCHER = new UriMatcher(UriMatcher.NO_MATCH);
        // 添加匹配规则 content://www.cc.cn
        MATCHER.addURI("www.cc.cn", null, MATCH_ROOT);
    }
    public Cursor query(Uri uri, String[] projection, String selection,
            String[] selectionArgs, String sortOrder) {
        Log.i("CustomProvider", "CustomProvider->query()...");
        return null;
    }//此处还有其他需要实现的方法,篇幅原因就不贴了,都未作处理。默认就好,只是简单实现query方法调用。
}

清单文件中注册:

        <!-- authorities:Uri的组成部分,其他app需要此才能访问(可理解为地址) -->
        <provider
            android:name="cn.cc.testandroid.CustomProvider"
            android:authorities="www.cc.cn"
            android:exported="true" >
        </provider>

内容调用者:

        ContentResolver cr = getContentResolver();
        Uri uri = Uri.parse("content://www.cc.cn"); 
        cr.query(uri, null, null, null, null);// 此处返回游标...
  1. AIDL(Android Interface definition language,Android接口定义语言)

    • 与之前三个相比,AIDL可能稍显复杂。
    • Service,Android四大组件之一,借助AIDL也可以实现跨进程访问服务
    • 下面实现跨进程访问Service,两个项目:TestAIDLServer是服务端的项目,TestAndroid2是客户端的项目,服务端安装后,客户端可以访问服务端的Service。客户端展示图如下:


      Paste_Image.png

      点击远程访问按钮后:


      Paste_Image.png
    • AIDL实现步骤(实体类为 Goods:货物):
      1) 服务端新建包aidl,并在包内自定义实体类Goods.java,让其实现Parcelable接口,实现Parcelable的详细步骤...

    Paste_Image.png

    ** 2)服务端相同包下**,创建Goods.aidl文件添加如下代码
    parcelable Goods;// 如果忘记添加,IAidl.aidl中的import语句可能会报错
    3)服务端相同包下,新建IAidl.aidl文件,添加如下代码

package cn.cc.testaidlserver;// 跟Goods实体类一样
import cn.cc.testaidlserver.Goods;// 手动导入实体类包
interface IServer{
  double getPrice();
  int getCount();
  String getName();
}

至此如果没有错误则Eclipse会在项目的gen目录下自动生成对应的包及其Iserver.java文件

![Paste_Image.png](http://upload-images.jianshu.io/upload_images/3266538-60b6a47a68d1b369.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

4) 服务端另外的包下,新建CustomService继承自Service

Paste_Image.png

CustomService.java代码:
<pre>
public class CustomService extends Service {
@Override
public IBinder onBind(Intent intent) {
return new CustomServiceImpl();
}
class CustomServiceImpl extends IServer.Stub{// 实现类
@Override
public double getPrice() throws RemoteException {
return 999;
}
@Override
public int getCount() throws RemoteException {
return 5;
}
@Override
public String getName() throws RemoteException {
return "地下城堡2-教团长剑x2";
}
}
}
</pre>
清单文件中注册Service

        <service android:name="cn.cc.test.server.CustomService">
            <intent-filter >
                <action android:name="cn.cc.test.server.CustomService"/>
            </intent-filter>
        </service>

5)将服务端的整个aidl包拷贝到客户端中 gen目录下也会自动生成IServier.java文件

Paste_Image.png

客户端剩余的代码比较简单了直接贴
布局文件:
<pre>
<TextView
android:layout_width="match_parent"
android:id="@+id/tv_message"
android:background="#dddddd"
android:layout_height="100dp"
/>
<Button
android:id="@+id/btn_show"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="访问远程Service获取商品名称"
/>
<Button
android:id="@+id/btn_clear"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Clear"
/>
</pre>
MainActivity.java代码:
<pre>
public class MainActivity extends Activity implements OnClickListener {
private IServer server;// 访问的接口
private ServiceConnection conn;
// 控件
private TextView tvMessage;
private Button btnShow;
private Button btnClear;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);

    // 初始化控件
    tvMessage = (TextView) findViewById(R.id.tv_message);
    btnShow = (Button) findViewById(R.id.btn_show);
    btnClear = (Button) findViewById(R.id.btn_clear);
    
    // 设置监听
    btnShow.setOnClickListener(this);
    btnClear.setOnClickListener(this);
    
    // ServiceConnection初始化
    conn = new ServiceConnection() {
        @Override
        public void onServiceDisconnected(ComponentName name) {
            Log.i("==================", "onServiceDisconnected");
        }
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            Log.i("==================", "onServiceConnected");
            // 获取远程service中onBind方法的return的对象
            server = IServer.Stub.asInterface(service);
        }
    };
    
    // 绑定远程Service
    Intent intent = new Intent("cn.cc.test.server.CustomService");
    bindService(intent, conn, Context.BIND_AUTO_CREATE);
}

@Override
public void onClick(View v) {
    switch (v.getId()) {
    case R.id.btn_show:// 获取远程Service信息
        try {
            if(null != server){
                String strRet = server.getName();// 调用远程Service的方法,此处简单调用一个。
                tvMessage.setText(strRet);
            }else{
                Toast.makeText(MainActivity.this, "访问远程Service失败", Toast.LENGTH_SHORT).show();
            }
        } catch (RemoteException e) {
            e.printStackTrace();
        }
        break;
    case R.id.btn_clear:// 清除
        tvMessage.setText("");
        break;
    }
}

}
</pre>
至此,按照步骤,中间没出差错的话已经实现了跨进程访问Service的方法了。

  • 注意事项:
    1)步骤1中最好是新建一个包,往客户端拷贝时直接拷贝整个包,这样不容易出错。
    2)服务端跟客户端的包名要相同
    3)CustomService类中,的onBind方法return的是IServer.Stub抽象类的实现类对象。
    4)MainActivity.java类中的bindService会消耗一定的时间。
  • 以上为学习中加深记忆、克服手懒、方便以后翻阅所用,若碰巧对其他童鞋有所帮助,不甚荣幸。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,133评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,682评论 3 390
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,784评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,508评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,603评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,607评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,604评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,359评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,805评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,121评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,280评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,959评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,588评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,206评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,193评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,144评论 2 352

推荐阅读更多精彩内容