Android快速开发之定制BaseTemplate

本篇内容有:

定制BaseActivity
定制BaseFragment
定制BaseApplication

前言

初学者肯定会遇到一个日常任务,那么就是findViewById,setOnClickListener(暂且把它们称为日常任务),而且很多人会把他们混在一起,导致项目结构混乱,最主要的是写多了会烦,不觉得吗?当项目的Activity越多时,每次添加控件都要重新写一次,想想都累

tv_cz_10 = (TextView) findViewById(R.id.tv_cz_10);
tv_cz_20 = (TextView) findViewById(R.id.tv_cz_20);
tv_cz_30 = (TextView) findViewById(R.id.tv_cz_30);
tv_cz_50 = (TextView) findViewById(R.id.tv_cz_50);
tv_cz_10.setOnClickListener(this);
tv_cz_20.setOnClickListener(this);
tv_cz_30.setOnClickListener(this);
tv_cz_50.setOnClickListener(this);

定制解决的问题:尽量写少的代码,做更多事
定制的目的:理清代码结构,让你编程更有逻辑性
定制的内容:一切都是根据项目的需求去实现

定制BaseActivity

我们就针对日常任务简单的定制一份我们的BaseActivity

public abstract class BaseActivity extends FragmentActivity implements View.OnClickListener {

 private SparseArray<View> mViews;

 public abstract int getLayoutId();

 public abstract void initViews();

 public abstract void initListener();

 public abstract void initData();

 public abstract void processClick(View v);

 public void onClick(View v) {
 processClick(v);
 }

 @Override
 protected void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 mViews = new SparseArray<>();
 setContentView(getLayoutId());
 initViews();
 initListener();
 initData();
 }

 /**
 * 通过ID找到View
 */
 public <E extends View> E findView(int viewId) {
 E view = (E) mViews.get(viewId);
 if (view == null) {
 view = (E) findViewById(viewId);
 mViews.put(viewId, view);
 }
 return view;
 }

 /**
 * View设置OnClick事件
 */
 public <E extends View> void setOnClick(E view){
 view.setOnClickListener(this);
 }
}

代码其实很简单,光从代码可能不知道这段代码的意思,那么就从实现这段代码来理解它的真正作用,下面是实现BaseActivity的代码

public class SearchActivity extends BaseActivity {

 BannerController bannerController;
 ShopController shopController;

 private ImageView iv_zxing;
 private TextView tv_sure;

 @Override
 public int getLayoutId() {
 //这里用来获取Activity的layout
 return R.layout.activity_search;
 }

 @Override
 public void initViews() {
 //这里用来初始化View
 iv_zxing = findView(R.id.iv_zxing);
 tv_sure = findView(R.id.tv_sure);
 }

 @Override
 public void initListener() {
 //这里用来初始化点击事件
 setOnClick(iv_zxing);
 setOnClick(tv_sure);
 }

 @Override
 public void initData() {
 //这里用来设置数据、获取数据、读取网络数据、这里所做的一切都可以在Controller实现
 bannerController = new BannerController(getActivity());
 shopController = new ShopController(getActivity());

 initShop();
 initBanner();
 }

 @Override
 public void processClick(View v) {
 //这里用来处理点击事件
 switch (v.getId()) {
 case R.id.iv_zxing:

 break;
 case R.id.tv_sure:

 break;
 }
 }

 private void initShop() {

 }

 private void initBanner() {

 }
}

是不是觉得代码结构很清晰,而且比起之前的日常任务来说,代码确实少了不少,各个方法都放着自己应该做的事情,这样能保证你在编程的时候逻辑不会出错,让别人读起来也很轻松,当然,除了常用的setOnClickListener还有setOnItemClickListener,这就需要根据项目需要而定制

如果你是很酷很有性格的人,那么也可以尝试下面这种用法,用一个字母作为方法,一切定制因你心情而定

public <E extends View> E F(int viewId) {
 E view = (E) mViews.get(viewId);
 if (view == null) {
 view = (E) findViewById(viewId);
 mViews.put(viewId, view);
 }
 return view;
}

public <E extends View> void C(E view){
 view.setOnClickListener(this);
}

//用起来也很帅哦
@Override
public void initViews() {
 iv_zxing = F(R.id.iv_zxing);
 tv_sure = F(R.id.tv_sure);
}

@Override
public void initListener() {
 C(iv_zxing);
 C(tv_sure);
}

定制BaseFragment

介绍完了Activity,那么Fragment就很简单了,可以模仿Activity实现,如果和上面的一模一样那么就没有乐趣了,这里由于个人项目原因,我把Fragment默认设置成了懒加载模式,并且只加载一次数据

public abstract class BaseFragment extends Fragment implements View.OnClickListener {

 private boolean isVisible = false;
 private boolean isInitView = false;
 private boolean isFirstLoad = true;

