IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 移动开发 -> GitHub标星5-8K+,附面试题 -> 正文阅读

[移动开发]GitHub标星5-8K+,附面试题


那这功能是在哪里做的呢?答案在下面的代码中。

//防重复点击间隔(秒)
public static final int CLICK_INTERVAL = 1;

/**

  • requireAll 是意思是是否需要绑定全部参数, false为否
  • View的onClick事件绑定
  • onClickCommand 绑定的命令,
  • isThrottleFirst 是否开启防止过快点击
    */
    @BindingAdapter(value = {“onClickCommand”, “isThrottleFirst”}, requireAll = false)
    public static void onClickCommand(View view, final BindingCommand clickCommand, final boolean isThrottleFirst) {
    if (isThrottleFirst) {
    RxView.clicks(view)
    .subscribe(new Consumer() {
    @Override
    public void accept(Object object) throws Exception {
    if (clickCommand != null) {
    clickCommand.execute();
    }
    }
    });
    } else {
    RxView.clicks(view)
    .throttleFirst(CLICK_INTERVAL, TimeUnit.SECONDS)//1秒钟内只允许点击1次
    .subscribe(new Consumer() {
    @Override
    public void accept(Object object) throws Exception {
    if (clickCommand != null) {
    clickCommand.execute();
    }
    }
    });
    }
    }

onClickCommand方法是自定义的,使用@BindingAdapter注解来标明这是一个绑定方法。在方法中使用了RxView来增强view的clicks事件,.throttleFirst()限制订阅者在指定的时间内重复执行,最后通过BindingCommand将事件回调出去,就好比有一种拦截器,在点击时先做一下判断,然后再把事件沿着他原有的方向传递。

是不是觉得有点意思,好戏还在后头呢!

##### [](

)2.2.3、自定义ImageView图片加载

绑定图片路径:

在ViewModel中定义

public String imgUrl = “http://img0.imgtn.bdimg.com/it/u=2183314203,562241301&fm=26&gp=0.jpg”;


在ImageView标签中

binding:url="@{viewModel.imgUrl}"


url是图片路径,这样绑定后,这个ImageView就会去显示这张图片,不限网络图片还是本地图片。

如果需要给一个默认加载中的图片,可以加这一句

binding:placeholderRes="@{R.mipmap.ic_launcher_round}"


> R文件需要在data标签中导入使用,如:`<import type="com.goldze.mvvmhabit.R" />`

BindingAdapter中的实现

@BindingAdapter(value = {“url”, “placeholderRes”}, requireAll = false)
public static void setImageUri(ImageView imageView, String url, int placeholderRes) {
if (!TextUtils.isEmpty(url)) {
//使用Glide框架加载图片
Glide.with(imageView.getContext())
.load(url)
.placeholder(placeholderRes)
.into(imageView);
}
}


很简单就自定义了一个ImageView图片加载的绑定,学会这种方式,可自定义扩展。

> 如果你对这些感兴趣,可以下载源码,在binding包中可以看到各类控件的绑定实现方式

##### [](

)2.2.4、RecyclerView绑定

> RecyclerView也是很常用的一种控件,传统的方式需要针对各种业务要写各种Adapter,如果你使用了mvvmhabit,则可大大简化这种工作量,从此告别setAdapter()。

在ViewModel中定义:

//给RecyclerView添加items
public final ObservableList observableList = new ObservableArrayList<>();
//给RecyclerView添加ItemBinding
public final ItemBinding itemBinding = ItemBinding.of(BR.viewModel, R.layout.item_network);


ObservableList<>和ItemBinding<>的泛型是Item布局所对应的ItemViewModel

在xml中绑定

<android.support.v7.widget.RecyclerView
android:layout_width=“match_parent”
android:layout_height=“match_parent”
binding:itemBinding="@{viewModel.itemBinding}"
binding:items="@{viewModel.observableList}"
binding:layoutManager="@{LayoutManagers.linear()}"
binding:lineManager="@{LineManagers.horizontal()}" />


layoutManager控制是线性(包含水平和垂直)排列还是网格排列,lineManager是设置分割线

网格布局的写法:`binding:layoutManager="@{LayoutManagers.grid(3)}`
水平布局的写法:`binding:layoutManager="@{LayoutManagers.linear(LinearLayoutManager.HORIZONTAL,Boolean.FALSE)}"`

使用到相关类,则需要导入该类才能使用,和导入Java类相似

