android 实现AOP 使用Aspectj Kotlin版

注意代码为Kotlin

什么是OOP、什么是AOP?

OOP(面向对象编程)针对业务处理过程的实体及其属性和行为进行抽象封装,以获得更加清晰高效的逻辑单元划分,而AOP则是针对业务处理过程中的切面进行提取,它所面对的是处理过程中的某个步骤或阶段,以获得逻辑过程中各部分之间低耦合性的隔离效果。

AOP的应用场景

AOP编程的主要用途有:日志记录,行为统计,安全控制,事务处理,异常处理,系统统一的认证、权限管理等。

AspectJ是什么?

AspectJ是一个面向切面编程的一个框架,它扩展了java语言,并定义了实现AOP的语法。在将.java文件编译为.class文件时默认使用javac编译工具,AspectJ会有一套符合java字节码编码规范的编译工具来替代javac,在将.java文件编译为.class文件时,会动态的插入一些代码来做到对某一类特定东西的统一处理。通过预编译方式和运行期动态代理实现在不修改源代码的情况下给程序动态统一添加功能的技术。对业务逻辑的各个部分进行隔离,耦合度降低,提高程序的可重用性,同时提高了开发的效率。

管网https://www.eclipse.org/aspectj/

Kotlin版https://github.com/HujiangTechnology/gradle_plugin_android_aspectjx

通过Aspect实现防止多点击

1.引入Aspectj
  • 在项目根目录下的build.gradle中,添加依赖
classpath 'com.hujiang.aspectjx:gradle-android-plugin-aspectjx:2.0.10'
  • 在app或其他module目录下的build.gradle中,添加以下代码:
plugins {
    id 'com.android.library'
    id 'kotlin-android'
    id 'android-aspectjx'//Aspectj添加
}
2.添加一个自定义注解
/**
 * AnnotationRetention.SOURCE:不存储在编译后的 Class 文件。
 * AnnotationRetention.BINARY:存储在编译后的 Class 文件,但是反射不可见。
 * AnnotationRetention.RUNTIME:存储在编译后的 Class 文件,反射可见。
 */
@Retention(AnnotationRetention.RUNTIME)
/**
 * AnnotationTarget.CLASS:类,接口或对象,注解类也包括在内。
 * AnnotationTarget.ANNOTATION_CLASS:只有注解类。
 * AnnotationTarget.TYPE_PARAMETER:Generic type parameter (unsupported yet)通用类型参数(还不支持)。
 * AnnotationTarget.PROPERTY:属性。
 * AnnotationTarget.FIELD:字段,包括属性的支持字段。
 * AnnotationTarget.LOCAL_VARIABLE:局部变量。
 * AnnotationTarget.VALUE_PARAMETER:函数或构造函数的值参数。
 * AnnotationTarget.CONSTRUCTOR:仅构造函数(主函数或者第二函数)。
 * AnnotationTarget.FUNCTION:方法(不包括构造函数)。
 * AnnotationTarget.PROPERTY_GETTER:只有属性的 getter。
 * AnnotationTarget.PROPERTY_SETTER:只有属性的 setter。
 * AnnotationTarget.TYPE:类型使用。
 * AnnotationTarget.EXPRESSION:任何表达式。
 * AnnotationTarget.FILE:文件。
 * AnnotationTarget.TYPEALIAS:@SinceKotlin("1.1") 类型别名,Kotlin1.1已可用。
 */
@Target(AnnotationTarget.FUNCTION)
annotation class AopOnclick(val outTime:Long = 500)
3.添加重复点击判断逻辑
import android.os.SystemClock

object AopClickUtil {
    /**
     * 最近一次点击的时间
     */
    private var mLastClickTime: Long = 0

    /**
     * 是否是快速点击
     *
     * @param intervalMillis  时间间期(毫秒)
     * @return  true:是,false:不是
     */
    fun isFastDoubleClick(intervalMillis: Long): Boolean {
        //        long time = System.currentTimeMillis();
        val time = SystemClock.elapsedRealtime()
        val timeInterval = Math.abs(time - mLastClickTime)
        return if (timeInterval < intervalMillis) {
            true
        } else {
            mLastClickTime = time
            false
        }
    }
}
4.AOP处理类
package com.wpf.api

import org.aspectj.lang.ProceedingJoinPoint
import org.aspectj.lang.annotation.Around
import org.aspectj.lang.annotation.Aspect
import org.aspectj.lang.annotation.Pointcut
import org.aspectj.lang.reflect.MethodSignature

/**
@Pointcut("execution(" +//执行语句
"@com.wpf.api.AopOnclick" +//注解筛选
"*" + //类路径,*为任意路径
"*" + //方法名,*为任意方法名
"(..)" +//方法参数,'..'为任意个任意类型参数
")" +
" && " +//并集
)
@Aspect:声明切面,标记类
@Pointcut(切点表达式):定义切点,标记方法
@Before(切点表达式):前置通知,切点之前执行
@Around(切点表达式):环绕通知,切点前后执行
@After(切点表达式):后置通知,切点之后执行
@AfterReturning(切点表达式):返回通知,切点方法返回结果之后执行
@AfterThrowing(切点表达式):异常通知,切点抛出异常时执行
 * */
@Aspect
class AopClickAspect {
    /**
     * 定义切点,标记切点为所有被@AopOnclick注解的方法
     * 注意:这里com.wpf.api.AopOnclick需要替换成
     * 你自己项目中AopOnclick这个类的全路径
     */
    @Pointcut("execution(@com.wpf.api.AopOnclick * *(..))")
    fun methodAnnotated(){}
    /**
     * 定义一个切面方法,包裹切点方法
     */
    @Around("methodAnnotated()")
    @Throws(Throwable::class)
    fun aroundJoinPoint(joinPoint: ProceedingJoinPoint) {
        // 取出方法的注解
        val methodSignature = joinPoint.signature as MethodSignature
        val method = methodSignature.method
        if (!method.isAnnotationPresent(AopOnclick::class.java)) {
            return
        }
        val aopOnclick = method.getAnnotation(AopOnclick::class.java)
        // 判断是否快速点击
        if (!AopClickUtil.isFastDoubleClick(aopOnclick.outTime)) {
            // 不是快速点击,执行原方法
            joinPoint.proceed()
        }
    }
}
5.方法调用
    @AopOnclick
    override fun onClick(v: View) {
        if (v.id == R.id.hello_World){
            Log.e("wpf","点击hello")
        }
    }

采坑记录

问题1:java.util.zip.ZipException: zip file is empty

解决方式:

android {
    aspectjx {
//排除所有package路径中包含`android.support`的class文件及库(jar文件)
        exclude 'android.support'
        exclude 'versions.9'
    }
}
问题2:Cannot cast object ‘com.android.build.gradle.internal.pipeline.TransformTask

解决方式:
根build.gradle文件中配置的gradle版本和aspectjx匹配,如我的配置。

buildscript {
    ext.kotlin_version = "1.4.32"
    repositories {
        google()
        jcenter()
    }
    dependencies {
        classpath "com.android.tools.build:gradle:4.1.3"
        classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
        //AspectJX
        classpath 'com.hujiang.aspectjx:gradle-android-plugin-aspectjx:2.0.10'
    }
}

代码链接

https://download.csdn.net/download/fepengwang/20398670

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值