距离我上次发表文章都有超过半年时间了,年前一直在复习,年后一段时间都在找工作,期间还去了一家公司三天,觉得不合适就溜了,感觉挺对不起那家公司的。最后等了一个多月(期间自己也有一段时间去了复习怎么做网页)才入职一家比较知名的国企,拿到自己想要的薪水,也是对上一年自己学习成果的回报吧,也实现了自己不想再待在外包公司小小的愿望。现在回想起2016刚毕业真的觉得有点苦,白天在外包公司工作量成倍的增长,晚上还坚持看书学习做笔记,最后真正实现了逃亡的目标的时候自己也想放松一段日子,所以在2017年的上半年自己基本处于一种半颓废的状态,直到最近在新公司接到一个搭建新框架的任务才重新投入安卓的怀抱,2017年自己的计划也正式的算是开始了。
好吧,不扯个人经历这种无聊的话题了,回归正题,通过一段时间的归纳和总结网上很多我觉得很好的代码,就自己搭建了一套快速开发的框架,下面我就把自己的愚见和网上收集的资料分享一下。
如果不想听我多啰嗦的话可以在GitHub上直接下载源码下载链接
1.首先,介绍一下BaseRecyclerView:
BaseRecyclerView适配器的使用.png
可以看到,封装之后的Adapter,我们只需要重写convert一个方法就可以实现该有的效果,比原来要重写多个方法代码简洁多了。
我们只用了
viewHolder.setText(R.id.tv_title,item.getTitle());
一句代码就完成了对View的初始化和运用,下面我们看看ViewHolder是怎么写的:
public class BaseViewHolder extends RecyclerView.ViewHolder { private SparseArray<View> mViews; public View mConvertView; public Context mContext; public BaseViewHolder(Context context,View itemView) { super(itemView); mViews = new SparseArray<>(); mConvertView = itemView; mContext = context; } public View getConvertView() { return mConvertView; } public static BaseViewHolder createViewHolder(Context context, View itemView){ BaseViewHolder holder = new BaseViewHolder(context, itemView); return holder; } public static BaseViewHolder createViewHolder(Context context, ViewGroup parent, int layoutId){ View itemView = LayoutInflater.from(context).inflate(layoutId, parent, false); BaseViewHolder holder = new BaseViewHolder(context, itemView); return holder; } public BaseViewHolder setText(@IdRes int viewId, CharSequence value) { TextView view = getView(viewId); view.setText(value); return this; } public BaseViewHolder setBackgroundColor(@IdRes int viewId, @ColorInt int color) { View view = getView(viewId); view.setBackgroundColor(color); return this; } public BaseViewHolder setBackgroundRes(@IdRes int viewId, @DrawableRes int backgroundRes) { View view = getView(viewId); view.setBackgroundResource(backgroundRes); return this; } public BaseViewHolder setTextColor(@IdRes int viewId, @ColorInt int textColor) { TextView view = getView(viewId); view.setTextColor(textColor); return this; } public BaseViewHolder setImageDrawable(@IdRes int viewId, Drawable drawable) { ImageView view = getView(viewId); view.setImageDrawable(drawable); return this; } public BaseViewHolder setImageBitmap(@IdRes int viewId, Bitmap bitmap) { ImageView view = getView(viewId); view.setImageBitmap(bitmap); return this; } public BaseViewHolder setVisible(@IdRes int viewId, boolean visible) { View view = getView(viewId); view.setVisibility(visible ? View.VISIBLE : View.GONE); return this; } //关于事件 public BaseViewHolder setOnClickListener(@IdRes int viewId, View.OnClickListener listener) { View view = getView(viewId); view.setOnClickListener(listener); return this; } public BaseViewHolder setOnTouchListener(@IdRes int viewId,View.OnTouchListener listener) { View view = getView(viewId); view.setOnTouchListener(listener); return this; } public BaseViewHolder setOnLongClickListener(@IdRes int viewId,View.OnLongClickListener listener) { View view = getView(viewId); view.setOnLongClickListener(listener); return this; } public <T extends View> T getView(int viewId){ View view = mViews.get(viewId); if(view == null){ view = itemView.findViewById(viewId); mViews.put(viewId,view); } return (T) view; } }
关键的三步:
1.用SparseArray来存储各个View; 2.用getView的方法去集体声明View; 3.用系统给出的setText等方法达到你想要的效果。
简单的封装就能省略了开发中初始化ViewHolder的很多重复繁杂的代码~
在BaseAdapter,我也做了一些小小的封装处理,把adapter的onCreateViewHolder(),onBindViewHolder(),getItemCount()三个必要写的方法也减少为只需要重写convert()一个方法。
BaseRecyclerView其他用法.png
可以看到,在adapter我同样封装了增加头部、尾部和子项监听事件,在下拉刷新,上拉加载方面,我用了之前自己写过的一个SwipeRecyclerView融合在一起,有兴趣的朋友可以去看看SwipeRecyclerView源码下载链接
至于怎么实现,我是参考了网上几个对RecyclerView封装的很好的框架进行修改的。以下是参考的链接:
BaseRecyclerViewHelperAdapter
为RecyclerView打造通用Adapter 让RecyclerView更加好用
2.RxJava和Retrofit2的两种网络链接方式:
因为各人都有自己的爱好,所以我听取别人给我的建议,选择去封装了两种不同链接方式。
(1)分参数上传:
分参数上传.png
分参数上传-View代码.png
分参数上传-Presenter代码.png
这种方式是RxJava和Retrofit2在MVP模式中最普遍的用法,它跟以前很多网络框架不一样要先确定参数和JavaBean,所以也有习惯了以前模式的人吐槽这种方式,特别是Presenter的代码量太多,冗余得代码也多,但是这种封装却是最能体现RxJava和Retrofit2特性的封装,以后查询接口可以一目了然地知道链接和所需要的参数。
这种链接方式比较常见,所以我也不做太多的解释。
以下是参考链接:
GeekNews
(2)批量参数上传:
批量参数上传-View层请求代码.png
批量参数上传-Presenter层请求代码.png
很明显,这种封装Presenter层的代码量看上去会减少很多,同时,这样的封装跟以前很多网络框架的模式很像:
(1)用一个HashMap装载所有的请求参数; (2)url和参数直接在View层显示; (3)在请求的时候返回是一个callback。
下面我们看看代码:
一开始我还以为Retrofit不能批量参数上传的,只能确定好每个参数,但是后来在网上才发现原来还有这种操作,所以就参照着几个好的框架来实现了一次~
public interface BaseServerApi { @POST("{path}") Observable<ResponseBody> post( @Path(value = "path", encoded = true) String path, @QueryMap Map<String, Object> map); @GET("{path}") Observable<ResponseBody> get( @Path(value = "path", encoded = true) String path, @QueryMap Map<String, Object> map); }
Retrofit2的相关配置和Get、Post上传方法
public class RetrofitHelper { private static final int CONNECT_TIME_OUT = 20; private static final int READ_TIME_OUT = 20; private static final int WRITE_TIME_OUT = 20; private static final int MAX_CACHE_SIZE = 1024 * 1024 * 50; private static final String HTTP_CACHE_DIR = Environment.getExternalStorageDirectory() + "/HTTP_LIBRARY_CACHE"; private static Retrofit retrofit; private static BaseServerApi baseServerApi; private static OkHttpClient mOkHttpClient = null; @Inject //初始化Retrofit public RetrofitHelper() { initOkHttp(); retrofit = new Retrofit.Builder() .baseUrl(AppURL.BaseURL) .client(mOkHttpClient) .addConverterFactory(GsonConverterFactory.create()) //自定义的gsonConverter 在里面处理了登录失效处理 .addCallAdapterFactory(RxJavaCallAdapterFactory.create()) .build(); baseServerApi = retrofit.create(BaseServerApi.class); } //get方法 public <T> T Get(String url, Map<String, Object> maps, BaseSubscriber<T> subscriber){ return (T) baseServerApi.get(url,maps) .compose(RxUtil.<ResponseBody>rxSchedulerHelper()) .compose(RxUtil.<T>handleResult()) .subscribe(subscriber); } //post方法 public <T> T Post(String url, Map<String, Object> maps, BaseSubscriber<T> subscriber){ return (T) baseServerApi.post(url,maps) .compose(RxUtil.<ResponseBody>rxSchedulerHelper()) .compose(RxUtil.<T>handleResult()) .subscribe(subscriber); } //配置okHttp的数据 public static void initOkHttp() { OkHttpClient.Builder builder = new OkHttpClient.Builder(); if (Logger.DEBUG) { HttpLoggingInterceptor loggingInterceptor = new HttpLoggingInterceptor(); loggingInterceptor.setLevel(HttpLoggingInterceptor.Level.BODY); builder.addInterceptor(loggingInterceptor); } File cacheFile = new File(HTTP_CACHE_DIR); Cache cache = new Cache(cacheFile, MAX_CACHE_SIZE); Interceptor cacheInterceptor = new CacheInterceptor(); //设置缓存 builder.addNetworkInterceptor(cacheInterceptor); builder.addInterceptor(cacheInterceptor); builder.cache(cache); //设置超时 builder.connectTimeout(CONNECT_TIME_OUT, TimeUnit.SECONDS); builder.readTimeout(READ_TIME_OUT, TimeUnit.SECONDS); builder.writeTimeout(WRITE_TIME_OUT, TimeUnit.SECONDS); //错误重连 builder.retryOnConnectionFailure(true); mOkHttpClient = builder.build(); } }
按照一位以前班里的大神说:
这种方式来说,可读性可能比较差点,但是如果以后你想换一种网络框架,只需要把callback稍微改一下就好了而不需要像第一种那样整个项目都要伤筋动骨,但是这样做却把RxJava的链式结构的特性给抹杀掉了。。。
其实两种链接方式各有各的优点,看你喜欢哪一种,或者按照个人习惯去选择,两种方式我都放在框架里面了。
以下是参考链接:
Android 巧妙封装,基于Retrofit+RxJava网络框架“Leopard”---完整浅析
Novate:Retrofit2.0和RxJava的又一次完美改进加强!
其实,很多时候我们请求数据回来都需要有第一层封装去处理,譬如说超时登录等等,所以接下来我们也来封装一下:
第二种网络链接方式的处理:
作者:19snow93
链接:https://www.jianshu.com/p/fc30267b89d4