本篇内容:

  1. 如何在Android中创建多进程?查看进程的三种方式有哪些?
  2. 多进程模式的运行机制?- 演示了多进程出现问题中的两种情况:
    •   静态成员失效
    •   Application多次创建
  3. IPC基础概念介绍

1.如何在Android中创建多进程?

AndroidManifes.xml

 <!-- Chapter 2 IPC机制 -->
<activity android:name=".Chapter2.Chapter2"></activity>
<activity
android:name=".Chapter2.SecondActivity"
android:process=":remote"
/>
<activity
android:name=".Chapter2.ThirdActivity"
android:process="com.art.exploration.remote"
/>

入口MainActivity没有指定android:process ,它运行在默认进程中,默认进程的进程名是包名

Chapter2.java

public class Chapter2 extends AppCompatActivity {

    @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_chapter2);
} public void goSecondActivity(View view) {
startActivity(new Intent(this, SecondActivity.class));
} public void goThirdActivity(View view) {
startActivity(new Intent(this, ThirdActivity.class));
}
}

SecondActivity和ThirdActivity均为新建的EmptyActivity,里面只有onCreate方法。

准备好了材料,我们开始做菜。。。

启动流程:通过MainActivity启动Chapter2,在Chapter2(Activity)中分别启动Second和Third ,然后观察系统中的进程:

这里给出了查看系统进程的 三种方式 

1. 在Android Profiler中查看:

2. DDMS中查看(推荐):

3. 通过adb shell :

查看全部 → adb shell ps
Linux → adb shell ps | grep com.art.exploration
Windows → adb shell "ps|grep com.art.exploration" 多了个双引号

由此可见,通过在清单文件中为四大组件(Activity,Service,Receiver,ContentProvider)配置android:process属性,在该应用中开启多线程。

 注意:SecondActivity对应的是进程 com.art.exploration:remote
    ThirdActivity对应的是进程 com.art.exploration.remote

Tips1:SecondActivity和ThirdActivity的android:process属性分别为 ":remote" 和 "com.art.exploration.remote",两者的区别有两方面:

  一、":" 是一种简单的写法,包名+进程名;"com.art.exploration.remote"是完整的命名方式,不需要加入包名;

  二、":"  是当前应用的私有进程,其他应用的组件不可以和它跑在同一进程里;

    "com.art.exploration.remote"是全局进程,其他应用通过ShareUID方式可以和它跑在同一进程里。

ShareUID 

2. 多线程模式的运行机制?

  • 静态成员变化?

我们先新建一个UserManager类:

public class UserManager {
public static int sUserId = 5;
}

我们知道,静态变量是共享的,并且一处修改处处同步。根据该特性,我们做如下演示:

在Chapter2中添加代码:

SecondActivity & ThirdActivity

Demo Process  : MainActivity→Chapter2→SecondActivity→backpress→ThirdActivity:

三个进程:

日志信息:

看到这里,大家应该明白了多线程所带来的问题,绝不是简简单单指定一个android:process属性那么简单。

原因分析:Android为每一个应用分配了一个独立的虚拟机,或者说为每一个进程分配了一个独立的虚拟机。

不同的虚拟机在内存分配上有不同的地址空间,这就导致在不同的虚拟机中访问同一个类的对象会产生多份副本。

拿我们这个例子来说, Chapter2中在默认进程中,修改sUserId只会影响当前的进程,对其他进程不会产生任何影响 ;

而SecondActivity在进程“com.art.exploration:remote”中,使用的是该进程中的UserManager对象副本,并不受Chapter2中的干扰,故sUserId为5,ThirdActivity同理。

 总结:

  • 下面我们演示Application多次创建的情况:

Demo Process :MainActivity→Chapter2→SecondActivity→backpress→ThirdActivity:

MyApplication.java

