ListView中按钮监听器 设置 及 优化
2015-06-27 09:45
281 查看
在应用开发中经常会用到ListView,并且每一个Item里面都会有按钮之类的需要进行事件监听的控件。在给按钮添加OnClickListener的时候,一开始很下意识的会想在ListView的adapter的getView方法中找到每个Button并new一个OnClickListener分配给这个button。
但是当数据量很大的时候,new出来这么多个监听器势必会对内存造成一定的压力,而且每个Listener的功能完全一样,Listener里面所需知道的,不过是调用者所在的Item的index而已。怎么样才能更好地利用内存呢?既然每个Listener的功能一样,那么完全可以用单例模式构造一个Listener。
如下:
[java] view
plaincopyprint?
![](https://code.csdn.net/assets/CODE_ico.png)
![](https://code.csdn.net/assets/ico_fork.svg)
class MyOnClickListener implements OnClickListener {
private static MyOnClickListener instance = null;
private MyOnClickListener() {
}
public static MyOnClickListener getInstance() {
if (instance == null)
instance = new MyOnClickListener() ;
return instance;
}
@Override
public void onClick(View view) {
//TODO: do something here
}
}
而在getView方法中,获取到button实例之后,只需要通过button.setOnClickListener(MyOnClickListener.getInstance());对按钮设置监听器了。这样的话每一个按钮便必然用的是同一个Listener对象。
但是我们的需求并不止于此,很多时候,我们还需要知道具体是哪个position的button被点击了,我们需要根据position在Listener里面做出不一样的动作。
想要在Listener内部了解外部控件的属性,我们首先想到的是传参,但是由于我们的Listener使用的是单例模式,每个按钮往Listner里面传的参数必然会
覆盖前一个按钮传的参数。于是我们的解决方案只剩下一种,那就是通过onClick函数的参数(View view)来获取该信息。然而,此处的view应该是一个Button,
而Button是不具备position信息的。又于是,自然而然的,解决方案出来了:重载Button类。
[java] view
plaincopyprint?
![](https://code.csdn.net/assets/CODE_ico.png)
![](https://code.csdn.net/assets/ico_fork.svg)
class MyButton extends Button {
private int index = -1;
public int getIndex() {
return index;
}
public void setIndex(int index) {
this.index = index;
}
public MyButton(Context context) {
super(context);
// TODO: do something here if you want
}
public MyButton(Context context, AttributeSet attrs) {
super(context, attrs);
// TODO: do something here if you want
}
public MyButton(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
// TODO: do something here if you want
}
}
接下来我们需要做的,就是在xml文件中,将item里面的Button的类型改成我们自定义的MyButton。
即将<Button> </Button>改成<your.package.name.MyButton> </your.package.name.MyButton>,而在adapter的getView函数里面则把findViewById()获得的返回值强制转换成为MyButton,并调用其setIndex函数设置Index值。同时MyOnClickListener中重载的的onClick函数也一样将view对象转换成MyButton类型,并通过调用getIndex函数获取position信息,以做相应操作。
Adapter中:
[java] view
plaincopyprint?
![](https://code.csdn.net/assets/CODE_ico.png)
![](https://code.csdn.net/assets/ico_fork.svg)
// ....
MyButton button = null;
// ....
@Override
public View getView(int position, View convertView, ViewGroup parentView) {
View view = convertView;
if (convertView == null) {
view = LayoutInflater.from(activity).inflate(R.layout.company_detail_campus_talk_item, null);
}
// ....
button = (MyButton) view.findViewById(R.id.YOUR_BUTTON_ID);
button.setIndex(position);
button.setOnClickListener(MyOnClickListener.getInstance());
}
MyOnClickListener中:
[java] view
plaincopyprint?
![](https://code.csdn.net/assets/CODE_ico.png)
![](https://code.csdn.net/assets/ico_fork.svg)
// ....
@Override
public void onClick(View view) {
int index = ((MyButton)view).getIndex();
// ....
}
这样,我们便实现了使用同一个Listener对ListView中不同Item的按钮进行事件监听处理的业务逻辑。
如果需要在Adapter和Listener之间共享数据的话,可以通过增加Listener的getInstance函数的参数以及Listener类的成员变量实现。
但是当数据量很大的时候,new出来这么多个监听器势必会对内存造成一定的压力,而且每个Listener的功能完全一样,Listener里面所需知道的,不过是调用者所在的Item的index而已。怎么样才能更好地利用内存呢?既然每个Listener的功能一样,那么完全可以用单例模式构造一个Listener。
如下:
[java] view
plaincopyprint?
![](https://code.csdn.net/assets/CODE_ico.png)
class MyOnClickListener implements OnClickListener {
private static MyOnClickListener instance = null;
private MyOnClickListener() {
}
public static MyOnClickListener getInstance() {
if (instance == null)
instance = new MyOnClickListener() ;
return instance;
}
@Override
public void onClick(View view) {
//TODO: do something here
}
}
而在getView方法中,获取到button实例之后,只需要通过button.setOnClickListener(MyOnClickListener.getInstance());对按钮设置监听器了。这样的话每一个按钮便必然用的是同一个Listener对象。
但是我们的需求并不止于此,很多时候,我们还需要知道具体是哪个position的button被点击了,我们需要根据position在Listener里面做出不一样的动作。
想要在Listener内部了解外部控件的属性,我们首先想到的是传参,但是由于我们的Listener使用的是单例模式,每个按钮往Listner里面传的参数必然会
覆盖前一个按钮传的参数。于是我们的解决方案只剩下一种,那就是通过onClick函数的参数(View view)来获取该信息。然而,此处的view应该是一个Button,
而Button是不具备position信息的。又于是,自然而然的,解决方案出来了:重载Button类。
[java] view
plaincopyprint?
![](https://code.csdn.net/assets/CODE_ico.png)
class MyButton extends Button {
private int index = -1;
public int getIndex() {
return index;
}
public void setIndex(int index) {
this.index = index;
}
public MyButton(Context context) {
super(context);
// TODO: do something here if you want
}
public MyButton(Context context, AttributeSet attrs) {
super(context, attrs);
// TODO: do something here if you want
}
public MyButton(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
// TODO: do something here if you want
}
}
接下来我们需要做的,就是在xml文件中,将item里面的Button的类型改成我们自定义的MyButton。
即将<Button> </Button>改成<your.package.name.MyButton> </your.package.name.MyButton>,而在adapter的getView函数里面则把findViewById()获得的返回值强制转换成为MyButton,并调用其setIndex函数设置Index值。同时MyOnClickListener中重载的的onClick函数也一样将view对象转换成MyButton类型,并通过调用getIndex函数获取position信息,以做相应操作。
Adapter中:
[java] view
plaincopyprint?
![](https://code.csdn.net/assets/CODE_ico.png)
// ....
MyButton button = null;
// ....
@Override
public View getView(int position, View convertView, ViewGroup parentView) {
View view = convertView;
if (convertView == null) {
view = LayoutInflater.from(activity).inflate(R.layout.company_detail_campus_talk_item, null);
}
// ....
button = (MyButton) view.findViewById(R.id.YOUR_BUTTON_ID);
button.setIndex(position);
button.setOnClickListener(MyOnClickListener.getInstance());
}
MyOnClickListener中:
[java] view
plaincopyprint?
![](https://code.csdn.net/assets/CODE_ico.png)
// ....
@Override
public void onClick(View view) {
int index = ((MyButton)view).getIndex();
// ....
}
这样,我们便实现了使用同一个Listener对ListView中不同Item的按钮进行事件监听处理的业务逻辑。
如果需要在Adapter和Listener之间共享数据的话,可以通过增加Listener的getInstance函数的参数以及Listener类的成员变量实现。
相关文章推荐
- 社説 20150627 安保法案審議 戦略的な曖昧性は確保したい
- 计数排序
- 社説 20150627 自民若手勉強会 看過できない「報道規制」発言
- Windows 10 10154最新预览版界面调整及更新一览视频
- 判断是否移动客户端
- 日经社説 20150627 企業は株主総会を市場との対話の起点に
- win10语音包安装与激活图文教程
- Linux Shell 常用技巧
- 微信运营须知:2015微信防封秘笈
- 日经社説 20150627 納得しがたい新競技場の工費
- 二进制十六进制转换 (C++版)
- komodo edit
- 剑指offer 48-不能被继承的类
- 关于UIImageView不显示图片的问题小记
- Shell脚本检查memcache进程并自动重启
- 日经春秋 20150627
- Android---60---Notification 通知栏的简单使用
- CSS兼容的几个点
- android在代码中四种设置控件(以及TextView的文字颜色)背景颜色的方法
- cacti监控工具之安装篇