注解實(shí)現(xiàn)三部曲
使用場(chǎng)景
- EventBus使用
@Subscriber(mode = ThreadMode.MAIN, tag = EventTag.STORE_OUT_SUMMARIZING)
internal fun refresh(list: Any) {
getOutList()
}
- bufferknife點(diǎn)擊事件使用
@OnClick(R.id.btn_login)
public void onViewClicked() {
}
原理:和監(jiān)聽有點(diǎn)類似谊惭,通過定義一個(gè)接口實(shí)現(xiàn)注解標(biāo)簽的使用乳愉,然后定義一個(gè)類掉房,通過反射對(duì)所有使用注解的文件進(jìn)行二級(jí)操作茫陆。
那么問題來了仗岖,如果我們想使用注解逃延,該如何做昵?
使用三部曲
- 1.定義一個(gè)接口
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Action {
public String source() default "nothing";
}
解釋:
@Target
@Target說明了Annotation所修飾的對(duì)象范圍:Annotation可被用于
packages轧拄、types(類揽祥、接口、枚舉檩电、Annotation類型)拄丰、類型成員(方法、構(gòu)造方法俐末、成員變量料按、枚舉值)、方法參數(shù)和本地變量(如循環(huán)變量卓箫、catch參數(shù))载矿。在Annotation類型的聲明中使用了target可更加明晰其修飾的目標(biāo)。
取值(ElementType)有:
1.CONSTRUCTOR:用于描述構(gòu)造器
2.FIELD:用于描述域
3.LOCAL_VARIABLE:用于描述局部變量
4.METHOD:用于描述方法
5.PACKAGE:用于描述包
6.PARAMETER:用于描述參數(shù)
7.TYPE:用于描述類烹卒、接口(包括注解類型) 或enum聲明
@Retention
@Retention定義了該Annotation被保留的時(shí)間長(zhǎng)短:某些Annotation僅出現(xiàn)在源代碼中恢准,而被編譯器丟棄魂挂;而另一些卻被編譯在class文件中;編譯在class文件中的Annotation可能會(huì)被虛擬機(jī)忽略馁筐,而另一些在class被裝載時(shí)將被讀取(請(qǐng)注意并不影響class的執(zhí)行坠非,因?yàn)锳nnotation與class在使用上是被分離的)敏沉。使用這個(gè)meta-Annotation可以對(duì)
Annotation的“生命周期”限制。
取值(RetentionPoicy)有:
1.SOURCE:在源文件中有效(即源文件保留)
2.CLASS:在class文件中有效(即class保留)
3.RUNTIME:在運(yùn)行時(shí)有效(即運(yùn)行時(shí)保留)
- 2.注解事件處理器
package com.chen.annotation
import android.view.View
import java.lang.reflect.InvocationHandler
import java.lang.reflect.Method
import java.lang.reflect.Proxy
import king.bird.annotation.Action
/**
* <pre>
* author : Wp
* e-mail : 18141924293@163.com
* time : 2018/09/10
* desc : 注解實(shí)例
* version: 1.0
*</pre>
*/
class ActionInstaller {
companion object {
fun processAnnotations(client: Any) {
val clientClass = client.javaClass
for (m in clientClass.declaredMethods) {
//獲取指定Annotation對(duì)象
val listener = m.getAnnotation(Action::class.java!!)
if (listener != null) {
try {
val f = clientClass.getDeclaredField(listener.source)
f.isAccessible = true
//控件對(duì)象
val focusView = f.get(client)
addListener(focusView, client, m)
} catch (e: Exception) {
e.printStackTrace()
}
}
}
}
@Throws(Exception::class)
private fun addListener(focusView: Any, client: Any, m: Method) {
val handler = InvocationHandler { proxy, method, args ->
//場(chǎng)景類調(diào)用 onBtnClick() 方法
m.invoke(client)
}
val onClickListener = Proxy.newProxyInstance(null, arrayOf<Class<*>>(View.OnClickListener::class.java), handler)
val setOnClickListenerMethod = focusView.javaClass.getMethod("setOnClickListener", View.OnClickListener::class.java)
setOnClickListenerMethod.invoke(focusView, onClickListener)
}
}
}
- 3.實(shí)際應(yīng)用
@Action(source = "mBtnTest")
fun onBtnClick() {
Log.e("Test", "mBtnTest點(diǎn)擊")
}
項(xiàng)目demo詳見Github
參與貢獻(xiàn)
- Fork 本項(xiàng)目
- 新建 Feat_xxx 分支
- 提交代碼
- 新建 Pull Request
github地址
本項(xiàng)目鏈接:https://github.com/pengMaster/annotation