public class MyApplication extends Application {
@Override
public void onCreate() {
super.onCreate();
String processName = getProccessNameByPID(getApplicationContext(), Process.myPid());
System.out.println("MyApplication processName = " + processName);
} /**
* 获取进程ID
*
* @return
*/
public static int getProcessPID() {
return android.os.Process.myPid();
} /**
* 通过进程id 获取进程名字
*
* @param context
* @param pid
* @return
*/
@TargetApi(Build.VERSION_CODES.N)
public static String getProccessNameByPID(Context context, int pid) {
ActivityManager am = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
List<ActivityManager.RunningAppProcessInfo> runningApps = am.getRunningAppProcesses();
if (runningApps == null) return null;
for (ActivityManager.RunningAppProcessInfo procInfo : runningApps) {
if (procInfo.pid == pid) {
return procInfo.processName;
}
}
return null;
}
}

log日志:

可见,Application的onCreate方法执行了三次,创建了三个Application,并且每个进程名和进程id都不一样。它们的进程名和我们为Activity指定的android:process属性一致。

这也就证明了在多进程模式中,不同进程的组件的确会拥有独立的虚拟机、内存空间以及Application。

Tips:跨进程通信的方式有:通过Intent来传递数据,共享文件和SharedPreferences,基于Binder的Messenger和AIDL,Socket等。

3. IPC基础概念介绍

  • 三方面内容

    Serializable接口、Parcelable接口以及Binder。

  • Serializable接口

注意的东西:1.静态成员变量属于类,不属于对象,不参与序列化过程;

2.使用transient关键字标记的成员变量不参与序列化过程。

  • Parcelable接口

    我们先看个实现P接口的User:

public class User implements Parcelable {

    public int userId;
public String userName;
public boolean isMale; public Book book; @Override
public int describeContents() {
return 0;
} @Override
public void writeToParcel(Parcel dest, int flags) {
dest.writeInt(this.userId);
dest.writeString(this.userName);
dest.writeByte(this.isMale ? (byte) 1 : (byte) 0);
dest.writeParcelable(this.book, flags);
} public User() {
} protected User(Parcel in) {
this.userId = in.readInt();
this.userName = in.readString();
this.isMale = in.readByte() != 0;
this.book = in.readParcelable(Book.class.getClassLoader());
// or
// this.book = in.readParcelable(Thread.currentThread().getContextClassLoader());
} public static final Parcelable.Creator<User> CREATOR = new Parcelable.Creator<User>() {
@Override
public User createFromParcel(Parcel source) {
return new User(source);
} @Override
public User[] newArray(int size) {
return new User[size];
}
};
}
Parcel:Parcel内部包装了可序列化的数据,可以再Binder中自由传输;
writeToParcel:序列化过程由writeToParcel方法负责完成,最终是通过Parcel中的一系列的write方法来完成的;
CREATOR:反序列化功能是由CREATOR来完成,其内部标明了如何创建序列化对象和数组,并通过Parcel的一系列read方法来完成反序列化功能。
describeContents:内容描述功能。“几乎在所有情况下这个方法都应该返回0”,仅当当前对象存在文件描述符时,此方法返回1 。
附:Parcelable的详细方法说明:

注:系统已经为我们提供了一些实现了Parcelable接口的类,它们都是可以直接序列化的,比如Intent、Bundle、Bitmap等,同时List和Map也可以序列化,前提是它们里面的每个元素都是可序列化的。

 
 
												

