有關ViewPager的使用及解決Android下ViewPager和PagerAdapter中調用notifyDataSetChanged失效的問題

ViewPager是android-support-v4.jar包中的一個系統控件,繼承自ViewGroup,專門用以實現左右滑動切換View的效果,使用時須要首先在Project->properties->Java Build Path->Libraries->Add External Jars中加入sdk目錄下的extras/android/support/v4/android-support-v4.jar(若是找不到,則須要用sdk manager下載android support package)。加入這個jar包以後就可使用ViewPager類了。
ViewPager的使用相似於ListView,須要有對應的Adapter進行數據綁定,實現圖片切換僅須要繼承PaperAdapter就能夠了。繼承後須要重寫以下四個方法。
    instantiateItem(ViewGroup, int)
    destroyItem(ViewGroup, int, Object)
    getCount()
    isViewFromObject(View, Object)

相似於BaseAdapter,其中instantiateItem方法用來獲得每一個View,destroyItem用以控制當某個View不須要的時候的回收處理。isViewFromObject用來實現判斷View和Object是否爲同一個View。html

先看一下效果圖:java

第一步:首先是在佈局文件裏添加viewPager佈局。代碼以下:android

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity" >

    <android.support.v4.view.ViewPager
        android:id="@+id/viewpager"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        android:layout_gravity="center" >
    </android.support.v4.view.ViewPager>

    <Button
        android:id="@+id/deleteBtn"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentBottom="true"
        android:layout_centerHorizontal="true"
        android:layout_marginBottom="20dp"
        android:text="刪除" />

</RelativeLayout>
第二步:建立item佈局用於填充在ViewPager裏,能夠自定義也能夠加載寫好的xml佈局文件。代碼以下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:gravity="center"
    android:orientation="vertical" >

    <LinearLayout
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginBottom="10dp"
        android:layout_marginTop="10dp"
        android:background="@drawable/item_bg"
        android:orientation="vertical"
        android:padding="10dp" >

        <TextView
            android:id="@+id/view_title"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="center_horizontal"
            android:layout_marginTop="15dp"
            android:singleLine="true"
            android:textSize="20sp" />

        <ImageView
            android:id="@+id/view_image"
            android:layout_width="240dp"
            android:layout_height="220dp"
            android:layout_gravity="center_horizontal"
            android:layout_marginLeft="8dp"
            android:layout_marginTop="2dp" />

        <TextView
            android:id="@+id/view_content"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="center_horizontal"
            android:layout_marginBottom="20dp"
            android:layout_marginTop="2dp"
            android:ellipsize="end"
            android:maxLines="2"
            android:text="很不錯哦!嘻嘻,嘿嘿,O(∩_∩)O哈哈哈~……"
            android:textSize="15sp" />
    </LinearLayout>

</LinearLayout>
第三步:而後就是Activity了,主要寫了左右滑動切換頁面。代碼以下:

package net.loonggg.viewpager;

import java.util.ArrayList;
import java.util.List;

import android.app.Activity;
import android.os.Bundle;
import android.support.v4.view.ViewPager;
import android.support.v4.view.ViewPager.OnPageChangeListener;
import android.view.LayoutInflater;
import android.view.View;
import android.view.Window;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.TextView;
import android.widget.Toast;

public class MainActivity extends Activity {
	private ViewPager viewPager;
	private Button deleteBtn;
	private List<View> listViews = null;
	private int[] imgs = { R.drawable.img0, R.drawable.img1, R.drawable.img2,
			R.drawable.img3, R.drawable.img4, R.drawable.img5, };
	private int index = 0;
	private ViewPagerAdapter adapter;

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		requestWindowFeature(Window.FEATURE_NO_TITLE);
		setContentView(R.layout.activity_main);
		viewPager = (ViewPager) findViewById(R.id.viewpager);
		deleteBtn = (Button) findViewById(R.id.deleteBtn);

