1. 模式的定义?
适配器模式就是将 某一些对象转成我们需要适配的对象。分为两种:类适配和对象适配;
- 类适配:直接传递类,请看simple2;
- 对象适配:直接传递对象,请看simple3;
比如说:ListView或者RecyclerView,从后台请求的数据是 列表数据(对象数组或者list集合),而ListView或者RecyclerView里边显示的是 View ,基于两者不匹配,所以就采用adapter设计模式将 对象数组适配成 ListView需要的View;
2. 模式的例子 - 支付事例
比如支付系统,第一个版本只是支持RMB,第二个版本需要支持 USD;
2.1: 类适配
人民币需要兼容美元,采用适配器模式,角色就是:
- 需要适配的接口UsdTarget(把人民币转为美元);
- 适配器对象Adapter;
1>:需要适配的接口:
/**
* Email: 2185134304@qq.com
* Created by Novate 2018/5/27 10:19
* Version 1.0
* Params:
* Description: 需要适配的目标接口usd(美元)
*/
public interface UsdTarget {
float getUsd() ;
}
2>:适配器对象Adapter
/**
* Email: 2185134304@qq.com
* Created by Novate 2018/5/27 10:20
* Version 1.0
* Params:
* Description: 适配器对象 - 把人民币转为美元
*/
public class UsdAdapter extends RMBAdapter implements UsdTarget{
public UsdAdapter(float rmb) {
super(rmb);
}
@Override
public float getUsd() {
return getRMB()/5.6f ;
}
}
3>:人民币的adapter:
/**
* Email: 2185134304@qq.com
* Created by Novate 2018/5/27 10:10
* Version 1.0
* Params:
* Description:
*/
public class RMBAdapter {
private float mRMB ;
public RMBAdapter(float rmb){
this.mRMB = rmb ;
}
/**
* 获取人民币
*/
public float getRMB(){
return mRMB ;
}
/**
* 获取美元
*/
public float getUsd(){
return mRMB/5.6f ;
}
}
4>:直接测试:
/**
* Email: 2185134304@qq.com
* Created by Novate 2018/5/27 10:12
* Version 1.0
* Params:
* Description: 第二个版本
*/
public class Client {
public static void main(String[] args){
// 第一个版本:只是显示人民币
RMBAdapter rmbAdapter = new RMBAdapter(560) ;
float rmb = rmbAdapter.getRMB() ;
System.out.println("人民币:"+rmb);
// 第二个版本:要去兼容美元 , 采用适配器模式
// 角色:需要适配的接口 UsdTarget(把人民币转为美元)、适配器对象(Adapter)
UsdAdapter adapter = new UsdAdapter(560) ;
rmb = adapter.getRMB() ;
System.out.println("人民币:"+rmb);
float usd = adapter.getUsd() ;
System.out.println("美元:"+usd);
}
}
2.2: 对象适配
1>:人民币的adapter如下:
/**
* Email: 2185134304@qq.com
* Created by Novate 2018/5/27 10:10
* Version 1.0
* Params:
* Description:
*/
public class RMBAdapter {
private float mRMB ;
public RMBAdapter(float rmb){
this.mRMB = rmb ;
}
/**
* 获取人民币
*/
public float getRMB(){
return mRMB ;
}
/**
* 获取美元
*/
public float getUsd(){
return mRMB/5.6f ;
}
}
2>:目标接口如下:
/**
* Email: 2185134304@qq.com
* Created by Novate 2018/5/27 10:19
* Version 1.0
* Params:
* Description: 需要适配的目标接口usd(美元)
*/
public interface UsdTarget {
float getUsd() ;
}
3>:让adapter实现目标接口
/**
* Email: 2185134304@qq.com
* Created by Novate 2018/5/27 10:20
* Version 1.0
* Params:
* Description: 对象适配:
* 适配器对象 - 把人民币转为美元
*/
public class Adapter implements UsdTarget {
private RMBAdapter rmbAdapter;
public Adapter(RMBAdapter rmbAdapter) {
this.rmbAdapter = rmbAdapter ;
}
@Override
public float getUsd() {
return rmbAdapter.getRMB()/5.6f ;
}
}
4>:直接测试:
/**
* Email: 2185134304@qq.com
* Created by Novate 2018/5/27 10:12
* Version 1.0
* Params:
* Description: 对象适配
*/
public class Client {
public static void main(String[] args){
// 第一个版本:只是显示人民币
RMBAdapter rmbAdapter = new RMBAdapter(560) ;
Adapter adapter = new Adapter(rmbAdapter) ;
float usd = adapter.getUsd() ;
System.out.println("美元:"+usd);
}
}
3. 使用场景
自定义View、BannerView轮播图、IndicatorView ViewPager指示器、FlowLayout热门标签布局、ListView、RecyclerView的adapter设计模式;
4. 自己写一个ListView
1>:先写一个目标接口 TargetAdapter
/**
* Email: 2185134304@qq.com
* Created by Novate 2018/5/27 11:27
* Version 1.0
* Params:
* Description: 目标接口 - 相当于 UsdTarget
*/
public interface TargetAdapter {
/**
* 获取多少条
*/
int getCount() ;
/**
* 获取View
*/
View getView(int position , ViewGroup parent) ;
}
2>:然后定义一个 具体的适配器adapter,属于对象适配,把items集合数据适配成 View:
/**
* Email: 2185134304@qq.com
* Created by Novate 2018/5/27 11:30
* Version 1.0
* Params: 属于对象适配
* Description: 具体的适配器 - 相当于simple3 的Adapter
* 把数据集合适配成View
*/
public class ListAdapter implements TargetAdapter{
private List<String> mItems ;
private Context context ;
public ListAdapter(List<String> items , Context context){
this.mItems = items ;
this.context = context ;
}
@Override
public int getCount() {
return mItems.size();
}
@Override
public View getView(int position, ViewGroup parent) {
TextView itemView = (TextView) LayoutInflater.from(context).inflate(R.layout.item_main , parent , false);
itemView.setText(mItems.get(position));
return itemView;
}
}
3>:自定义一个 NovateListView:
/**
* Email: 2185134304@qq.com
* Created by Novate 2018/5/27 11:12
* Version 1.0
* Params:
* Description: 自定义ListView - 不考虑复用
*/
public class NovateListView extends ScrollView {
// 把所有创建的View都添加到 mContainer里边
private LinearLayout mContainer ;
// 目标接口
private TargetAdapter mAdapter ;
public NovateListView(Context context) {
this(context,null);
}
public NovateListView(Context context, AttributeSet attrs) {
this(context, attrs,0);
}
public NovateListView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
mContainer = new LinearLayout(context) ;
mContainer.setOrientation(LinearLayout.VERTICAL);
addView(mContainer,0);
}
@Override
public void addView(View child) {
mContainer.addView(child);
}
public void setAdapter(ListAdapter adapter) {
this.mAdapter = adapter ;
// 观察者,注册、反注册
int count = mAdapter.getCount() ;
for (int i = 0; i < count; i++) {
View childView = mAdapter.getView(i , mContainer) ;
addView(childView);
}
}
}
4>:最后在MainActivity中测试即可:
public class MainActivity extends AppCompatActivity {
private NovateListView novate_lv;
private List<String> items = new ArrayList<>() ;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
novate_lv = (NovateListView) findViewById(R.id.novate_lv);
// 假设后台返回的数据是一个集合,而我们要显示的是列表,是View,
for (int i = 0 ; i<10 ; i++){
items.add(i+"") ;
}
// ListView 需要的是 View ,后台给我们返回的是列表对象数组 ,我们采用适配器模式适配
// 一般写法 - 不断的for循环,不断添加View
/*for (String item : items) {
TextView itemView = (TextView) LayoutInflater.from(this).inflate(R.layout.item_main , null);
itemView.setText(item);
novate_lv.addView(itemView);
}*/
// 改进版 - adapter设计模式
novate_lv.setAdapter(new ListAdapter(items , this)) ;
}
}
5. adapter设计模式套路总结
1>:先写一个目标接口 TargetAdapter;
2>:然后定义一个 具体适配器对象的adapter,让其实现 目标接口 TargetAdapter;
3>:最后在MainActivity中采用适配器模式,也就是adapter设计模式,把items集合数据适配成 View即可;
代码已上传至github:
https://github.com/shuai999/Architect_day14.git