Android ARouter拦截器实现方法

 更新时间:2026年03月09日 09:23:13   作者:zh_xuan  
本文介绍了如何使用ARouter在Android应用中实现拦截器功能,通过拦截器可以实现页面跳转前的校验,本文通过示例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧

修改下之前的demo,测试下ARouter拦截器。页面如下:

未登录状态,点击下面红色按钮,跳转登录页面:

输入一个名字点击登录按钮跳转到下面用户中心页面:

回到首页,点击退出登录,重新点击最下面按钮,直接跳转到用户中心页面:

ok. 代码如下:  

在业务模块新增LoginActivity:

package com.example.module_2;
import android.os.Bundle;
import android.widget.Button;
import android.widget.EditText;
import android.widget.LinearLayout;
import android.widget.Toast;
import androidx.appcompat.app.AppCompatActivity;
import com.alibaba.android.arouter.facade.annotation.Autowired;
import com.alibaba.android.arouter.facade.annotation.Route;
import com.alibaba.android.arouter.launcher.ARouter;
import com.example.common.LoginService;
import com.example.module_2.R;
/**
 * 登录页面
 */
@Route(path = "/login/login")
public class LoginActivity extends AppCompatActivity {
    @Autowired(name = "goto_path")
    String gotoPath; // 登录成功后要跳转的路径
    private EditText etUsername;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        // 注入参数
        ARouter.getInstance().inject(this);
        // 创建简单的 UI
        LinearLayout layout = new LinearLayout(this);
        layout.setOrientation(LinearLayout.VERTICAL);
        layout.setPadding(50, 100, 50, 100);
        etUsername = new EditText(this);
        etUsername.setHint("请输入用户名");
        etUsername.setTextSize(16);
        Button btnLogin = new Button(this);
        btnLogin.setText("登录");
        btnLogin.setTextSize(16);
        btnLogin.setPadding(20, 30, 20, 30);
        layout.addView(etUsername);
        layout.addView(btnLogin);
        setContentView(layout);
        // 设置登录按钮点击事件
        btnLogin.setOnClickListener(v -> {
            String username = etUsername.getText().toString().trim();
            if (username.isEmpty()) {
                Toast.makeText(this, "请输入用户名", Toast.LENGTH_SHORT).show();
                return;
            }
            // 执行登录
            LoginService.login(username);
            Toast.makeText(this, "登录成功!欢迎:" + username, Toast.LENGTH_LONG).show();
            if (gotoPath != null) {
                ARouter.getInstance().build(gotoPath).navigation(); // 跳转到目标页面
                finish();
            } else {
                finish();
            }
        });
    }
}

个人中心页面:

package com.example.module_2;
import android.os.Bundle;
import android.widget.TextView;
import androidx.appcompat.app.AppCompatActivity;
import com.alibaba.android.arouter.facade.annotation.Route;
import com.example.common.LoginService;
/**
 * 需要登录才能访问的页面
 * 
 * 路径以 /needlogin/ 开头,会被 LoginInterceptor 拦截检查
 */
@Route(path = "/needlogin/personal")
public class PersonalCenterActivity extends AppCompatActivity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        TextView textView = new TextView(this);
        textView.setText(String.format("个人中心页面\n\n %s", LoginService.getUsername()));
        textView.setTextSize(18);
        textView.setPadding(50, 100, 50, 100);
        setContentView(textView);
    }
}

拦截器也写在该业务模块,如下:

package com.example.module_2.interceptor;
import android.content.Context;
import android.util.Log;
import com.alibaba.android.arouter.facade.Postcard;
import com.alibaba.android.arouter.facade.annotation.Interceptor;
import com.alibaba.android.arouter.facade.callback.InterceptorCallback;
import com.alibaba.android.arouter.launcher.ARouter;
/**
 * 日志记录拦截器
 * 
 * 作用:记录所有路由跳转的详细信息,调试用
 * 
 * 优先级:数字小的优先级高,先执行
 */
@Interceptor(priority = -1, name = "日志记录拦截器")
public class LoggerInterceptor implements com.alibaba.android.arouter.facade.template.IInterceptor {
    private static final String TAG = "ARouter-Logger";
    @Override
    public void process(Postcard postcard, InterceptorCallback callback) {
        // 记录路由信息
        Log.d(TAG, "========================================");
        Log.d(TAG, ">>> 路由跳转开始");
        Log.d(TAG, ">>> 目标路径:" + postcard.getPath());
        // 打印所有参数
        if (postcard.getExtras() != null && !postcard.getExtras().isEmpty()) {
            Log.d(TAG, ">>> 携带参数:");
            for (String key : postcard.getExtras().keySet()) {
                Object value = postcard.getExtras().get(key);
                Log.d(TAG, "      " + key + " = " + value);
            }
        } else {
            Log.d(TAG, ">>> 无参数");
        }
        // 继续处理,传递自定义回调
        callback.onContinue(postcard); // 交给下一个拦截器处理
    }
    @Override
    public void init(Context context) {
        Log.d(TAG, ">>> LoggerInterceptor 初始化完成");
    }
}
package com.example.module_2.interceptor;
import android.content.Context;
import android.util.Log;
import android.widget.Toast;
import com.alibaba.android.arouter.facade.Postcard;
import com.alibaba.android.arouter.facade.annotation.Interceptor;
import com.alibaba.android.arouter.facade.callback.InterceptorCallback;
import com.alibaba.android.arouter.launcher.ARouter;
import com.example.common.LoginService;
/**
 * 登录检查拦截器
 * 
 * 作用:拦截所有需要登录的页面,如果未登录则跳转到登录页
 * 
 */
