寫在前面
最近一直在找時(shí)間重構(gòu)代碼,每一次重構(gòu)都能帶來(lái)許多好處,比如精簡(jiǎn)代碼,提高代碼質(zhì)量,減輕團(tuán)隊(duì)之間的問(wèn)題,當(dāng)然最重要的就是以后可以偷懶啦。而這次改進(jìn)也是為了節(jié)省時(shí)間,提高團(tuán)隊(duì)的效率。
先體驗(yàn)一下效果


DataBinding
不了解的請(qǐng)百度,google或翻看以前的文章
AspectJ
貼一張圖:
簡(jiǎn)單理解就是在編譯期和加載時(shí)進(jìn)行代碼注入。通俗點(diǎn)來(lái)說(shuō)就是非侵入的在一段方法前后加入一點(diǎn)自己的邏輯。作用和OKhttp的Interceptor攔截請(qǐng)求有點(diǎn)像。
最開始認(rèn)識(shí)到AspectJ的時(shí)候是在今年二月份,在github搜索databinding的時(shí)候偶遇north2016/T-MVP,這個(gè)項(xiàng)目太贊了,給了我相當(dāng)多的啟發(fā),也算是我android-AOP的啟蒙。
想了解AOP的推薦文章:
安卓AOP三劍客:APT,AspectJ,Javassist
PS:以前上學(xué)的時(shí)候?qū)WSSH的時(shí)候有接觸過(guò)AOP面向切面編程,雖然都忘了,但是思想還在(大學(xué)真該好好學(xué),感謝泡圖書館的日子)
防止多次點(diǎn)擊
現(xiàn)在來(lái)說(shuō)說(shuō),防止多次點(diǎn)擊。
以前使用的是J神的Rxbinding庫(kù),結(jié)合throttleFirst操作符來(lái)進(jìn)行控制。
由于使用的是DataBinding庫(kù),一般來(lái)說(shuō)事件是綁定在xml當(dāng)中的,但為了限制頻繁觸發(fā)和使用Rxbinding,還是在代碼里進(jìn)行事件處理。
RxView.clicks(view).throttleFirst(600, TimeUnit.MILLISECONDS).subscribe(...);
這樣可以處理普通布局中的點(diǎn)擊事件,但RecyclerView中的Item就不那么好處理了,需要獲取ItemBinding找到view去做以上的操作,很麻煩。
為了節(jié)約時(shí)間(偷懶),提高效率(偷懶),加入AspectJ就勢(shì)在必行了。
加入AspectJ,給你的DataBinding插上翅膀
如何在項(xiàng)目中使用AspecJ呢?
-
使用大神們提供的腳本,android10寫過(guò)一個(gè)Android-AOPExample
的庫(kù),并提供了一段腳本,可以作為參考。
本文推薦使用第二種,比較簡(jiǎn)單方便。
接入方法可以看ReadMe或者徐醫(yī)生的這篇文章看AspectJ在Android中的強(qiáng)勢(shì)插入
講重點(diǎn)
配置好AspectJ后,我們需要給點(diǎn)擊事件加一個(gè)切入點(diǎn),首先添加一個(gè)注解
@Retention(RetentionPolicy.CLASS)
@Target(ElementType.METHOD)
public @interface SingleClick {
}
然后編寫我們的Aspect類(代碼來(lái)自north2016)
/**
* Created by baixiaokang on 16/12/9.
* {link https://github.com/north2016/T-MVP/blob/master/app/src/main/java/com/aop/SingleClickAspect.java}
* 防止View被連續(xù)點(diǎn)擊,間隔時(shí)間600ms
*/
@Aspect
public class SingleClickAspect {
public static final String TAG="SingleClickAspect";
public static final int MIN_CLICK_DELAY_TIME = 600;
static int TIME_TAG = R.id.click_time;
@Pointcut("execution(@com.ditclear.app.aop.annotation.SingleClick * *(..))")//方法切入點(diǎn)
public void methodAnnotated(){
}
@Around("methodAnnotated()")//在連接點(diǎn)進(jìn)行方法替換
public void aroundJoinPoint(ProceedingJoinPoint joinPoint) throws Throwable{
View view=null;
for (Object arg: joinPoint.getArgs()) {
if (arg instanceof View) view= ((View) arg);
}
if (view!=null){
Object tag=view.getTag(TIME_TAG);
long lastClickTime= (tag!=null)? (long) tag :0;
if (BuildConfig.DEBUG) {
Log.d(TAG, "lastClickTime:" + lastClickTime);
}
long currentTime = Calendar.getInstance().getTimeInMillis();
if (currentTime - lastClickTime > MIN_CLICK_DELAY_TIME) {//過(guò)濾掉600毫秒內(nèi)的連續(xù)點(diǎn)擊
view.setTag(TIME_TAG, currentTime);
if (BuildConfig.DEBUG) {
Log.d(TAG, "currentTime:" + currentTime);
}
joinPoint.proceed();//執(zhí)行原方法
}
}
}
}
接下來(lái)是使用
先看看MainActivity
public class MainActivity extends AppCompatActivity {
private ActivityMainBinding mBinding;
private MainViewModel mViewModel;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mBinding = DataBindingUtil.setContentView(this, R.layout.activity_main);
mViewModel = new MainViewModel();
mBinding.setVm(mViewModel);
}
}
很簡(jiǎn)單,初始化binding,設(shè)置了viewModel
MainViewModel
package com.ditclear.app;
import android.databinding.BaseObservable;
import android.databinding.ObservableField;
import android.view.View;
import com.ditclear.app.aop.annotation.SingleClick;
/**
* 頁(yè)面描述:viewmodel
*
* Created by ditclear on 2017/8/12.
*/
public class MainViewModel extends BaseObservable {
public ObservableField<String > normalText=new ObservableField<>("");
public ObservableField<String > hookText=new ObservableField<>("");
/**
* 普通的點(diǎn)擊事件
* @param view view
*/
public void onNormalClick(View view) {
normalText.set(String.format("%s click\n",normalText.get()));
}
/**
* 防止多次點(diǎn)擊
* @param view view
*/
@SingleClick
public void onHookClick(View view) {
hookText.set(String.format("%s click\n",hookText.get()));
}
}
普通點(diǎn)擊事件和需要攔截的方法以@SingleClick注解來(lái)區(qū)分,由于需要在SingleClickAspect類中攔截view 參數(shù),獲取點(diǎn)擊時(shí)間,所以需要傳遞view參數(shù)
@Around("methodAnnotated()")//在連接點(diǎn)進(jìn)行方法替換
public void aroundJoinPoint(ProceedingJoinPoint joinPoint) throws Throwable{
View view=null;
for (Object arg: joinPoint.getArgs()) {
if (arg instanceof View) view= ((View) arg);
}
if (view!=null){
...
}
}
activity_main.xml
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="match_parent"
android:layout_weight="1"
android:gravity="center_horizontal"
android:orientation="vertical">
<Button
android:id="@+id/hook_btn"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="@{(v)->vm.onHookClick(v)}"
android:text="hook"/>
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:gravity="center"
android:text="@{vm.hookText}"/>
</LinearLayout>
大功告成,當(dāng)我們點(diǎn)擊的時(shí)候在執(zhí)行點(diǎn)擊事件的同時(shí)會(huì)在控制臺(tái)輸出時(shí)間

而且只有兩次點(diǎn)擊時(shí)間間隔在600ms以上才會(huì)執(zhí)行方法,否則會(huì)被攔截。
最后
DataBinding庫(kù)解決了View和Data之間的綁定問(wèn)題,再搭配上AspectJ真的是如虎添翼。而且二者的功能都遠(yuǎn)不止如此,AspectJ還能進(jìn)行日志埋點(diǎn),性能監(jiān)控,動(dòng)態(tài)權(quán)限申請(qǐng)等等,看你發(fā)揮。
githud地址:https://github.com/ditclear/DataBinding-AspectJ