> `<import type="me.tatarka.bindingcollectionadapter2.LayoutManagers" />`
> `<import type="me.goldze.mvvmhabit.binding.viewadapter.recyclerview.LineManagers" />`
> `<import type="android.support.v7.widget.LinearLayoutManager" />`

这样绑定后,在ViewModel中调用ObservableList的add()方法,添加一个ItemViewModel,界面上就会实时绘制出一个Item。在Item对应的ViewModel中,同样可以以绑定的形式完成逻辑

> 可以在请求到数据后,循环添加`observableList.add(new NetWorkItemViewModel(NetWorkViewModel.this, entity));`详细可以参考例子程序中NetWorkViewModel类。

**注意:** 在以前的版本中,ItemViewModel是继承BaseViewModel,传入Context,新版本3.x中可继承ItemViewModel,传入当前页面的ViewModel

更多RecyclerView、ListView、ViewPager等绑定方式,请参考 [https://github.com/evant/binding-collection-adapter](

)

### [](

)2.3、网络请求

> 网络请求一直都是一个项目的核心,现在的项目基本都离不开网络,一个好用网络请求框架可以让开发事半功倍。

#### [](

)2.3.1、Retrofit+Okhttp+RxJava

> 现今,这三个组合基本是网络请求的标配,如果你对这三个框架不了解,建议先去查阅相关资料。

square出品的框架,用起来确实非常方便。**MVVMHabit**中引入了

api “com.squareup.okhttp3:okhttp:3.10.0”
api “com.squareup.retrofit2:retrofit:2.4.0”
api “com.squareup.retrofit2:converter-gson:2.4.0”
api “com.squareup.retrofit2:adapter-rxjava2:2.4.0”


构建Retrofit时加入

Retrofit retrofit = new Retrofit.Builder()
.addConverterFactory(GsonConverterFactory.create())
.addCallAdapterFactory(RxJavaCallAdapterFactory.create())
.build();


或者直接使用例子程序中封装好的RetrofitClient

#### [](

)2.3.2、网络拦截器

**LoggingInterceptor:** 全局拦截请求信息,格式化打印Request、Response,可以清晰的看到与后台接口对接的数据,

LoggingInterceptor mLoggingInterceptor = new LoggingInterceptor
.Builder()//构建者模式
.loggable(true) //是否开启日志打印
.setLevel(Level.BODY) //打印的等级
.log(Platform.INFO) // 打印类型
.request(“Request”) // request的Tag
.response(“Response”)// Response的Tag
.addHeader(“version”, BuildConfig.VERSION_NAME)//打印版本
.build()


构建okhttp时加入

OkHttpClient okHttpClient = new OkHttpClient.Builder()
.addInterceptor(mLoggingInterceptor)
.build();


**CacheInterceptor:** 缓存拦截器,当没有网络连接的时候自动读取缓存中的数据,缓存存放时间默认为3天。
创建缓存对象

//缓存时间
int CACHE_TIMEOUT = 10 * 1024 * 1024
//缓存存放的文件
File httpCacheDirectory = new File(mContext.getCacheDir(), “goldze_cache”);
//缓存对象
Cache cache = new Cache(httpCacheDirectory, CACHE_TIMEOUT);


构建okhttp时加入

OkHttpClient okHttpClient = new OkHttpClient.Builder()
.cache(cache)
.addInterceptor(new CacheInterceptor(mContext))
.build();


#### [](

)2.3.3、Cookie管理

**MVVMHabit**提供两种CookieStore:**PersistentCookieStore** (SharedPreferences管理)和**MemoryCookieStore** (内存管理),可以根据自己的业务需求,在构建okhttp时加入相应的cookieJar

OkHttpClient okHttpClient = new OkHttpClient.Builder()
.cookieJar(new CookieJarImpl(new PersistentCookieStore(mContext)))
.build();


或者

OkHttpClient okHttpClient = new OkHttpClient.Builder()
.cookieJar(new CookieJarImpl(new MemoryCookieStore()))
.build();


#### [](

)2.3.4、绑定生命周期

请求在ViewModel层。默认在BaseActivity中注入了LifecycleProvider对象到ViewModel,用于绑定请求的生命周期,View与请求共存亡。

RetrofitClient.getInstance().create(DemoApiService.class)
.demoGet()
.compose(RxUtils.bindToLifecycle(getLifecycleProvider())) // 请求与View周期同步
.compose(RxUtils.schedulersTransformer()) // 线程调度
.compose(RxUtils.exceptionTransformer()) // 网络错误的异常转换
.subscribe(new Consumer<BaseResponse>() {
@Override
public void accept(BaseResponse response) throws Exception {

    }
}, new Consumer<ResponseThrowable>() {
    @Override
    public void accept(ResponseThrowable throwable) throws Exception {

    }
});

