详解App相互唤醒的几种方式

下文皆使用Client表示操作的App,Server表示需要被唤起的远端App,Server的包名为“com.jxx.server”

1. ComponentName

使用ComponentName唤起Server步骤很简单,需要注意的是Server的Activity需要在manifest配置种设置exported为true

Server的配置如下:

<activity android:name="com.jxx.server.ServerActivity"
 android:exported="true"/>   

Client调用如下:

Intent intent1 = new Intent();
ComponentName componentName = new ComponentName("com.jxx.server", "com.jxx.server.ServerActivity");
intent1.setComponent(componentName);
intent1.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent1);    

Intent中添加ComponentName还有另外一种写法

Intent intent2 = new Intent();
intent2.setClassName("jxx.com.server", "jxx.com.server.MainActivity");
intent2.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent1); 

只不过setClassName内部帮我们new ComponentName的实例

public @NonNull Intent setClassName(@NonNull String packageName, @NonNull String className) {
 mComponent = new ComponentName(packageName, className);
 return this;
}

既然是用Intent来唤起Activity,那就能使用Intent的特性,例如使用Bundle传递数据

Intent intent1 = new Intent();
ComponentName componentName = new ComponentName("com.jxx.server", "com.jxx.server.ServerActivity");
intent1.setComponent(componentName); 

Bundle bundle1 = new Bundle();
bundle1.putString("remote_invoke", "from_client");
intent1.putExtras(bundle1);      

intent1.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent1);

在Server端提取数据也很简单

Bundle bundle = getIntent().getExtras();

2. 隐式跳转,Uri

Android中唤起拨号页面是这样的

Intent intent = new Intent(Intent.ACTION_CALL,Uri.parse("tel:" + phoneNumber));
startActivity(intent);

其实就是用Uri的形式唤起Server,并传递数据,我们来自己实现一下。 这种方式下,Server端的配置如下,必须添加要有action、data以及category:

<activity android:name=".SecondActivity">        

 <intent-filter>
  <action android:name="com.jxx.server.ServerActivity" />
  <data
   android:host="com.jxx.server"
   android:scheme="ServerActivity" />
  <category android:name="android.intent.category.DEFAULT" />
 </intent-filter>            </activity>

Client调用:

Intent intent2 = new Intent("com.jxx.server.ServerActivity");
Uri uri = Uri.parse("ServerActivity://com.jxx.server?remote_invoke=from_client");
intent2.setData(uri);
startActivity(intent2);

我们看到uri中?后面还加了"remote_invoke=from_client",这其实是用来给Server传递数据用的,我们可以在Server中解析出来

Uri uri = getIntent().getData();
String from = uri.getQueryParameter("remote_invoke");
//from = "from_client"

这里还有一个需要注意的点是,如果Client在调用时没有指定Action,同时Server中又有多个Activity注册了相同的scheme和host,那么在页面跳转时,系统会弹框让我们选择跳转到哪个页面,如下图所示:

3. 通过PackageManager唤起

只需要知道Server的包名即可

PackageManager packageManager = getPackageManager();
Intent intent3 = packageManager.getLaunchIntentForPackage("com.jxx.server");
if (intent3 != null) {
 startActivity(intent3);
} 

4. 静态广播接收者

只需要Server端注册一个静态广播接收者,在广播接收者中跳转Activity即可,客户端只需要发送一个广播。

Server定义广播接收者:

public class ServerBroadCastReceiver extends BroadcastReceiver {

 @Override
 public void onReceive(Context context, Intent intent) {
  Intent intent1 = new Intent(context, MainActivity.class);
  //注意,这里必须要添加这个flag,
  //原因在于这里的context并不是一个Activity类型的context,无法直接开启activity
  intent1.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
  context.startActivity(intent1);
 }
}

并在manifest中注册为静态广播接收者,并定义action

<receiver
 android:name=".ServerBroadCastReceiver"
 android:enabled="true"
 android:exported="true">
 <intent-filter>
  <action android:name="server.ServerBroadCastReceiver" />
 </intent-filter>
</receiver>

Client中发送广播即可

Intent intent4 = new Intent("server.ServerBroadCastReceiver");
//这里加上componentName用于解决8.0以上不能唤起的问题
ComponentName componentName = new ComponentName("com.jxx.server", "com.jxx.server.ServerBroadCastReceiver");
intent4.setComponent(componentName);
sendBroadcast(intent4); 

5. Service

