### 有人想要Android面向切面編程,今天他來了!😜,輕鬆完成各種騷操作!登錄狀態攔截,日誌攔截,權限攔截,輕鬆搞定!
https://github.com/TanZhiL/OkAspectj
更新日誌:
v1.02 2019-10.17
- 第一次發佈
快速對指定函數進行切面攔截:
- 註解完全自定義
- 攔截規則自定義
- 無需手動編寫切面代碼,APT自動生成切面文件
- 支持組件化
Installation:
1.project.gradle 添加(同步完成後再進行下一步!!!)
buildscript {
repositories {
google()
jcenter()
maven { url 'https://jitpack.io' }
}
dependencies {
classpath 'com.android.tools.build:gradle:3.2.1'
classpath 'org.aspectj:aspectjtools:1.8.9'
classpath 'org.aspectj:aspectjweaver:1.8.9'
}
}
2.app.gradle 添加(注意每個需要生成切面的文件的組件都需要添加annotationProcessor)
dependencies {
implementation 'org.aspectj:aspectjrt:1.8.14'
implementation 'com.github.TanZhiL:OkAspectjAnnotation:1.0.4'
annotationProcessor 'com.github.TanZhiL:OkAspectjCompiler:1.0.4'
}
/*******************獨立運行時**********************************/
import org.aspectj.bridge.IMessage
import org.aspectj.bridge.MessageHandler
import org.aspectj.tools.ajc.Main
final def log = project.logger
final def variants = project.android.applicationVariants
variants.all { variant ->
if (!variant.buildType.isDebuggable()) {
log.debug("Skipping non-debuggable build type '${variant.buildType.name}'.")
return;
}
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);
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;
}
}
}
}
/***********************END****************************/
/*******************作爲組件時**********************************/
import com.android.build.gradle.LibraryPlugin
import org.aspectj.bridge.IMessage
import org.aspectj.bridge.MessageHandler
import org.aspectj.tools.ajc.Main
android.libraryVariants.all { variant ->
LibraryPlugin plugin = project.plugins.getPlugin(LibraryPlugin)
JavaCompile javaCompile = variant.javaCompile
javaCompile.doLast {
String[] args = ["-showWeaveInfo",
"-1.5",
"-inpath", javaCompile.destinationDir.toString(),
"-aspectpath", javaCompile.classpath.asPath,
"-d", javaCompile.destinationDir.toString(),
"-classpath", javaCompile.classpath.asPath,
"-bootclasspath", plugin.project.android.bootClasspath.join(
File.pathSeparator)]
MessageHandler handler = new MessageHandler(true);
new Main().run(args, handler)
def log = project.logger
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:
case IMessage.INFO:
log.info message.message, message.thrown
break;
case IMessage.DEBUG:
log.debug message.message, message.thrown
break;
}
}
}
}
/***********************END****************************/
Usage:
- 在自己想要攔截的註解之上添加 @OkAspectj註解
@OkAspectj
@Target(ElementType.METHOD)
public @interface NeedLogin {
int value()default 0;
}
- 在想要攔截的方法加入自己的註解
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
test();
test1();
}
@NeedLogin(2)
private void test() {
}
@TestAnnotaion
private void test1() {
}
}
3.在Application設置全局切面攔截處理
public class App extends Application {
private static final String TAG = "App";
@Override
public void onCreate() {
super.onCreate();
OkAspectjHelper.setmHandler(new PointHandler() {
@Override
public void handlePoint(Class clazz, ProceedingJoinPoint joinPoint) {
Log.d(TAG, "handlePoint() called with: clazz = [" + clazz + "]");
if(clazz==NeedLogin.class){
MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature();
NeedLogin annotation = methodSignature.getMethod().getAnnotation(NeedLogin.class);
Log.d(TAG, "handlePoint() called with: joinPoint = [" + annotation.value() + "]");
try {
joinPoint.proceed();
} catch (Throwable throwable) {
throwable.printStackTrace();
}
}
}
});
}
}
4.配置完成,可以在handlePoint(Class clazz, ProceedingJoinPoint joinPoint)中自由發揮你的騷操作了!
5.也可自己編寫切面文件,然後通過調用OkAspectjHelper.notifyHandler(Class clazz,ProceedingJoinPoint joinPoint),發送切點信息進行統一處理.
致謝
- 感謝所有開源庫的大佬
- 借鑑 https://github.com/JakeWharton/butterknife
問題反饋
歡迎加星,打call https://github.com/TanZhiL/OkAspectj
- email:[email protected]
關於作者
譚志龍
開源項目
- 快速切面編程開源庫 https://github.com/TanZhiL/OkAspectj
- 高仿喜馬拉雅聽Android客戶端 https://github.com/TanZhiL/Zhumulangma
License
Copyright (C) tanzhilong OkAspectjFramework Open Source Project
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.