相似於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進行更新便可。