一文带你深入了解Android的Scheme协议!

最近在一个技术公众号里看见Scheme,由于之前没有接触过。触及到了我的知识盲点,于是花了些功夫去了解这个协议。

URL Scheme 有什么用?使用场景

Scheme 用于从浏览器或其他应用中启动本应用。也就是说要从其他应用中跳转本应用的界面或者网页跳转本应用打开特定的界面。

如何定义Scheme协议

首先我们来看看URL Scheme 的格式
客户端自定义的 URL 作为从一个应用调用另一个的基础,遵循 RFC 1808 (Relative Uniform Resource Locators) 标准。这跟我们常见的网页内容 URL 格式一样。
一个普通的 URL 分为几个部分,scheme、host、port、relativePath、query、fragment(定义一个url包含了你定义的scheme,主机名或者域名,端口(可选)路径,查询条件等)
URL的一般语法格式为:
(带方括号[]的为可选项):
protocol 😕/ hostname[:port] / path / [;parameters][?query]#fragment

protocol: 协议也就是你定义的scheme
hostname: 地址域
port:端口
path:路径
params:参数

在Android中的用法

在AndroidManifest.xml中定义intent-filter,这里我给出部分代码

<activity
            android:name=".SchemeTargetActivity"
            android:launchMode="singleTask">
            <!-- 要想在别的App上能成功调起App,必须添加intent过滤器 -->
            <intent-filter>
                <!-- 协议部分 可以随便设置 还可以加一些host port path 等,协议规则越详细定位界面更精确,-->
                <!--<data android:scheme="bruce"></data>-->

                <data android:scheme="bruce"
                    android:host="baidu"
                    android:port="8080"
                    android:path="/tieba"
                    ></data>
                <!-- 必须设置 -->
                <action android:name="android.intent.action.VIEW" />
                <category android:name="android.intent.category.DEFAULT" />
                <!-- 如果需要外部网页打开本页面,还需要设置这个 -->
                <category android:name="android.intent.category.BROWSABLE" />

            </intent-filter>

 </activity>

为什么设置singleTask就是为了防止生成多个实例多次启动这个页面。如果有传参,我们可以重写onNewIntent获取参数
下面我给出两个activity的代码,注意这里两个activity来自不同的应用。我姑且定为C应用,D应用。 C调用D

C应用的activity代码

/**
 * DATE:2018/3/21
 * USER: liuzj
 * DESC:
 * email:liuzj@hi-board.com
 */

public class BActivity extends AppCompatActivity {

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_b);


    }

    public void scheme(View view) {
        if (hasApplication()) {
            Toast.makeText(this, "有目标界面", Toast.LENGTH_SHORT).show();
            Uri uri = Uri.parse("bruce://baidu:8080/tieba");
            Intent intent = new Intent(Intent.ACTION_VIEW, uri);
            startActivity(intent);
        }else {
            Toast.makeText(this, "没有目标界面", Toast.LENGTH_SHORT).show();
        }

    }

    /**
     * 判断是否安装了应用
     * @return true 为已经安装
     */
    private boolean hasApplication() {
        PackageManager manager = getPackageManager();
        Intent action = new Intent(Intent.ACTION_VIEW);
        action.setData(Uri.parse("bruce://baidu:8080/tieba"));
        List list = manager.queryIntentActivities(action, PackageManager.GET_RESOLVED_FILTER);
        return list != null && list.size() > 0;
    }
}

D应用的activit代码如下:

/**
 * DATE:2018/3/22
 * USER: liuzj
 * DESC:
 * email:liuzj@hi-board.com
 */

public class SchemeTargetActivity extends AppCompatActivity {

