先上效果压阵
AOP与AspectJ的介绍就不多说。
APP的登录方式,一种像QQ微信类,先登录后使用。另一种像美团饿了么,将产品展示给用户,用户产生兴趣进行专属操作时再登录。
第一种不做讨论,第二种情况下,如果未登录时的产品展示有很多功能,这时候一个一个的进行判断恐怕不是程序员想看到的,并且繁杂的程序更容易出错。
这里我们使用面向切面编程的思想——AOP,对需要登录的功能方法进行注解,统一处理登录检验。
AspectJ导入
AspectJ的配置很麻烦,这里使用大神的框架AspectJX
按照连接里面配置完成后,有一个需要注意:
如果AOP是放在单独Module中,则不仅AOP的Module需要
apply plugin: 'com.hujiang.android-aspectjx'
,并且使用到AOP功能的Module都需要apply plugin: 'com.hujiang.android-aspectjx'
写注解NeedLogin
//注解类型:方法注解
@Target(ElementType.METHOD)
//注解声明期:运行时注解
@Retention(RetentionPolicy.RUNTIME)
public @interface NeedLogin {
/**
* 展示Dialog提示登录
*/
int SHOW_DIALOG = 0;
/**
* 弹出吐司提示登录
*/
int SHOW_TOAST = 1;
/**
* 没有响应
*/
int NO_RESPONSE = 2;
/**
* 提示类型
*/
int tipType() default SHOW_TOAST;
/**
* 登录Activity
*/
Class<?> loginActivity();
String tipToast() default "当前尚未登录,请先登录";
String tipDialog() default "当前尚未登录,是否前往登录?";
}
这里只做了三种类型,可以根据需求增添.
写Aspect类
@Aspect
public class NeedLoginAspect {
@Pointcut("execution(" +//执行语句
"@com.martin.aspectj_module.annotation.NeedLogin" +//注解筛选
" * " + //类路径,*为任意路径
"*" + //方法名,*为任意方法名
"(..)" +//方法参数,'..'为任意个任意类型参数
")" +
" && " +//并集
"@annotation(needLogin)"//注解筛选,这里主要用于下面方法的'NeedLogin'参数获取
)
public void pointcutNeedLogin(NeedLogin needLogin) {
}
@Around("pointcutNeedLogin(needLogin)")
public void aroundNeedLogin(ProceedingJoinPoint joinPoint, final NeedLogin needLogin) throws Throwable {
if (AopUtil.getInstance().isLogin) {
//方法执行
joinPoint.proceed();
} else {
final Context context = AopUtil.getInstance().getContext();
switch (needLogin.tipType()) {
case NeedLogin.SHOW_TOAST:
Toast.makeText(context, needLogin.tipToast(), Toast.LENGTH_SHORT).show();
break;
case NeedLogin.SHOW_DIALOG:
/*
判断context是否为Activity
如果不是 , 使用Dialog会crash
*/
if (context instanceof Activity) {
new AlertDialog.Builder(context)
.setTitle("登录提示")
.setMessage(needLogin.tipDialog())
.setNegativeButton("取消", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
dialog.dismiss();
}
})
.setPositiveButton("前往登录", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
dialog.dismiss();
Intent intent = new Intent(context, needLogin.loginActivity());
context.startActivity(intent);
}
}).show();
} else {
// 这里处理直接跳到登录界面
Toast.makeText(context, needLogin.tipToast(), Toast.LENGTH_SHORT).show();
Intent intent = new Intent(context, needLogin.loginActivity());
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent);
}
break;
//无响应类型
case NeedLogin.NO_RESPONSE:
default:
break;
}
}
}
}
代码里使用到AOPUtil类,这里也贴出来:
public class AopUtil {
private Context context;
@SuppressLint("StaticFieldLeak")
private static AopUtil instance;
/**
* 是否登录
*/
public boolean isLogin = false;
private AopUtil(Context context) {
this.context = context;
}
public static void init(Context context) {
instance = new AopUtil(context);
}
public static AopUtil getInstance() {
return instance;
}
public Context getContext() {
return context;
}
}
到此,AspectJ实现登录检测的功能就完成了.主要是文笔太差,不知道该怎么写怎么说,感觉有些东西想着却说不出来,所以把一些东西都放在代码的注释中.
这里贴上源码
源码中,将AOP放在单独一个Module,可以直接拿来使用.