@Interceptor(priority = 1, name = "登录检查拦截器")
public class LoginInterceptor implements com.alibaba.android.arouter.facade.template.IInterceptor {
    private static final String TAG = "LoginInterceptor";
    private Context context;
    @Override
    public void process(Postcard postcard, InterceptorCallback callback) {
        Log.d(TAG, ">>> LoginInterceptor 开始处理。 \n 嘿嘿");
        // 获取目标路径
        String path = postcard.getPath();
        // 只拦截特定路径(以 /needlogin/ 开头的路径都需要登录)
        if (path != null && path.startsWith("/needlogin/")) {
            Log.d(TAG, "检测到需要登录的路径:" + path);
            // 检查是否已登录
            if (!LoginService.isLogin()) {
                Log.d(TAG, "用户未登录,准备跳转到登录页");
                // 保存原始请求信息
//                postcard.withString("goto_path", path); // 和47行路由传参一样,调用一处就行。
                // 重定向到登录页
                ARouter.getInstance()
                    .build("/login/login")
                    .withString("goto_path", path)
                    .navigation();
                // 中断路由流程
                callback.onInterrupt(new RuntimeException("请先登录哈"));
                // 显示提示信息
                if (context != null) {
                    Toast.makeText(context, "请先登录哦", Toast.LENGTH_SHORT).show();
                }
                return;
            } else {
                Log.d(TAG, "用户已登录,允许通过");
            }
        } else {
            Log.d(TAG, "该路径不需要登录验证:" + path);
        }
        // 继续处理下一个拦截器或目标页面
        callback.onContinue(postcard);
    }
    @Override
    public void init(Context context) {
        this.context = context;
        Log.d(TAG, ">>> LoginInterceptor 初始化完成");
    }
}

主模块注册业务模块的activity:

主页MainActivity关键代码,测试需要校验登录的页面(即个人中心页面):

// 测试拦截器 - 访问需要登录的页面
        View btnNeedLogin = findViewById(R.id.btnNeedLogin);
        if (btnNeedLogin != null) {
            btnNeedLogin.setOnClickListener(v -> {
                Log.d("MainActivity", "准备访问需要登录的页面");
                ARouter.getInstance()
                    .build("/needlogin/personal")
                    .navigation();
            });
        }

ok.  看下未登录时,点击跳转个人中心,拦截器日志:

ok. 日志拦截器先打印了路由信息。然后登录拦截器校验了未登录,重定向到登录页面。

然后登录页面登录成功,跳转个人中心的时候,拦截器日志如下:

ok. 这次允许通过,成功跳转到个人中心页面。

ok. 拦截器挺有用。能用于实现一些公共功能。

到此这篇关于Android使用ARouter实现拦截器功能的文章就介绍到这了,更多相关android arouter拦截器内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Android 实现沉浸式状态栏的方法

    Android 实现沉浸式状态栏的方法

    沉浸式状态栏的来源就是很多手机用的是实体按键,没有虚拟键,于是开了沉浸模式就只有状态栏消失了。下面脚本之家小编给大家介绍Android 实现沉浸式状态栏,需要的朋友可以参考下
    2015-09-09
  • android 应用内部悬浮可拖动按钮简单实现代码

    android 应用内部悬浮可拖动按钮简单实现代码

    本篇文章主要介绍了android 应用内部悬浮可拖动按钮简单实现代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-10-10
  • Android开发Jetpack组件ViewModel与LiveData使用讲解

    Android开发Jetpack组件ViewModel与LiveData使用讲解

    Jetpack是一个由多个技术库组成的套件,可帮助开发者遵循最佳做法,减少样板代码并编写可在各种Android版本和设备中一致运行的代码,让开发者精力集中编写重要的代码
    2022-09-09
  • Android解决getExternalStorageDirectory在29后废弃问题(推荐)

    Android解决getExternalStorageDirectory在29后废弃问题(推荐)

    这篇文章主要介绍了Android解决getExternalStorageDirectory在29后废弃问题(推荐),本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-02-02
  • Android完整Socket解决方案

    Android完整Socket解决方案

    这篇文章主要介绍了Android完整Socket解决方案并通过代码给大家做了实例分析,对此有兴趣的朋友学习下。
    2018-02-02
  • Kotlin协程launch原理详解

    Kotlin协程launch原理详解

    这篇文章主要为大家介绍了Kotlin协程launch原理的示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-08-08
  • 图文详解Android Studio搭建Android集成开发环境的过程

    图文详解Android Studio搭建Android集成开发环境的过程

    这篇文章主要以图文的方式详细介绍了Android Studio搭建Android集成开发环境的过程,文中安装步骤介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2015-12-12
  • Android仿支付宝密码输入效果封装

    Android仿支付宝密码输入效果封装

    这篇文章主要为大家详细介绍了Android仿支付宝密码输入效果的封装,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-12-12
  • Android开发导入项目报错Ignoring InnerClasses attribute for an anonymous inner class的解决办法

    Android开发导入项目报错Ignoring InnerClasses attribute for an anonym

    今天小编就为大家分享一篇关于Android开发导入项目报错Ignoring InnerClasses attribute for an anonymous inner class的解决办法,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2018-12-12
  • Android巧用DecorView实现对话框功能

    Android巧用DecorView实现对话框功能

    本篇文章主要介绍了Android巧用DecorView实现对话框功能,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-04-04

最新评论