    private static final String TAG = "SchemeTargetActivity";
    private TextView text;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_scheme);
        text = (TextView) findViewById(R.id.tv);
        Uri uri = getIntent().getData();
        if (uri != null) {
            dispatchUri(uri);
        } else {
            Log.e(TAG, "Uri is null");
        }
    }

    /**
     * 这里可以做一些获取你从uri中设置的信息
     *
     * @param uri
     */
    private void dispatchUri(Uri uri) {

        String uriStr = uri.toString();
        Log.e(TAG, "dispatchUri: 39---" + uriStr);
        Log.e(TAG, "dispatchUri: 40---" + uri.getScheme());
        text.setText(uri.getScheme());
    }

    @Override
    protected void onNewIntent(Intent intent) {
        super.onNewIntent(intent);
        Uri uri = intent.getData();
        Log.e(TAG, "onNewIntent: 48--------" + uri.toString());
        text.setText(uri.toString());
    }
}

你以为到了这里就要结束了吗?怎么可能。我们来玩一玩通过短信链接来打开应用
把协议改一改

<activity
            android:name=".SchemeTargetActivity"
            android:launchMode="singleTask">
            <!-- 要想在别的App上能成功调起App,必须添加intent过滤器 -->
            <intent-filter>
                <!-- 协议部分 可以随便设置 还可以加一些host port path 等,协议规则越详细定位界面更精确,-->
                <!--<data android:scheme="bruce"></data>-->

                <data
                    android:scheme="bruce"
                    android:host="mytest.com"
                    android:path="/get"
                    >

                </data>
                <!-- 必须设置 -->
                <action android:name="android.intent.action.VIEW" />
                <category android:name="android.intent.category.DEFAULT" />
                <!-- 如果需要外部网页打开本页面,还需要设置这个 -->
                <category android:name="android.intent.category.BROWSABLE" />

            </intent-filter>

        </activity>

其实scheme设置成http也是可以的但是怕用户会通过浏览器去打开,所以为了避免这种问题这里的scheme我们自己写一个

写个html

<html lang="zh-cn">
<head>
    <meta charset="utf-8">
    <meta http-equiv="Refresh" content="0;url=bruce://mytest.com/get" />
    <title>Android测试</title>
</head>
<body>
</body>
</html>

即只要点击网址我们就会重定向去打开app,前端代码写的不好,多见谅。实际开发中前端代码还可通过超时判断用户是否安装了app,没有安装就去下载页面,安装了app就可以打开app跳转到该页面。

最后

我把自己这段时间整理的Android最重要最热门的学习方向资料放在了我的GitHub,里面还有不同方向的自学编程路线、面试题集合/面经、及系列技术文章等。

资源持续更新中,欢迎大家一起学习和探讨。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android Scheme跳转协议是一种在Android操作系统中用于app之间通信的机制。通过Scheme跳转协议,一个app可以通过调用另一个app的特定功能或界面,实现跳转和传递参数的功能。 Android Scheme跳转协议的实现过程如下: 1. 在待跳转的app中,需要先定义一个特定的Scheme,用于唯一标识该app。一般情况下,Scheme由app的包名或自定义的scheme名称组成。 2. 在相应的Activity或Fragment中,通过Intent设置该Activity或Fragment的Scheme,以及需要传递的参数,如数据、标志位等。 3. 在需要跳转到该app的app中,通过调用系统提供的隐式Intent的方式,并设置跳转的Scheme以及传递的参数,发起跳转。 4. 如果目标app已安装在设备上,在符合条件的应用列表中,用户可以选择使用该app打开链接。 5. 目标app接收到跳转请求后,在其Manifest文件中通过声明Intent过滤器,匹配相应的Scheme和路径,并执行对应的操作或界面跳转。 需要注意的是,为了保证Scheme跳转的可用性,需要在app安装时注册相应的Scheme,以便系统能够正确地将请求导向目标app。 Scheme跳转协议在实际应用中常被用于app之间的跳转和通信,例如在支付宝中使用支付宝Scheme跳转到其他第三方APP完成支付,或者在浏览器中使用特定的Scheme跳转到其他APP打开指定的页面等。 总之,Android Scheme跳转协议是一种方便实现app之间通信的机制,通过定义特定的Scheme和使用Intent实现跳转和传参,能够提供更丰富的用户体验。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值