| Java后端人员,业余搞着玩,不对的地方请见谅。 1、build.gradle 配置在app/build.gradle中配置如下 导入gradle的插件包//buildscript中的声明是gradle脚本自身需要使用的资源,而在build.gradle文件中直接声明的依赖项、仓库地址等信息是项目自身需要的资源。放在文件开头。
buildscript {
    repositories {
        mavenCentral()
    }
    dependencies {
        classpath 'org.aspectj:aspectjtools:1.8.9'
        classpath 'org.aspectj:aspectjweaver:1.8.9'
    }
}
 导入jar包dependencies {
    implementation 'org.aspectj:aspectjrt:1.9.6' //引入 aspectj
}
 ?使 AspectJ 配置生效
import org.aspectj.bridge.IMessage
import org.aspectj.bridge.MessageHandler
import org.aspectj.tools.ajc.Main
// 获取log打印工具和构建配置
final def log = project.logger
final def variants = project.android.applicationVariants
variants.all { variant ->
    if (!variant.buildType.isDebuggable()) {
        // 判断是否debug,如果打release把return去掉就可以
        log.debug("Skipping non-debuggable build type '${variant.buildType.name}'.")
        // return;
    }
    // 使aspectj配置生效
    JavaCompile javaCompile = variant.javaCompile
    javaCompile.doLast {
        String[] args = ["-showWeaveInfo",
                         "-1.8",
                         "-inpath", javaCompile.destinationDir.toString(),
                         "-aspectpath", javaCompile.classpath.asPath,
                         "-d", javaCompile.destinationDir.toString(),
                         "-classpath", javaCompile.classpath.asPath,
                         "-bootclasspath", project.android.bootClasspath.join(File.pathSeparator)]
        log.debug "ajc args: " + Arrays.toString(args)
        MessageHandler handler = new MessageHandler(true);
        new Main().run(args, handler);
        //在编译时打印信息如警告、error等等
        for (IMessage message : handler.getMessages(null, true)) {
            switch (message.getKind()) {
                case IMessage.ABORT:
                case IMessage.ERROR:
                case IMessage.FAIL:
                    log.error message.message, message.thrown
                    break;
                case IMessage.WARNING:
                    log.warn message.message, message.thrown
                    break;
                case IMessage.INFO:
                    log.info message.message, message.thrown
                    break;
                case IMessage.DEBUG:
                    log.debug message.message, message.thrown
                    break;
            }
        }
    }
}
 2、自定义注解
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/**自定义注解*/
@Retention(RetentionPolicy.RUNTIME)//定义我们自己写的注解何时有效
@Target(ElementType.METHOD)//定义我们写的注解可以描述的成员
public @interface LoginFilter {
    //在注解里加了个loginDefine,就是为了给用户提供自定义实现,如根据loginDefine值不同做不同的登录处理。
    int loginDefine() default 0;
}
 3、配置切面
import android.content.Context;
import com.app.pest.anno.LoginFilter;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.Signature;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MemberSignature;
import org.aspectj.lang.reflect.MethodSignature;
/**
 * 登录切面处理
 */
@Aspect//此注解描述的类为切面类,其内部可以定义切入点表达式和通知方法
public class LoginFilterAspect {
    private static final String TAG = "LoginFilterAspect";
    //@annotation(包名.注解名)
    @Pointcut("@annotation(com.app.pest.anno.LoginFilter)")
    public void LoginFilter() {//承载切入点表达式的定义,方法不写任何内容
    }
    // @Around("@annotation(com.app.pest.anno.LoginFilter)")
    @Around("LoginFilter()")
    public void aroundLoginPoint(ProceedingJoinPoint joinPoint) throws Throwable {
        //获取用户实现的ILogin类,如果没有初始化就抛出异常。
        ILoginFilter iLoginFilter = LoginAssistant.getInstance().getILoginFilter();
        if (iLoginFilter == null) {
            throw new RuntimeException("LoginManger没有初始化");
        }
        //先得到方法的签名methodSignature,然后得到@LoginFilter注解,如果注解为空,就不再往下走。
        Signature signature = joinPoint.getSignature();
        if (!(signature instanceof MemberSignature)) {
            throw new RuntimeException("该注解只能用于方法上");
        }
        MethodSignature methodSignature = (MethodSignature) signature;
        LoginFilter loginFilter = methodSignature.getMethod().getAnnotation(LoginFilter.class);
        if (loginFilter == null) {
            return;
        }
        Context mContext = LoginAssistant.getInstance().getApplicationContext();
        //调用iLogin的isLogin()方法判断是否登录,这个isLogin是留给使用者自己实现的,如果登录,就会继续执行方法体调用方法直到完成,如果没有登录,执行下一个
        if (iLoginFilter.isLogin(mContext)) {//已经登录
            joinPoint.proceed();//执行目标方法
        } else {
            iLoginFilter.login(mContext, loginFilter.loginDefine());
        }
    }
}
 4、使用登录拦截器在需要拦截的方法上加上注解即可     @LoginFilter(loginDefine = 0)
    @Override
    //跳转到需要拦截登录的Activity
    public void onClick(View v) {
        startActivity(new Intent(this, SecondActivity.class));
    }
  ?  ?  ? 
 源码: https://github.com/antporter/pest |