 public View convertView;
 private SparseArray<View> mViews;

 public abstract int getLayoutId();

 public abstract void initViews();

 public abstract void initListener();

 public abstract void initData();

 public abstract void processClick(View v);

 @Override
 public void onClick(View v) {
 processClick(v);
 }

 @Override
 public void setUserVisibleHint(boolean isVisibleToUser) {
 super.setUserVisibleHint(isVisibleToUser);
 if (isVisibleToUser) {
 isVisible = true;
 lazyLoad();
 } else {
 //设置已经不是可见的
 isVisible = false;
 }
 }

 @Override
 public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
 mViews = new SparseArray<>();
 convertView = inflater.inflate(getLayoutId(), container, false);
 initViews();

 isInitView = true;
 lazyLoad();
 return convertView;
 }

 //懒加载
 private void lazyLoad() {
 if (!isFirstLoad || !isVisible || !isInitView) {
 //如果不是第一次加载、不是可见的、不是初始化View,则不加载数据
 return;
 }
 //加载数据
 initListener();
 initData();
 //设置已经不是第一次加载
 isFirstLoad = false;
 }

 public <E extends View> E findView(int viewId) {
 if (convertView != null) {
 E view = (E) mViews.get(viewId);
 if (view == null) {
 view = (E) convertView.findViewById(viewId);
 mViews.put(viewId, view);
 }
 return view;
 }
 return null;
 }

 public <E extends View> void setOnClick(E view){
 view.setOnClickListener(this);
 }
}

这里和Activity最大的区别

1.convertView:由于Fragment的findID需要convertView,我们只好抽取出来
2.setUserVisibleHint:这个方法当切换Fragment时会调用,会返回当前Fragment是否用户可见

public class HomeFragment extends BaseFragment {

 ShopController shopController;

 private ImageView iv_speech;
 private TextView tv_search;

 @Override
 public int getLayoutId() {
 return R.layout.fragment_home;
 }

 @Override
 public void initViews() {
 iv_speech = findView(R.id.iv_speech);
 tv_search = findView(R.id.tv_search);
 }

 @Override
 public void initData() {
 shopController = new ShopController(getActivity());

 initShop();
 }

 @Override
 public void initListener() {
 setOnClick(iv_speech);
 setOnClick(tv_search);
 }

 @Override
 public void processClick(View v) {
 switch (v.getId()) {
 case R.id.iv_speech:

 break;
 case R.id.tv_search:

 break;
 }
 }

 private void initShop() {

 }
}

这里采用的是ViewPager+Fragment,如果需要让Fragment进行缓存,那么必须对ViewPager进行缓存设置

//设置缓存页面
viewPager.setOffscreenPageLimit(5);

下面是设置了缓存的懒加载模式的效果图,可以看到第一次切换需要加载数据,而第二次切回去则界面不会变化,效果和手机淘宝一样

定制BaseApplication

定制BaseApplication那就简单了,在Application中经常用到的就是第三方的设置、数据库的加载,具体可以根据项目需求进行定制

public abstract class BaseApplication extends Application {

 public abstract void initConfigs();

 @Override
 public void onCreate() {
 super.onCreate();
 initConfigs();
 }

}

结语

学习完之后,建议大家将BaseTemplate用到你们的项目中,当然从中也要学习抽象方法,抽取常用的方法,比如:在加载数据的时候可以抽取BaseController,在Adapter中可以抽取通用的BaseAdapter,具体还需要大家去研究。