在请求时关键需要加入组合操作符`.compose(RxUtils.bindToLifecycle(getLifecycleProvider()))`
**注意:** 由于BaseActivity/BaseFragment都实现了LifecycleProvider接口,并且默认注入到ViewModel中,所以在调用请求方法时可以直接调用getLifecycleProvider()拿到生命周期接口。如果你没有使用 **mvvmabit** 里面的BaseActivity或BaseFragment,使用自己定义的Base,那么需要让你自己的Activity继承RxAppCompatActivity、Fragment继承RxFragment才能用`RxUtils.bindToLifecycle(lifecycle)`方法。

#### [](

)2.3.5、网络异常处理

网络异常在网络请求中非常常见,比如请求超时、解析错误、资源不存在、服务器内部错误等,在客户端则需要做相应的处理(当然,你可以把一部分异常甩锅给网络,比如当出现code 500时,提示:请求超时,请检查网络连接,此时偷偷将异常信息发送至后台(手动滑稽))。

在使用Retrofit请求时,加入组合操作符`.compose(RxUtils.exceptionTransformer())`,当发生网络异常时,回调onError(ResponseThrowable)方法,可以拿到异常的code和message,做相应处理。

> mvvmhabit中自定义了一个[ExceptionHandle](

),已为你完成了大部分网络异常的判断,也可自行根据项目的具体需求调整逻辑。

**注意:** 这里的网络异常code,并非是与服务端协议约定的code。网络异常可以分为两部分,一部分是协议异常,即出现code = 404、500等,属于HttpException,另一部分为请求异常,即出现:连接超时、解析错误、证书验证失等。而与服务端约定的code规则,它不属于网络异常,它是属于一种业务异常。在请求中可以使用RxJava的filter(过滤器),也可以自定义BaseSubscriber统一处理网络请求的业务逻辑异常。由于每个公司的业务协议不一样,所以具体需要你自己来处理该类异常。

## [](

)3、辅助功能

> 一个完整的快速开发框架,当然也少不了常用的辅助类。下面来介绍一下**MVVMabit**中有哪些辅助功能。

### [](

)3.1、事件总线

> 事件总线存在的优点想必大家都很清楚了,android自带的广播机制对于组件间的通信而言,使用非常繁琐,通信组件彼此之间的订阅和发布的耦合也比较严重,特别是对于事件的定义,广播机制局限于序列化的类(通过Intent传递),不够灵活。

#### [](

)3.3.1、RxBus

RxBus并不是一个库,而是一种模式。相信大多数开发者都使用过EventBus,对RxBus也是很熟悉。由于**MVVMabit**中已经加入RxJava,所以采用了RxBus代替EventBus作为事件总线通信,以减少库的依赖。

使用方法:

在ViewModel中重写registerRxBus()方法来注册RxBus,重写removeRxBus()方法来移除RxBus

//订阅者
private Disposable mSubscription;
//注册RxBus
@Override
public void registerRxBus() {
super.registerRxBus();
mSubscription = RxBus.getDefault().toObservable(String.class)
.subscribe(new Consumer() {
@Override
public void accept(String s) throws Exception {

        }
    });
//将订阅者加入管理站
RxSubscriptions.add(mSubscription);

}

//移除RxBus
@Override
public void removeRxBus() {
super.removeRxBus();
//将订阅者从管理站中移除
RxSubscriptions.remove(mSubscription);
}


在需要执行回调的地方发送

RxBus.getDefault().post(object);


#### [](

)3.3.2、Messenger

Messenger是一个轻量级全局的消息通信工具,在我们的复杂业务中,难免会出现一些交叉的业务,比如ViewModel与ViewModel之间需要有数据交换,这时候可以轻松地使用Messenger发送一个实体或一个空消息,将事件从一个ViewModel回调到另一个ViewModel中。

使用方法:

定义一个静态String类型的字符串token

public static final String TOKEN_LOGINVIEWMODEL_REFRESH = “token_loginviewmodel_refresh”;


在ViewModel中注册消息监听

//注册一个空消息监听
//参数1:接受人(上下文)
//参数2:定义的token
//参数3:执行的回调监听
Messenger.getDefault().register(this, LoginViewModel.TOKEN_LOGINVIEWMODEL_REFRESH, new BindingAction() {
@Override
public void call() {

}

});