Android DevArt5:如何在Android中创建多线程?的更多相关文章

  1. 转:MFC中创建多线程

    MFC中创建多线程   MFC的多线程函数必须声明为静态的或者是全局函数(不同的在于全局函数不能访问类的私有静态成员,而静态类函数可以):但这样的线程函数只能访问静态的成员变量,要实现访问类的其他成员 ...

  2. 如何在VMware中创建虚拟机

    今天给大家分享如何在VMware中创建虚拟机,具体的教程如下.在这里小编提前下载了Ubuntu14.04桌面系统,为后面在虚拟机中安装Ubuntu14.04桌面系统做准备. 1.从官网上或者直接百度上 ...

  3. Android Studio 如何在TextView中设置图标并按需调整图标大小

    •任务 相信大家对这张图片都不陌生,没错,就是 QQ动态 向我们展示的界面. 如何实现呢? •添加文字并放入图标 新建一个 Activity,取名为 QQ,Android Studio 自动为我们生成 ...

  4. java中创建多线程的方式

    在java中比较常用的有三种创建多线程的方式. 方式一:继承Thread类,要重写run方法. 在MyThread类 public class MyThread extends Thread { @O ...

  5. 如何在Mac中创建MiniKube

    转载请标明出处: http://blog.csdn.net/forezp/article/details/82563153 本文出自方志朋的博客 这篇文章介绍了如何在Mac系统中创建MiniKube. ...

  6. 如何在ARM中创建Express Route

    很早之前就想试试Azure的express route,但是一直没有找到合适的机会,正好有个客户需要上express route,所以最近先自己研究研究,防止在做poc的时候耗费更多时间,本次场景我们 ...

  7. 如何在github中创建演示demo

    在项目中创建一个新的gh-pages分支: 将你想要展示的示例demo上传到改分支,具体操作就看你自己啦,是merge其他分支的过来还是重新写代码都可以. 最后,通过http://<userna ...

  8. 如何在MFC中创建非矩形button

    一般情况下,我们创建的按钮都是矩形的,但有时为了满足特殊的需求,我们要在对话框中创建一个非矩形的按钮,比如,圆形,椭圆等. 要实现一个非矩形的按钮,这就涉及到了自绘控件.自绘控件的方法有很多,可以参考 ...

  9. PS网页设计教程XXVI——如何在PS中创建一个专业的网页布局

    作为编码者,美工基础是偏弱的.我们可以参考一些成熟的网页PS教程,提高自身的设计能力.套用一句话,“熟读唐诗三百首,不会作诗也会吟”. 本系列的教程来源于网上的PS教程,都是国外的,全英文的.本人尝试 ...

随机推荐

  1. Servlet基础学习

    Servlet学习 Servlet是Server与Applet的缩写,是服务端小程序的意思.使用Java语言编写的服务器端程序,可以像生成动态的WEB页,Servlet主要运行在服务器端,并由服务器调 ...

  2. [SQL]SQL Prompt5的工具栏按钮介绍

    这是SQL Prompt5的工具栏按钮 Refresh Suggestions 刷新提示(数据库更改表结构后,需要刷新一下) Format SQL 格式化代码(Pro版本才能启用,可定制,所有前边带红 ...

  3. HBase脚本命令

    1. 脚本使用小结1.开启集群 start-hbase.sh 2.关闭集群 stop-hbase.sh 3.开启/关闭[所有]的regionserver.zookeeper hbase-daemons ...

  4. 深入理解Apache Flink

    Apache Flink(下简称Flink)项目是大数据处理领域最近冉冉升起的一颗新星,其不同于其他大数据项目的诸多特性吸引了越来越多人的关注.本文将深入分析Flink的一些关键技术与特性,希望能够帮 ...

  5. 基于拖放布局的 Twitter Bootstrap 网站生成器

    简单的几个拖放操作就能做出漂亮的 Twitter Bootstrap 网站?是的,LayoutIt 是一个 Twitter Bootstrap 界面生成器,能够帮助你快速制作出网站和界面模型,同时能够 ...

  6. Android原生和H5交互;Android和H5混合开发;WebView点击H5界面跳转到Android原生界面。

    当时业务的需求是这样的,H5有一个活动商品列表的界面,IOS和Android共用这一个界面,点击商品可以跳转到Android原生的商品详情界面并传递商品ID:  大概就是点击H5界面跳转到Androi ...

  7. nginx完全关闭log

    nginx.conf中要在http一节里面添加 access_log off; error_log off;

  8. NodeJs安装以及注意事项

    1.测试NodeJs是否安装成功 node --version npm -v 配置node的可执行文件路径到环境变量path 2.安装相关环境 npm install express -g npm i ...

  9. vuejs实现瀑布流布局(二)

    瀑布流布局已然完成,那么剩下的就是另一个比较大的工程了——无限加载. 之前说了,这个活动项目是基于SUI-Mobile搭建的,所以可以直接使用sui内建组件“无限加载”来实现这个功能. 没有真实的数据 ...

  10. django总结 --》内容(django建project开始的大致流程、ORM简介)

    1 安装: pip  install django==1.11.9 另外:在pycharm中安装 django,在下图中七步走 2. 新建Django项目  django-admin startpro ...