Android ListView中按鈕監聽器設定及優化

gao_chun的專欄發表於2015-05-02

在Android應用開發中經常會用到ListView,並且每一個Item裡面都會有按鈕之類的需要進行事件監聽的控制元件。在給按鈕新增OnClickListener的時候,一開始很下意識的會想在ListView的adapter的getView方法中找到每個Button並new一個OnClickListener分配給這個button。

但是當資料量很大的時候,new出來這麼多個監聽器勢必會對記憶體造成一定的壓力,而且每個Listener的功能完全一樣,Listener裡面所需知道的,不過是呼叫者所在的Item的index而已。怎麼樣才能更好地利用記憶體呢?既然每個Listener的功能一樣,那麼完全可以用單例模式構造一個Listener。

如下:

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類。

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中:

// ....  
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中:

// ....  
@Override  
public void onClick(View view) {  

    int index = ((MyButton)view).getIndex();  

   // ....  
}

這樣,我們便實現了使用同一個Listener對ListView中不同Item的按鈕進行事件監聽處理的業務邏輯。

如果需要在Adapter和Listener之間共享資料的話,可以通過增加Listener的getInstance函式的引數以及Listener類的成員變數實現。

相關文章