//注册一个带数据回调的消息监听
//参数1:接受人(上下文)
//参数2:定义的token
//参数3:实体的泛型约束
//参数4:执行的回调监听
Messenger.getDefault().register(this, LoginViewModel.TOKEN_LOGINVIEWMODEL_REFRESH, String.class, new BindingConsumer() {
@Override
public void call(String s) {

}

});


在需要回调的地方使用token发送消息

//发送一个空消息
//参数1:定义的token
Messenger.getDefault().sendNoMsg(LoginViewModel.TOKEN_LOGINVIEWMODEL_REFRESH);

//发送一个带数据回调消息
//参数1:回调的实体
//参数2:定义的token
Messenger.getDefault().send(“refresh”,LoginViewModel.TOKEN_LOGINVIEWMODEL_REFRESH);


> token最好不要重名,不然可能就会出现逻辑上的bug,为了更好的维护和清晰逻辑,建议以`aa_bb_cc`的格式来定义token。aa:TOKEN,bb:ViewModel的类名,cc:动作名(功能名)。

> 为了避免大量使用Messenger,建议只在ViewModel与ViewModel之间使用,View与ViewModel之间采用ObservableField去监听UI上的逻辑,可在继承了Base的Activity或Fragment中重写initViewObservable()方法来初始化UI的监听

注册了监听,当然也要解除它。在BaseActivity、BaseFragment的onDestroy()方法里已经调用`Messenger.getDefault().unregister(viewModel);`解除注册,所以不用担心忘记解除导致的逻辑错误和内存泄漏。

### [](

)3.2、文件下载

文件下载几乎是每个app必备的功能,图文的下载,软件的升级等都要用到,mvvmhabit使用Retrofit+Okhttp+RxJava+RxBus实现一行代码监听带进度的文件下载。

下载文件

String loadUrl = “你的文件下载路径”;
String destFileDir = context.getCacheDir().getPath(); //文件存放的路径
String destFileName = System.currentTimeMillis() + “.apk”;//文件存放的名称
DownLoadManager.getInstance().load(loadUrl, new ProgressCallBack(destFileDir, destFileName) {
@Override
public void onStart() {
//RxJava的onStart()
}

@Override
public void onCompleted() {
    //RxJava的onCompleted()
}

@Override
public void onSuccess(ResponseBody responseBody) {
    //下载成功的回调
}

@Override
public void progress(final long progress, final long total) {
    //下载中的回调 progress:当前进度 ,total:文件总大小
}

@Override
public void onError(Throwable e) {
    //下载错误回调
}

});


> 在ProgressResponseBody中使用了RxBus,发送下载进度信息到ProgressCallBack中,继承ProgressCallBack就可以监听到下载状态。回调方法全部执行在主线程,方便UI的更新,详情请参考例子程序。

### [](

)3.3、ContainerActivity

一个盛装Fragment的一个容器(代理)Activity,普通界面只需要编写Fragment,使用此Activity盛装,这样就不需要每个界面都在AndroidManifest中注册一遍

使用方法:

在ViewModel中调用BaseViewModel的方法开一个Fragment

startContainerActivity(你的Fragment类名.class.getCanonicalName())


在ViewModel中调用BaseViewModel的方法,携带一个序列化实体打开一个Fragment

Bundle mBundle = new Bundle();
mBundle.putParcelable(“entity”, entity);
startContainerActivity(你的Fragment类名.class.getCanonicalName(), mBundle);


在你的Fragment中取出实体

Bundle mBundle = getArguments();
if (mBundle != null) {
entity = mBundle.getParcelable(“entity”);
}


### [](

)3.4、6.0权限申请

> 对RxPermissions已经熟悉的朋友可以跳过。

使用方法:

例如请求相机权限,在ViewModel中调用

//请求打开相机权限
RxPermissions rxPermissions = new RxPermissions((Activity) context);
rxPermissions.request(Manifest.permission.CAMERA)
.subscribe(new Consumer() {
@Override
public void accept(Boolean aBoolean) throws Exception {
if (aBoolean) {
ToastUtils.showShort(“权限已经打开,直接跳入相机”);
} else {

  移动开发 最新文章
Vue3装载axios和element-ui
android adb cmd
【xcode】Xcode常用快捷键与技巧
Android开发中的线程池使用
Java 和 Android 的 Base64
Android 测试文字编码格式
微信小程序支付
安卓权限记录
知乎之自动养号
【Android Jetpack】DataStore
上一篇文章           查看所有文章
加:2021-09-11 18:56:29  更:2021-09-11 18:59:01 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/23 16:56:43-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码