		listViews = new ArrayList<View>();
		for (int i = 0; i < imgs.length; i++) {
			View view = LayoutInflater.from(getApplicationContext()).inflate(
					R.layout.viewpager_item, null);
			TextView title = (TextView) view.findViewById(R.id.view_title);
			title.setText("頭像");
			ImageView iv = (ImageView) view.findViewById(R.id.view_image);
			iv.setBackgroundResource(imgs[i]);
			listViews.add(view);
		}

		adapter = new ViewPagerAdapter(listViews);
		viewPager.setAdapter(adapter);
		viewPager.setOnPageChangeListener(new PageChangeListener());

		deleteBtn.setOnClickListener(new View.OnClickListener() {

			@Override
			public void onClick(View v) {
				if (listViews.size() > 0) {
					listViews.remove(index);
					adapter.notifyDataSetChanged();
				}
			}
		});
	}

	private class PageChangeListener implements OnPageChangeListener {

		@Override
		public void onPageScrollStateChanged(int arg0) {

		}

		@Override
		public void onPageScrolled(int arg0, float arg1, int arg2) {

		}

		@Override
		public void onPageSelected(int arg0) {
			Toast.makeText(getApplicationContext(), arg0 + "", 0).show();
			index = arg0;
		}

	}
}
第四步:是有關ViewPager的適配器的重寫。代碼以下:

package net.loonggg.viewpager;

import java.util.List;

import android.support.v4.view.PagerAdapter;
import android.view.View;
import android.view.ViewGroup;

public class ViewPagerAdapter extends PagerAdapter {
	private List<View> list;

	public ViewPagerAdapter(List<View> list) {
		this.list = list;
	}

	@Override
	public int getCount() {

		if (list != null && list.size() > 0) {
			return list.size();
		} else {
			return 0;
		}
	}

	@Override
	public boolean isViewFromObject(View arg0, Object arg1) {
		return arg0 == arg1;
	}

	@Override
	public void destroyItem(ViewGroup container, int position, Object object) {
		container.removeView((View) object);
	}

	@Override
	public Object instantiateItem(ViewGroup container, int position) {
		container.addView(list.get(position));
		return list.get(position);
	}

	@Override
	public int getItemPosition(Object object) {
		return POSITION_NONE;
	}

}
到這裏有關ViewPager的使用就講完了, 其實更重要的是想講:如何解決Android下ViewPager和PagerAdapter中調用notifyDataSetChanged失效的問題 。

具體講解以下:app

Google在Android 3.0SDK中推出的ViewPager控件很大程度上知足了開發者開發頁面左右移動切換的功能,使用很是方便。可是使用中發現,在刪除或者修改數據的時候,PagerAdapter沒法像BaseAdapter那樣僅經過notifyDataSetChanged方法通知刷新View。
最基本的方法:
針對於child view比較簡單的狀況(例如僅有TextView、ImageView等,沒有ListView等展現數據的狀況),能夠在本身的Adapter中加入代碼:ide

    @Override  
    public int getItemPosition(Object object) {  
        return POSITION_NONE;  
    }  
這樣既可達到通常狀況下要求的效果。 存在的問題: 這不是PagerAdapter中的Bug,一般狀況下,調用notifyDataSetChanged方法會讓ViewPager經過Adapter的getItemPosition方法查詢一遍全部child view,這種狀況下,全部child view位置均爲POSITION_NONE,表示全部的child view都不存在,ViewPager會調用destroyItem方法銷燬,而且從新生成,加大系統開銷,並在一些複雜狀況下致使邏輯問題。特別是對於只是但願更新child view內容的時候,形成了徹底沒必要要的開銷。 更有效地方法: 更爲靠譜的方法是因地制宜,根據本身的需求來實現notifyDataSetChanged的功能,好比,在僅須要對某個View內容進行更新時,在instantiateItem()時,用View.setTag方法加入標誌,在須要更新信息時,經過findViewWithTag的方法找到對應的View進行更新便可。
相關文章
相關標籤/搜索