在Android Service详解(二) 中我们介绍了如何通过Service实现IPC通信,这当然也能用来唤起App,这里就不再过多介绍了,有兴趣的同学可以点击查看。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android唤醒、解锁屏幕代码实例

    解锁.唤醒屏幕用到KeyguardManager,KeyguardLock,PowerManager,PowerManager.WakeLock   所需权限: 复制代码 代码如下: <uses-permission android:name="android.permission.WAKE_LOCK" /> <uses-permission android:name="android.permission.DISABLE_KEYGUARD" /&

  • Android保持屏幕常亮唤醒状态的方法

    本文实例讲述了Android保持屏幕常亮唤醒状态的方法.分享给大家供大家参考,具体如下: 第一步: 首先添加权限: 复制代码 代码如下: <uses-permission android:name="android.permission.WAKE_LOCK"></uses-permission> 第二步:代码实现如下: public class ScreenActivity extends Activity { PowerManager powerManager

  • Android应用程序保持后台唤醒(使用WakeLock实现)

    在使用一些产品列如微信.QQ之类的,如果有新消息来时,手机屏幕即使在锁屏状态下也会亮起并提示声音,这时用户就知道有新消息来临了.但是,一般情况下手机锁屏后,Android系统为了省电以及减少CPU消耗,在一段时间后会使系统进入休眠状态,这时,Android系统中CPU会保持在一个相对较低的功耗状态.针对前面的例子,收到新消息必定有网络请求,而网络请求是消耗CPU的操作,那么如何在锁屏状态乃至系统进入休眠后,仍然保持系统的网络状态以及通过程序唤醒手机呢?答案就是Android中的WakeLock机

  • 详解App相互唤醒的几种方式

    下文皆使用Client表示操作的App,Server表示需要被唤起的远端App,Server的包名为"com.jxx.server" 1. ComponentName 使用ComponentName唤起Server步骤很简单,需要注意的是Server的Activity需要在manifest配置种设置exported为true Server的配置如下: <activity android:name="com.jxx.server.ServerActivity"

  • 详解IOS 单例的两种方式

    详解IOS 单例的两种方式 方法一: #pragma mark - #pragma mark sharedSingleton methods //单例函数 static RtDataModel *sharedSingletonManager = nil; + (RtDataModel *)sharedManager { @synchronized(self) { if (sharedSingletonManager == nil) { sharedSingletonManager = [[sel

  • 详解vue 路由跳转四种方式 (带参数)

    1.  router-link 1. 不带参数 <router-link :to="{name:'home'}"> <router-link :to="{path:'/home'}"> //name,path都行, 建议用name // 注意:router-link中链接如果是'/'开始就是从根路由开始,如果开始不带'/',则从当前路由开始. 2.带参数 <router-link :to="{name:'home', para

  • 详解ubuntu安装CMake的几种方式

    apt安装CMake sudo apt install cmake 这种方式安装方便,缺点是如果想要自己交叉编译Android平台的opencv会提示版本太低,因为ubuntu16.04源里的cmake版本只有3.5.1,而Android交叉编译工具链android.toolchain.cmake要求cmake版本最低是3.6.0 下载源码编译CMake 到cmake官网下载最新的cmake https://cmake.org/download/ 下载后解压,然后进入目录执行: ./bootst

  • 详解记录Java Log的几种方式

    在Java中记录日志的方式有如下几种: 一.System.out.println(最简单) 1.输出到控制台:System.out.println("XXX"); 2.输出到指定文件: import java.io.PrintStream; PrintStream ps = new PrintStream("D:\\test.txt"); System.setOut(ps); System.out.println("XXX"); 二.java.u

  • 详解Springboot下载Excel的三种方式

    汇总一下浏览器下载和代码本地下载实现的3种方式. (其实一般都是在代码生成excel,然后上传到oss,然后传链接给前台,但是我好像没有实现过直接点击就能在浏览器下载的功能,所以这次一起汇总一下3种实现方式.)

  • 详解JUnit5参数化测试的几种方式

    目录 依赖 简单示例 七种方式 参数类型转换 隐式转换 显式转换 参数聚合 自定义显示名字 小结 参数化测试一直是津津乐道的话题,我们都知道JMeter有四种参数化方式:用户自定义变量.用户参数.CSV文件.函数助手,那么JUnit5有哪些参数化测试的方式呢? 依赖 JUnit5需要添加junit-jupiter-params依赖才能使用参数化: <dependency> <groupId>org.junit.jupiter</groupId> <artifact

  • 详解Spring集成Redis的两种方式

    目录 一.使用Jedis方式集成 1.增加依赖 2.配置项 3.配置连接池 4.测试 使用spring-data-redis 1.引入依赖 2.配置项 3.使用 4.可能会遇到的坑 哨兵和集群 总结: 在工作中,我们用到分布式缓存的时候,第一选择就是Redis,今天介绍一下SpringBoot如何集成Redis的,分别使用Jedis和Spring-data-redis两种方式. 一.使用Jedis方式集成 1.增加依赖 <!-- spring-boot-starter-web不是必须的,这里是为

  • 详解SpringBoot禁用Swagger的三种方式

    目录 摘要 方法 禁用方法1: 禁用方法2: 禁用方法3: 摘要 在生产环境下,我们需要关闭swagger配置,避免暴露接口的这种危险行为. 方法 禁用方法1: 使用注解 @Value() 推荐使用 package com.dc.config; import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.context.annotation.Be

  • 详解QTreeWidget隐藏节点的两种方式

    目录 简述 方法一:直接隐藏式 方法二:间接隐藏式 结尾 简述 关于QTreeWidget隐藏节点有两种方式,一种是直接隐藏,一种是间接隐藏,但是两种方式各有差异,下面请听具体解说. 方法一:直接隐藏式 Qt助手里面提供了QTreeWidgetItem::setHidden方法,我们可以调用setHide(false)直接隐藏当前item.但是调用此方法会隐藏该节点下面的所有子节点. 图 1-1: 我们通过图1-2看到,调用此方法会将自己所有的孩子节点都给隐藏了,如果有这种需求的直接调用此方法即

随机推荐