源码下载:BaseTemplate

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android中BaseActivity自定义标题栏

    再做一个项目的时候,要求标题栏的标题再中间,样式,字体大小都要自定义.左边一个返回按钮,一个关闭按钮,右边定义一个提交按钮,有时候显示有时候隐藏.因为原生的title标题是再左边的,然后去给Titlebar设置自定义View的时候,也会不尽人意,标题不是再正中间的,标题栏太高等问题. 我们要求的是这样的,右边的按钮可以显示或者隐藏. 于是就决定自己写一个BaseActivity,所有的都去继承这个基类,然后自己去定义标题栏的样式就可以就可以了. 下面来讲一下这个界面是怎么实现的: 首先定义一个类

  • Android快速开发之定制BaseTemplate

    本篇内容有: 定制BaseActivity 定制BaseFragment 定制BaseApplication 前言 初学者肯定会遇到一个日常任务,那么就是findViewById,setOnClickListener(暂且把它们称为日常任务),而且很多人会把他们混在一起,导致项目结构混乱,最主要的是写多了会烦,不觉得吗?当项目的Activity越多时,每次添加控件都要重新写一次,想想都累 tv_cz_10 = (TextView) findViewById(R.id.tv_cz_10); tv_

  • Android快速开发系列 10个常用工具类实例代码详解

    打开大家手上的项目,基本都会有一大批的辅助类,今天特此整理出10个基本每个项目中都会使用的工具类,用于快速开发~~在此感谢群里给我发项目中工具类的兄弟/姐妹~ 1.日志工具类L.java package com.zhy.utils; import android.util.Log; /** * Log统一管理类 * * * */ public class L { private L() { /* cannot be instantiated */ throw new UnsupportedOpe

  • 实例详解Android快速开发工具类总结

    一.日志工具类 Log.java public class L { private L() { /* 不可被实例化 */ throw new UnsupportedOperationException("Cannot be instantiated!"); } // 是否需要打印bug,可以在application的onCreate函数里面初始化 public static boolean isDebug = true; private static final String TAG

  • Android NDK开发入门

    神秘的Android NDK开发往往众多程序员感到兴奋,但又不知它为何物,由于近期开发应用时,为了是开发的.apk文件不被他人解读(反编译),查阅了很多资料,其中有提到使用NDK开发,怀着好奇的心理,通过在线视频教育网站,我初步了解了NDK的神秘面纱,好东西自然要分享,接下来我们就一起来认识一下Android NDK开发. 一.NDK产生的背景 Android平台从诞生起,就已经支持C.C++开发.众所周知,Android的SDK基于Java实现,这意味着基于Android SDK进行开发的第三

  • Android NDK开发详细介绍

    Android之NDK开发 一.NDK产生的背景 Android平台从诞生起,就已经支持C.C++开发.众所周知,Android的SDK基于Java实现,这意味着基于Android SDK进行开发的第三方应用都必须使用Java语言.但这并不等同于"第三方应用只能使用Java".在Android SDK首次发布时,Google就宣称其虚拟机Dalvik支持JNI编程方式,也就是第三方应用完全可以通过JNI调用自己的C动态库,即在Android平台上,"Java+C"的

  • Android程序开发之防止密码输入错误 密码明文显示功能

    在使用App的时候,首次登录都需要用户输入密码的,有些朋友为了安全起见密码设置的比较长,导致很多次密码都输入错误,严重影响了用户体验效果.这一点移动开发者做好了准备工作,因为手机的私密性比较强,在输入密码的时候,可以显示输入,增强准确性,提升用户体验度.这当然要付出代价的,需要额外的代码编写功能.下面通过本文给大家介绍如何编写密码明文显示的功能,仅供参考. 本文源码的GitHub下载地址 要点 (1) 重写EditText, 添加提示密码显示和隐藏的图片. (2) 判断点击位置, 切换EditT

  • Android NDK 开发教程

    Android NDK 是在SDK前面又加上了"原生"二字,即Native Development Kit,因此又被Google称为"NDK". 众所周知,Android程序运行在Dalvik虚拟机中,NDK允许用户使用类似C / C++之类的原生代码语言执行部分程序. NDK包括了: 从C / C++生成原生代码库所需要的工具和build files. 将一致的原生库嵌入可以在Android设备上部署的应用程序包文件(application packages fi

  • Android实现的状态栏定制和修改方法

    本文实例讲述了Android实现的状态栏定制和修改方法.分享给大家供大家参考.具体如下: 大家都知道定制在android开发中的重要性,因为通过定制,你才能制造出差异化的产品,才能满足更多消费者的需求, 像HTC生产的手机都通过了深层次的二次开发,今天我也来分享一下我的状态栏定制. 废话不说了,直接上图: 主要更换了背景,文字颜色以及icon的显示顺序. 2. 关键代码部分 a) 代码在系统中的位置 status bar 的相关代码位于:frameworks/base/services/java

  • Android高级开发之性能优化典范

    本章介绍android高级开发中,对于性能方面的处理.主要包括电量,视图,内存三个性能方面的知识点. 1.视图性能 (1)Overdraw简介 Overdraw就是过度绘制,是指在一帧的时间内(16.67ms)像素被绘制了多次,理论上一个像素每次只绘制一次是最优的,但是由于重叠的布 局导致一些像素会被多次绘制,而每次绘制都会对应到CPU的一组绘图命令和GPU的一些操作,当这个操作耗时超过16.67ms时,就会出现掉帧现象,表现为应用卡顿,所以对重叠不可见元素的重复绘制会产生额外的开销,需要尽量减

  • Android编程开发之性能优化技巧总结

    本文详细总结了Android编程开发之性能优化技巧.分享给大家供大家参考,具体如下: 1.http用gzip压缩,设置连接超时时间和响应超时时间 http请求按照业务需求,分为是否可以缓存和不可缓存,那么在无网络的环境中,仍然通过缓存的httpresponse浏览部分数据,实现离线阅读. 2.listview 性能优化 1).复用convertView 在getItemView中,判断convertView是否为空,如果不为空,可复用.如果couvertview中的view需要添加listern

随机推荐