- 前言
- 介绍
- 使用
- 数据的解析
- Google约束
- 具体使用
- 完全具体实现
- adb命令测试方法
- 补充
前言
随着线上推广力度加大,不可避免的需要通过网页的形式向用户推广,进行APP的引流以及下载,这就使用了深度链接技术
介绍
深度链接(Deeplinking)是通过网页链接直接启动原生应用的方法。确切地说是通过映射预定义行为到唯一的链接上,让用户无缝跳转到相关内容页面。
官方推荐的使用深度链接的方法是通过定义URI(统一资源定位符),唯一标识一个APP的具体行为;
例如:eganprojest://needstartactivity
其对应的定义是:scheme://host
或者是网络地址的写法:
例如:https://www.eganproject.com/needstartactivity
起对应的定义是:scheme://host//pathPrefix
实际上,只要按照约定好即可。
使用
按照官方文档:
1. 在需要的页面注册位置进行配置:
<activity
android:name=".Task2Activity">
<intent-filter>
<action android:name="android.intent.action.VIEW" />
<category android:name="android.intent.category.DEFAULT"/>
<category android:name="android.intent.category.BROWSABLE"/>
<!-- 接受以"testproject://task2”开头的 URIs -->
<data android:scheme="testproject"
android:host="task2" />
<!-- 接受以"https://www.eganproject.com/aaaa”开头的 URIs -->
<data android:scheme="https"
android:host="www.eganproject.com"
android:pathPrefix="aaaa" />
</intent-filter>
</activity>
对上面的属性进行意义阐释:
- 为页面的添加intent-filter
- <action android:name="android.intent.action.VIEW" />
– 指定ACTION_VIEW的操作,使得Google搜索可以触及intent filter
- <category android:name="android.intent.category.BROWSABLE"/>
– BROWSABLE category
指定该页面可以响应浏览器,如果不添加,APP无法响应深度链接
- <category android:name="android.intent.category.DEFAULT"/>
– DEFAULT category
是可选的,但建议添加。没有这个category,activity只能够使用app组件名称以显示(explicit)intent启动。
- <data android:scheme="testproject"/>
– data
标签可以一到多个,不同属性指定对于深度链接的不同解析
- android:scheme
必须包含的标签,其一般指定所对应的启动的App
- android:host
具体的动作/应用的唯一标识
- android:pathPrefix
页面
数据的解析
如果我们一个页面可能支持了多个 标签,那我们如何区分是哪个URI启动的呢?Android
已将uri
的信息直接保存到启动Activity
的Intent
中,可以方便的通过下面的方法直接获取
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
// 获得启动页面的 Intent.
Intent intent = getIntent();
// 获得启动页面的 Uri 信息.
Uri data = intent.getData();
}
Google约束
遵守下面这些惯例来提高用户体验:
深度链接应直接为用户打开内容,不需要任何提示,插播式广告页和登录页面。要确保用户能看到app的内容,即使之前从没打开过这个应用。当用户从启动器打开app时,可以在操作结束后给出提示。这个准则也同样适用于网站的first click free体验。
遵循Navigation with Back and Up中的设计指导,来使你的app能够满足用户通过深度链接进入app后,向后导航的需求。
但是我们实际开发中可能并不能或者是并不容易达到这个要求,比如打开的App
目的页面需要App
的启动信息或者登录信息,或者产品的需求等;
所以下面说一下实际开发中的使用;
具体使用
前面的准备工作,也就是知识的储备是必须的;
只是需要对一些配置信息、以及数据的解析进行简单的改动;
- 所需要启动的目的页面不需要配置,只需要在App
的LaunchActivity(启动页面)
进行配置
<activity
android:name=".LaunchActivity">
<intent-filter>
<action android:name="android.intent.action.VIEW" />
<category android:name="android.intent.category.DEFAULT"/>
<category android:name="android.intent.category.BROWSABLE"/>
<!-- 接受以"testproject://xxx”开头的 URIs -->
<data android:scheme="testproject"/>
</intent-filter>
</activity>
我们只需要配置action
、DEFAULT category
、BROWSABLE category
以及data
中的scheme
属性
这样的话我们 App
会响应所有以 testproject://
开头的URI
,并且是在APP的启动页面,这样,可以保证我们APP流程的完整性(PS:比如启动流程)
在 Launch 页面,我们可以通过 getData() 获取到具体的URI信息,并且对数据进行解析存储(key-value),在特点的页面,比如 APP的首页,再次对所存储的数据进行解析、区分,进行不同业务的处理这同时也保证了业务完成后返回能够回到 APP 的首页,而不是直接退出 APP
完全具体实现
页面配置
<!-- 启动页面 -->
<activity android:name=".LaunchActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
<!-- 支持深度链接的配置 -->
<intent-filter>
<action android:name="android.intent.action.VIEW" />
<category android:name="android.intent.category.DEFAULT" />
<category android:name="android.intent.category.BROWSABLE" />
<data
android:host="www.egan.testproject"
android:scheme="testproject" />
</intent-filter>
</activity>
<!-- 主页面 -->
<activity android:name=".MainActivity" />
<!-- 目标页面 -->
<activity android:name=".Task2Activity" />
public class DeepLinks {
// 每一个短链接中有一个参数的key是 tag,用了唯一区别该短链
// PS:也可以通过 path 来区分,一样的,这里用的第一种
private static final String TAG_KEY = "tag";
private final Map<String, String> dataMap = new HashMap<>();
private DeepLinks() {
}
public static DeepLinks getInstance() {
return SingletonHolder.deepLinks;
}
private static final class SingletonHolder {
static final DeepLinks deepLinks = new DeepLinks();
}
/**
* 深度链接的数据解析.
*
* @param intent 携带链接数据的 intent.
*/
void parseData(Intent intent) {
final Uri data = intent.getData();
if (null != data) {
// todo...
Log.d("YSK", "DeepLinks:parseData >>> " + data);
Set<String> parameterNames = data.getQueryParameterNames();
Observable.fromIterable(parameterNames).subscribe(new Consumer<String>() {
@Override
public void accept(String s) throws Exception {
// 将数据进行数据的存储.
dataMap.put(s, data.getQueryParameter(s));
}
}, new Consumer<Throwable>() {
@Override
public void accept(Throwable throwable) throws Exception {
Log.d("YSK", "DeepLinks:accept >>> " + throwable.getMessage());
}
});
}
/**
* 数据处理.
*/
void processData() {
// 可补充,比如增加App的数据有效性校验.
// 获取目标活动数据 key
if (dataMap.containsKey(TAG_KEY)) {
// 打印数据
logMap(dataMap);
String key = dataMap.get(TAG_KEY);
// 根据具体业务区分处理。
switch (key) {
case "Task2":
break;
case "Task3":
break;
default:
break;
}
// 存储的深度链接信息处理后,清空,防止后续深度链接数据的准确性.
dataMap.clear();
}
}
/**
* 打印 map 信息.
*/
private void logMap(final Map<String, String> map) {
Observable.fromIterable(map.keySet()).subscribe(new Consumer<String>() {
@Override
public void accept(String o) throws Exception {
Log.d("YSK", "DeepLinks:accept >>> " + o + " : " + map.get(o));
}
});
}
/**
* 启动页面
*/
public class LaunchActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_launch);
// 深度链接启动的数据解析.
DeepLinks.getInstance().parseData(getIntent());
// 延时3s发送事件,模拟启动页面的广告时间.
Observable.timer(3, TimeUnit.SECONDS).subscribe(new Consumer<Long>() {
@Override
public void accept(Long aLong) throws Exception {
startActivity(new Intent(LaunchActivity.this, MainActivity.class));
finish();
}
});
}
}
/**
* 主页面.
*/
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// 深度链接启动的信息处理.
DeepLinks.getInstance().processData();
}
}
adb命令测试方法
google官方提供了通过ADB命令直接测试深度链接效果的命令(不用再费力的专门写一个html了)
adb shell am start
-W -a android.intent.action.VIEW
-d <URI-定义的URI> <PACKAGE-需要测试的应用包名>
比如我需要测试我的Demo,可以直接通过命令
adb shell am start
-W -a android.intent.action.VIEW
-d "testproject://www.egan.testproject/aler?tag=Task2&p1=1" com.egan.testproject
补充
深度链接一般我们可能都需要传递数据,比如我需要两个参数,这种情况下的测试,使用官方提供的adb命令无法完全获得两个参数,比如我执行的是 testproject://www.egan.testproject/aler?tag=Task2&p1=1
,但是通过 getIntent().getData()
获取到的 uri 是 testproject://www.egan.testproject/aler?tag=Task2
,只能获取到第一个参数;
所以请编写一个 html 自行测试;
- 业务区分的常用方式
testproject://www.egan.testproject/business_1
testproject://www.egan.testproject/business_2
testproject://www.egan.testproject/alert?tag=business_1
testproject://www.egan.testproject/alert?tag=business_2
- 携带参数的常用方式:
testproject://www.egan.testproject/alert?param1=param1¶m2=param2
testproject://www.egan.testproject/alert?data={'param1':'param1', 'param2':'param2'}