在做安卓應用開發的時候很經常會用到ListView,並且每一個Item裡面都會有按鈕之類的需要進行事件監聽的控制項。在給按鈕添加OnClickListener的時候,一開始很下意識的會想在ListView的adapter的getView方法中找到每個Button並new一個OnClickListener分配給這個button。但是當資料量很大的時候,new出來這麼多個監聽器勢必會對記憶體造成一定的壓力,而且每個Listener的功能完全一樣,Listener裡面所需知道的,不過是調用者所在的Item的index而已。怎麼樣才能更好地利用記憶體呢?
既然每個Listener的功能一樣,那麼完全可以用單例模式構造一個Listener。如下:
1 class MyOnClickListener implements OnClickListener {
2
3 private static MyOnClickListener instance = null;
4
5 private MyOnClickListener() {
6 }
7
8 public static MyOnClickListener getInstance() {
9 if (instance == null)
10 instance = new MyOnClickListener() ;
11 return instance;
12 }
13
14 @Override
15 public void onClick(View view) {
16 //TODO: do something here
17 }
18 }
而在getView方法中,擷取到button執行個體之後,只需要通過button.setOnClickListener(MyOnClickListener.getInstance());對按鈕設定監聽器了。這樣的話每一個按鈕便必然用的是同一個Listener對象。
但是我們的需求並不止於此,很多時候,我們還需要知道具體是哪個position的button被點擊了,我們需要根據position在Listener裡面做出不一樣的動作。
想要在Listener內部瞭解外部控制項的屬性,我們首先想到的是傳參,但是由於我們的Listener使用的是單例模式,每個按鈕往Listner裡面傳的參數必然會覆蓋前一個按鈕傳的參數。於是我們的解決方案只剩下一種,那就是通過onClick函數的參數(View view)來擷取該資訊。然而,此處的view應該是一個Button,而Button是不具備position資訊的。又於是,自然而然的,解決方案出來了:重載Button類。
1 class MyButton extends Button {
2
3 private int index = -1;
4
5 public int getIndex() {
6 return index;
7 }
8
9 public void setIndex(int index) {
10 this.index = index;
11 }
12
13 public MyButton(Context context) {
14 super(context);
15 // TODO: do something here if you want
16 }
17
18 public MyButton(Context context, AttributeSet attrs) {
19 super(context, attrs);
20 // TODO: do something here if you want
21 }
22
23 public MyButton(Context context, AttributeSet attrs, int defStyle) {
24 super(context, attrs, defStyle);
25 // TODO: do something here if you want
26 }
27 }
接下來我們需要做的,就是在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中:
1 // ....
2 MyButton button = null;
3 // ....
4 @Override
5 public View getView(int position, View convertView, ViewGroup parentView) {
6 View view = convertView;
7 if (convertView == null) {
8 view = LayoutInflater.from(activity).inflate(R.layout.company_detail_campus_talk_item, null);
9 }
10
11 // ....
12
13 button = (MyButton) view.findViewById(R.id.YOUR_BUTTON_ID);
14 button.setIndex(position);
15 button.setOnClickListener(MyOnClickListener.getInstance());
16 }
MyOnClickListener中:
1 // ....
2
3 @Override
4 public void onClick(View view) {
5
6 int index = ((MyButton)view).getIndex();
7
8 // ....
9 }
這樣,我們便實現了使用同一個Listener對ListView中不同Item的按鈕進行事件監聽處理的商務邏輯。如果需要在Adapter和Listener之間共用資料的話,可以通過增加Listener的getInstance函數的參數以及Listener類的成員變數實現。