日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 运维知识 > Android >内容正文

Android

【读书笔记《Android游戏编程之从零开始》】6.Android 游戏开发常用的系统控件(TabHost、ListView)...

發(fā)布時(shí)間:2024/1/17 Android 31 豆豆
生活随笔 收集整理的這篇文章主要介紹了 【读书笔记《Android游戏编程之从零开始》】6.Android 游戏开发常用的系统控件(TabHost、ListView)... 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

3.9 TabSpec與TabHost

TabHost類(lèi)官方文檔地址:http://developer.android.com/reference/android/widget/TabHost.html

Android?實(shí)現(xiàn)tab視圖有2種方法,一種是在布局頁(yè)面中定義<tabhost>標(biāo)簽,另一種就是繼承tabactivity.但是我比較喜歡第二種方式,應(yīng)為如果頁(yè)面比較復(fù)雜的話你的XML文件會(huì)寫(xiě)得比較龐大,用第二種方式XML頁(yè)面相對(duì)要簡(jiǎn)潔得多。

?

<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="match_parent"android:background="@drawable/mm1"android:orientation="vertical" ><Buttonandroid:id="@+id/btn"android:layout_width="match_parent"android:layout_height="wrap_content"android:text="第一個(gè)Tab" /><EditTextandroid:id="@+id/et"android:layout_width="match_parent"android:layout_height="wrap_content"android:hint="第二個(gè)Tab" /><LinearLayoutandroid:id="@+id/myLayout"android:layout_width="match_parent"android:layout_height="match_parent"android:background="@drawable/mm2"android:orientation="vertical" ><Buttonandroid:layout_width="match_parent"android:layout_height="wrap_content"android:text="第三個(gè)Tab" /><EditTextandroid:layout_width="match_parent"android:layout_height="wrap_content"android:hint="第三個(gè)Tab" /></LinearLayout></LinearLayout> activity_main.xml import android.app.TabActivity; import android.os.Bundle; import android.view.LayoutInflater; import android.widget.TabHost; import android.widget.TabHost.OnTabChangeListener; import android.widget.TabHost.TabSpec; import android.widget.Toast;public class MainActivity extends TabActivity implements OnTabChangeListener {private TabSpec ts1, ts2, ts3;// 聲明3個(gè)分頁(yè)private TabHost tabHost;// 分頁(yè)菜單(tab容器) @Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);tabHost = this.getTabHost();// 實(shí)例(分頁(yè))菜單// 利用LayoutInflater將布局與分頁(yè)菜單一起顯示LayoutInflater.from(this).inflate(R.layout.activity_main,tabHost.getTabContentView());ts1 = tabHost.newTabSpec("tabOne");// 實(shí)例化一個(gè)分頁(yè)ts1.setIndicator("分頁(yè)1");// 設(shè)置此分頁(yè)顯示的標(biāo)題ts1.setContent(R.id.btn);// 設(shè)置此分頁(yè)的資源Idts2 = tabHost.newTabSpec("tabTwo");// 設(shè)置此分頁(yè)顯示的標(biāo)題和圖標(biāo)ts2.setIndicator("分頁(yè)2",getResources().getDrawable(R.drawable.ic_launcher));ts2.setContent(R.id.et);ts3 = tabHost.newTabSpec("tabThree");ts3.setIndicator("分頁(yè)3");ts3.setContent(R.id.myLayout);// 設(shè)置此分頁(yè)的布局IDtabHost.addTab(ts1);// 菜單中添加ts1分頁(yè) tabHost.addTab(ts2);tabHost.addTab(ts3);tabHost.setOnTabChangedListener(this);}@Overridepublic void onTabChanged(String tabId) {//這里的tabId對(duì)應(yīng)的是實(shí)例中每個(gè)分頁(yè)傳入的分頁(yè)ID,而不是TabSpec.setIndicator()設(shè)置的標(biāo)題if (tabId.equals("tabOne")) {Toast.makeText(this, "分頁(yè)1", Toast.LENGTH_SHORT).show();}if (tabId.equals("tabTwo")) {Toast.makeText(this, "分頁(yè)2", Toast.LENGTH_SHORT).show();}if (tabId.equals("tabThree")) {Toast.makeText(this, "分頁(yè)3", Toast.LENGTH_SHORT).show();}}} MainActivity.class

上面這個(gè)Activity繼承了TabActivity

TabActivity的現(xiàn)狀

官方文檔在介紹TabActivity有下面這么一句話

大概的意思是說(shuō):這個(gè)類(lèi)已經(jīng)在Android4.0的系統(tǒng)中被棄用了,新的應(yīng)用程序應(yīng)該使用Fragment來(lái)代替該類(lèi)的開(kāi)發(fā)

TabActivity是否還有存在的必要性

其實(shí)谷歌有此舉動(dòng),我們也應(yīng)該早就想到了,為什么會(huì)這么說(shuō)呢?那就要從TabActivity的原理開(kāi)始說(shuō)起了。

做個(gè)假定先: 比如我們最外面的Activity是MainActivity, 第一個(gè)tab是FirstActivty, 第二個(gè)tab是SecondActivity。
相信大家都用過(guò)TabActivity, 它是一個(gè)特殊的Activity,它特殊的地方在哪里?有以下幾點(diǎn)為證:?
<1> 它看起來(lái)違反了Activity的單一窗口的原則。因?yàn)樗梢酝瑫r(shí)加載幾個(gè)activity, 當(dāng)用戶點(diǎn)擊它上面的tab時(shí),就會(huì)跳到相應(yīng)的Activity上面去。
<2> 用戶首先進(jìn)去FirstActivity,然后進(jìn)去SecondActivity,再點(diǎn)擊返回鍵的時(shí)候。它返回的界面不是FirstActivity,而是退出我們的應(yīng)用程序。
<3> 當(dāng)用戶在FirstActivity按返回鍵的時(shí)候,如果MainActivity和FirstActivity通過(guò)重寫(xiě)onKeyDown()方法,那么收到事件回調(diào)的只有FirstActivity。

谷歌當(dāng)時(shí)的困擾

<1> 首先我們要明白一點(diǎn),android系統(tǒng)是單窗口系統(tǒng),不像windows是多窗口的(比如在windows系統(tǒng)上,我們可以一邊聊QQ,一邊斗地主等等)。也就是說(shuō),在一個(gè)時(shí)刻,android里面只有一個(gè)activity可以顯示給用戶。這樣就大大降低了操作系統(tǒng)設(shè)計(jì)的復(fù)雜性(包括事件派發(fā)等等)。
<2> 但是像TabActivity那種效果又非常必要,用戶體驗(yàn)也比較好。所以我覺(jué)得當(dāng)時(shí)google開(kāi)發(fā)人員肯定很糾結(jié),于是,一個(gè)畸形的想法產(chǎn)生了,就是在單窗口系統(tǒng)下加載多個(gè)activity,它就是TabActivity。

TabActivity實(shí)現(xiàn)加載多個(gè)Activity原理

我們都知道,想啟動(dòng)一個(gè)Activity,一般是調(diào)用startActivty(Intent i)方法,然后這個(gè)方法會(huì)輾轉(zhuǎn)調(diào)用到ams(ActivityManagerService)來(lái)啟動(dòng)目標(biāo)activity,所以,TabActivity實(shí)現(xiàn)的要點(diǎn)有兩個(gè):
<1> 找到一個(gè)入口,這個(gè)入口可以訪問(wèn)到ActivityThread類(lèi)(這個(gè)類(lèi)是隱藏的,應(yīng)用程序是訪問(wèn)不到的),然后調(diào)用ActivityThread里面的啟動(dòng)activity方法
<2> 繞開(kāi)ams,就是我們TabActivity加載的FirstActivity和SecondActivity是不能讓ams知道的。

所以,一個(gè)新的類(lèi)誕生了 ---- LocalActivityManager , 它的作用如下:
<1> 這個(gè)類(lèi)和ActivityThread處于一個(gè)包內(nèi),所以它有訪問(wèn)ActivityThread的權(quán)限。
<2> 這個(gè)類(lèi)提供了類(lèi)似Ams管理Activity的方法,比如調(diào)用activity的onCreate方法,onResume()等等,維護(hù)了activity生命周期。

也正如其名字一樣,它是本地的activity管理。就是說(shuō)它運(yùn)行的進(jìn)程和它管理的Activity是在一個(gè)進(jìn)程里面。所以,當(dāng)TabActivity要啟動(dòng)一個(gè)activity的時(shí)候,會(huì)調(diào)用到LocalActivityManager的創(chuàng)建activity方法,然后調(diào)用ActivityThread.startActivityNow(),這個(gè)方法繞過(guò)了ams,就是說(shuō)ams此時(shí)根本不知道LocalActivityManager已經(jīng)在暗渡陳倉(cāng)的啟動(dòng)了一個(gè)activity(所以ams的task列表里面沒(méi)有新啟動(dòng)activity的記錄,所以用戶按back鍵就直接退出我們的應(yīng)用)。然后和正常啟動(dòng)activity一樣,初始化activity,在初始化activity的時(shí)候,有個(gè)方法非常重要:activity.attch()

final void attach(...){ .... mWindow.setCallback(this); ..... }

?mWindow.setCallback(this)這個(gè)方法非常重要,它設(shè)置了window的回調(diào)接口,這是我們activity能夠接受到key事件的關(guān)鍵所在!因?yàn)樵贒ecorView在接受到事件的時(shí)候,會(huì)回調(diào)這個(gè)接口,如:

final Callback cb = getCallback(); final boolean handled = cb != null && mFeatureId < 0 ? cb.dispatchKeyEvent(event) : super.dispatchKeyEvent(event);

當(dāng)我們啟動(dòng)FirstActivity的時(shí)候,我們?cè)O(shè)置FirstActivity為PhoneWindow的回調(diào)實(shí)現(xiàn),所以,按back鍵的時(shí)候,調(diào)用的是FirstActivity的onKeyDown方法。

TabActivity小結(jié)

從以上的種種分析來(lái)看,TabActivity只是一個(gè)怪胎而已。所以,在后面的發(fā)展中肯定會(huì)被代替,只是沒(méi)想到會(huì)被替代的這么快。不經(jīng)讓我有了一種英雄暮路,美人辭暮的感覺(jué),至少TabActivity曾經(jīng)在Android2.2/2.3版本那么顯赫一時(shí),不過(guò)終究還是逃不過(guò)被谷歌遺棄的命運(yùn)。

?

TabActivity實(shí)現(xiàn)方法

說(shuō)了這么多,那就讓我們來(lái)看看它當(dāng)年到底是怎樣的叱咤風(fēng)云,我們將使用兩種不同的方式來(lái)實(shí)現(xiàn),但是最終的效果都是一樣的,

如下圖所示:

?

具體的編碼實(shí)現(xiàn)

(1)第一種實(shí)現(xiàn)方式:自定義TabWidget

1、首先創(chuàng)建一個(gè)TabWidget的布局文件,main_tab_layout1.xml:

<?xml version="1.0" encoding="utf-8"?> <TabHost xmlns:android="http://schemas.android.com/apk/res/android"android:id="@android:id/tabhost" android:layout_width="fill_parent"android:layout_height="fill_parent"><LinearLayout android:orientation="vertical"android:layout_width="fill_parent" android:layout_height="fill_parent"><FrameLayout android:id="@android:id/tabcontent"android:layout_width="fill_parent" android:layout_height="0.0dip"android:layout_weight="1.0" /><TabWidget android:id="@android:id/tabs" android:layout_width="fill_parent" android:layout_height="wrap_content"android:padding="2dip"android:background="@drawable/tab_widget_background"android:layout_weight="0.0"/></LinearLayout></TabHost> main_tab_layout1.xml 注意:? <1>?不管你是使用TabActivity 還是自定義TabHost,都要求以TabHost作為XML布局文件的根; <2> 將FrameLayout的屬性值layout_weight設(shè)置為了1.0,這樣就可以把TabWidget的組件從頂部擠了下來(lái)變成了底部菜單欄。 <3> <TabWidger> 和<FrameLayout>的Id 必須使用系統(tǒng)id,分別為android:id/tabs 和 android:id/tabcontent 。因?yàn)橄到y(tǒng)會(huì)使用者兩個(gè)id來(lái)初始化TabHost的兩個(gè)實(shí)例變量(mTabWidget 和 mTabContent)。 2、然后在定義一個(gè)tab_item_view.xml布局文件,這個(gè)布局文件在后面初始化Tab按鈕的時(shí)候會(huì)用到? <?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" ><ImageViewandroid:id="@+id/imageview"android:layout_width="wrap_content"android:layout_height="wrap_content"android:focusable="false"android:padding="3dp" ></ImageView><TextViewandroid:id="@+id/textview"style="@style/tab_item_text_style"android:layout_width="wrap_content"android:layout_height="wrap_content" ></TextView></LinearLayout> tab_item_view.xml

3、這里我為了方便Tab按鈕字體和背景格式的統(tǒng)一,在styles.xml數(shù)據(jù)文件中還添加了以下內(nèi)容:

<style name="tab_item_text_style"><item name="android:textSize">10.0dip</item><item name="android:textColor">#ffffff</item><item name="android:ellipsize">marquee</item><item name="android:singleLine">true</item></style><style name="tab_item_background"><item name="android:textAppearance">@style/tab_item_text_style</item><item name="android:gravity">center_horizontal</item><item name="android:background">@drawable/selector_tab_background2</item><item name="android:layout_width">fill_parent</item><item name="android:layout_height">wrap_content</item><item name="android:button">@null</item><item name="android:drawablePadding">3.0dip</item><item name="android:layout_weight">1.0</item></style>

4、定義一個(gè)自定義Tab按鈕資源文件,selector_tab_background.xml:

<?xml version="1.0" encoding="utf-8"?> <selector xmlns:android="http://schemas.android.com/apk/res/android"><item android:drawable="@drawable/tab_item_p" android:state_pressed="true"/><item android:drawable="@drawable/tab_item_d" android:state_selected="true"/></selector>

5、最后在定義幾個(gè)用來(lái)存放Tab選項(xiàng)卡內(nèi)容的activity布局文件,由于幾個(gè)布局文件的內(nèi)容都差不多,所以這里就列出一個(gè)給讀者參考,有需要的話可以直接下載源碼,layout_activity1.xml:

<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="fill_parent"android:layout_height="fill_parent"><ImageViewandroid:id="@+id/imageview"android:layout_width="fill_parent"android:layout_height="fill_parent"android:scaleType="fitCenter"android:src="@drawable/mm1" ></ImageView></LinearLayout>

6、布局完畢,接下來(lái)講解java代碼,定義一個(gè)常量工具類(lèi),Constant.java:

/*** 功能描述:常量工具類(lèi)*/ public class Constant {public static final class ConValue{/*** Tab選項(xiàng)卡的圖標(biāo)*/public static int mImageViewArray[] = {R.drawable.tab_icon1,R.drawable.tab_icon2,R.drawable.tab_icon3,R.drawable.tab_icon4,R.drawable.tab_icon5};/*** Tab選項(xiàng)卡的文字*/public static String mTextviewArray[] = {"主頁(yè)", "關(guān)于", "設(shè)置", "搜索", "更多"};/*** 每一個(gè)Tab界面*/public static Class mTabClassArray[]= {Activity1.class,Activity2.class,Activity3.class,Activity4.class,Activity5.class};} } Constant.java 7、定義自定義Tab選項(xiàng)卡Activity類(lèi),在這個(gè)類(lèi)中我們可以采用兩種方法編寫(xiě)標(biāo)簽頁(yè): <1> 第一種是繼承TabActivity ,然后使用getTabHost()獲取TabHost對(duì)象; <2> 第二種方法是使用自定的TabHost在布局文件上<TabHost>的自定義其ID,然后通過(guò)findViewById(),方法獲得TabHost對(duì)象。? 本文中采用是繼承TabActivity的方法,TabActivity1.java: package com.example.hiyou;import android.app.TabActivity; import android.content.Intent; import android.os.Bundle; import android.view.LayoutInflater; import android.view.View; import android.widget.ImageView; import android.widget.TabHost; import android.widget.TabHost.TabSpec; import android.widget.TextView;import com.example.hiyou.Constant.ConValue; /*** 功能描述:第一種實(shí)現(xiàn)方法,自定義TabHost*/ public class TabActivity1 extends TabActivity {//定義TabHost對(duì)象private TabHost tabHost; //定義一個(gè)布局private LayoutInflater layoutInflater;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.main_tab_layout1); initView();}/*** 初始化組件*/private void initView(){//實(shí)例化TabHost對(duì)象,得到TabHosttabHost = getTabHost();//實(shí)例化布局對(duì)象layoutInflater = LayoutInflater.from(this);//得到Activity的個(gè)數(shù)int count = ConValue.mTabClassArray.length; for(int i = 0; i < count; i++){ //為每一個(gè)Tab按鈕設(shè)置圖標(biāo)、文字和內(nèi)容TabSpec tabSpec = tabHost.newTabSpec(ConValue.mTextviewArray[i]).setIndicator(getTabItemView(i)).setContent(getTabItemIntent(i));//將Tab按鈕添加進(jìn)Tab選項(xiàng)卡中 tabHost.addTab(tabSpec);//設(shè)置Tab按鈕的背景 tabHost.getTabWidget().getChildAt(i).setBackgroundResource(R.drawable.selector_tab_background);}}/*** 給Tab按鈕設(shè)置圖標(biāo)和文字*/private View getTabItemView(int index){View view = layoutInflater.inflate(R.layout.tab_item_view, null);ImageView imageView = (ImageView) view.findViewById(R.id.imageview);if (imageView != null){imageView.setImageResource(ConValue.mImageViewArray[index]);} TextView textView = (TextView) view.findViewById(R.id.textview);textView.setText(ConValue.mTextviewArray[index]);return view;}/*** 給Tab選項(xiàng)卡設(shè)置內(nèi)容(每個(gè)內(nèi)容都是一個(gè)Activity)*/private Intent getTabItemIntent(int index){Intent intent = new Intent(this, ConValue.mTabClassArray[index]);return intent;} } TabActivity1.java 這段代碼比較復(fù)雜,我們需要詳細(xì)分析一下: <1> 首先需要做的是獲取TabHost對(duì)象,可以通過(guò)TabActivtiy里的getTabHsot()方法; <2> 接著向TabHost添加tabs.即調(diào)用tabHost.addTab(TabSpec) 方法。TabSpec主要包含了setIndicator 和 setContent 方法,通過(guò)這兩個(gè)方法來(lái)指定Tab 和 TanContent; <3> TabSpec 通過(guò)?.newTabSpec(String tag)來(lái)創(chuàng)建實(shí)例。實(shí)例化后對(duì)其屬性進(jìn)行設(shè)置。setIndicator()設(shè)置tab,它有3個(gè)重載的函數(shù):
  • public TabHost.TabSpec? setIndicatior(CharSwquence label,Drawable icon).指定tab的標(biāo)題和圖標(biāo)。
  • public TabHost.TabSpec (View view)通過(guò)View來(lái)自定義tab
  • public TabHost.TabSpec(CharSequence label) 指定tab的標(biāo)題,此時(shí)無(wú)圖標(biāo)。
<4>?setContent 指定tab的展示內(nèi)容,它也有3種重載:
  • public TabHost.TabSpec setContent(TabHost.TabContentFactory )
  • public TabHost.TabSpec setContent(int ViewId)
  • public TabHost.TabSpec setContent(Intent intent)  
后兩種方法比較后理解一個(gè)是通過(guò) ViewId指定顯示的內(nèi)容,如.setContent(R.id.Team_EditText),第三種則是直接通過(guò)Intent加載一個(gè)新的Activity頁(yè)。如.setContent(new Intent(this, MeetingActivity.class))); 8、最后再定義Tab選項(xiàng)卡內(nèi)容的Activity,顯示對(duì)應(yīng)的布局頁(yè)面就行了,這里只列出一個(gè),Activity1.java: package com.example.hiyou;import android.app.Activity; import android.os.Bundle;public class Activity1 extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.layout_activity1);} } (二)第二中實(shí)現(xiàn)方式:隱藏TabWidget,通過(guò)RadioGroup和RadioButton實(shí)現(xiàn)底部菜單欄

這種方式更漂亮,也更靈活,大部分的應(yīng)用程序基本都是使用這種方式,通過(guò)setCurrentTabByTag()方法來(lái)切換不同的選項(xiàng)卡。?

1、首先創(chuàng)建一個(gè)布局界面,main_tab_layout2.xml: <?xml version="1.0" encoding="utf-8"?> <TabHost xmlns:android="http://schemas.android.com/apk/res/android"android:id="@android:id/tabhost"android:layout_width="fill_parent"android:layout_height="fill_parent" ><LinearLayoutandroid:layout_width="fill_parent"android:layout_height="fill_parent"android:orientation="vertical" ><FrameLayoutandroid:id="@android:id/tabcontent"android:layout_width="fill_parent"android:layout_height="0.0dip"android:layout_weight="1.0" /><TabWidgetandroid:id="@android:id/tabs"android:layout_width="fill_parent"android:layout_height="wrap_content"android:layout_weight="0.0"android:visibility="gone" /><RadioGroupandroid:id="@+id/main_radiogroup"android:layout_width="fill_parent"android:layout_height="wrap_content"android:layout_gravity="bottom"android:background="@drawable/tab_widget_background"android:gravity="center_vertical"android:orientation="horizontal"android:padding="2dip" ><RadioButtonandroid:id="@+id/RadioButton0"style="@style/tab_item_background"android:drawableTop="@drawable/tab_icon1"android:text="主頁(yè)" android:textColor="#ffffff"/><RadioButtonandroid:id="@+id/RadioButton1"style="@style/tab_item_background"android:drawableTop="@drawable/tab_icon2"android:text="關(guān)于" android:textColor="#ffffff"/><RadioButtonandroid:id="@+id/RadioButton2"style="@style/tab_item_background"android:drawableTop="@drawable/tab_icon3"android:text="設(shè)置" android:textColor="#ffffff"/><RadioButtonandroid:id="@+id/RadioButton3"style="@style/tab_item_background"android:drawableTop="@drawable/tab_icon4"android:text="搜索" android:textColor="#ffffff"/><RadioButtonandroid:id="@+id/RadioButton4"style="@style/tab_item_background"android:drawableTop="@drawable/tab_icon5"android:text="更多" android:textColor="#ffffff"/></RadioGroup></LinearLayout></TabHost> main_tab_layout2.xml

2、然后在定義幾個(gè)用來(lái)存放Tab選項(xiàng)卡內(nèi)容的activity布局文件,同上activity1_layout.xml。

3、最后再定義一個(gè)自定義Tab按鈕的資源文件,selector_tab_background2.xml:

<?xml version="1.0" encoding="utf-8"?> <selector xmlns:android="http://schemas.android.com/apk/res/android"><item android:drawable="@drawable/tab_item_p" android:state_pressed="true"/><item android:drawable="@drawable/tab_item_d" android:state_checked="true"/></selector>

4、布局界面講解完畢,接下來(lái)詳細(xì)講解java代碼

package com.example.hiyou;import android.app.TabActivity; import android.content.Intent; import android.os.Bundle; import android.widget.RadioButton; import android.widget.RadioGroup; import android.widget.RadioGroup.OnCheckedChangeListener; import android.widget.TabHost; import android.widget.TabHost.TabSpec;import com.example.hiyou.Constant.ConValue; /*** 功能描述:第二種實(shí)現(xiàn)方式,自定義RadioGroup*/ public class TabActivity2 extends TabActivity {//定義TabHost對(duì)象private TabHost tabHost;//定義RadioGroup對(duì)象private RadioGroup radioGroup;public void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.main_tab_layout2);initView();initData();}/*** 初始化組件*/private void initView(){//實(shí)例化TabHost,得到TabHost對(duì)象tabHost = getTabHost();//得到Activity的個(gè)數(shù)int count = ConValue.mTabClassArray.length; for(int i = 0; i < count; i++){ //為每一個(gè)Tab按鈕設(shè)置圖標(biāo)、文字和內(nèi)容TabSpec tabSpec = tabHost.newTabSpec(ConValue.mTextviewArray[i]).setIndicator(ConValue.mTextviewArray[i]).setContent(getTabItemIntent(i));//將Tab按鈕添加進(jìn)Tab選項(xiàng)卡中 tabHost.addTab(tabSpec);}//實(shí)例化RadioGroupradioGroup = (RadioGroup) findViewById(R.id.main_radiogroup);}/*** 初始化組件*/private void initData() {// 給radioGroup設(shè)置監(jiān)聽(tīng)事件radioGroup.setOnCheckedChangeListener(new OnCheckedChangeListener() {@Overridepublic void onCheckedChanged(RadioGroup group, int checkedId) {switch (checkedId) {case R.id.RadioButton0:tabHost.setCurrentTabByTag(ConValue.mTextviewArray[0]);break;case R.id.RadioButton1:tabHost.setCurrentTabByTag(ConValue.mTextviewArray[1]);break;case R.id.RadioButton2:tabHost.setCurrentTabByTag(ConValue.mTextviewArray[2]);break;case R.id.RadioButton3:tabHost.setCurrentTabByTag(ConValue.mTextviewArray[3]);break;case R.id.RadioButton4:tabHost.setCurrentTabByTag(ConValue.mTextviewArray[4]);break;}}});((RadioButton) radioGroup.getChildAt(0)).toggle();}/*** 給Tab選項(xiàng)卡設(shè)置內(nèi)容(每個(gè)內(nèi)容都是一個(gè)Activity)*/private Intent getTabItemIntent(int index){Intent intent = new Intent(this, ConValue.mTabClassArray[index]); return intent;}} TabActivity2.java

5、最后再定義Tab選項(xiàng)卡內(nèi)容的Activity,同上Activity1.java。

源代碼下載:HiYou.zip

資料來(lái)源:【Android UI設(shè)計(jì)與開(kāi)發(fā)】第06期:底部菜單欄(一)使用TabActivity實(shí)現(xiàn)底部菜單欄

?

3.10 ListView

?ListView類(lèi)官方文檔地址:http://developer.android.com/reference/android/widget/ListView.html

ListView(列表視圖)是一個(gè)常用的組件,ListView里面的每個(gè)子項(xiàng)Item可以是一個(gè)字符串,也可以是一個(gè)組合控件。其數(shù)據(jù)內(nèi)容以列表形式直接展示出來(lái),比如做一個(gè)游戲的排行榜,對(duì)話列表等等都可以使用列表來(lái)實(shí)現(xiàn),且ListView的優(yōu)點(diǎn)是列表中的數(shù)據(jù)可以自適應(yīng)屏幕大小。

在android中,由于數(shù)據(jù)來(lái)源多種多樣,如從資源文件讀取、從數(shù)據(jù)庫(kù)中讀取、從網(wǎng)絡(luò)上其他地方讀取,而最終這些數(shù)據(jù)都將被展示在ListView中,所以android就用adapter設(shè)計(jì)模式,對(duì)應(yīng)每種數(shù)據(jù)來(lái)源使用對(duì)應(yīng)的adapter來(lái)連接數(shù)據(jù)和視圖。Adapter就是數(shù)據(jù)和視圖之間的橋梁,數(shù)據(jù)在adapter中做處理,然后顯示到ListView上面。

下面主要介紹三種adapter:ArrayAdapter<T>、SimpleAdapter和SimpleCursorAdapter。
1.ArrayAdapter<T>:最簡(jiǎn)單的適配器

ArrayAdapter類(lèi)官方文檔地址:http://developer.android.com/reference/android/widget/ArrayAdapter.html

?

首先創(chuàng)建存放ListView的Activity所需要的布局activity_main.xml文件。

<LinearLayout 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" > <ListView android:id="@+id/list" android:layout_width="match_parent" android:layout_height="match_parent" /> </LinearLayout>

上面代碼創(chuàng)建了一個(gè)布局配置文件,里面只放了一個(gè)ListView控件,將其ID設(shè)置為:list。
接下來(lái)是list_item.xml,用來(lái)設(shè)置ListView中每個(gè)Item的布局,是ListItem的XML實(shí)現(xiàn)。
Android提供了多種ListItem的Layout (R.layout),以下是較為常用的:

android.R.layout.simple_list_item_1 //一行textandroid.R.layout.simple_list_item_2 //一行title,一行textandroid.R.layout.simple_list_item_single_choice //單選按鈕android.R.layout.simple_list_item_multiple_choice //多選按鈕android.R.layout.simple_list_item_checked //checkbox

我們可以自定義自己的Layout(list_item.xml):

<TextView xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:textStyle="bold" android:textSize="30sp" android:padding="10sp"> </TextView>

要注意的是自定義list_item.xml的根節(jié)點(diǎn)必須是TextView,否則就會(huì)有ArrayAdapter requires the resource ID to be a TextView的錯(cuò)誤。
最后是MainActivity.java代碼,先找出ListView,然后往ListView里填充數(shù)組data。

package com.example.hiyou;import android.app.Activity; import android.os.Bundle; import android.view.View; import android.widget.AdapterView; import android.widget.AdapterView.OnItemClickListener; import android.widget.ArrayAdapter; import android.widget.ListView; import android.widget.Toast;public class MainActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);initView();}/*** 初始化組件*/private void initView() {String[] data = { "列表1", "列表2", "列表3", "列表4", "列表5" };// 綁定XML中的ListView,作為data的容器ListView listview = (ListView) findViewById(R.id.list);/** 實(shí)例化適配器 * 第一個(gè)參數(shù):Context * 第二個(gè)參數(shù):ListView中每一行布局樣式 * 第三個(gè)參數(shù):列表數(shù)據(jù)容器*/ArrayAdapter<String> arrayAdapter = new ArrayAdapter<String>(this,R.layout.list_item, data);listview.setAdapter(arrayAdapter);// 將適配器數(shù)據(jù)映射ListView上listview.setOnItemClickListener(new OnItemClickListener() {@Overridepublic void onItemClick(AdapterView<?> arg0, View arg1, int arg2,long arg3) {Toast.makeText(MainActivity.this,"當(dāng)前選中列表項(xiàng)的下標(biāo)為:" + arg2, Toast.LENGTH_SHORT).show();}});}} MainActivity.class

顯示一個(gè)帶有數(shù)據(jù)的ListView的步驟如下:
1.實(shí)例一個(gè)添加數(shù)據(jù)的容器,并將數(shù)據(jù)放入容器。
2.實(shí)例列表適配器,并且實(shí)例適配器時(shí)將數(shù)據(jù)傳入。
3.實(shí)例一個(gè)ListView,并且為其設(shè)置適配器。
4.利用setContentView()函數(shù)顯示ListView
因?yàn)榱斜碇忻恳豁?xiàng)數(shù)據(jù)都是一個(gè)Item,所以將ListView綁定使用OnItemClickListener項(xiàng)單擊監(jiān)聽(tīng)器,并且重寫(xiě)監(jiān)聽(tīng)器中的onItemClick()函數(shù)。
onItemClick()函數(shù)的第一個(gè)參數(shù)是出發(fā)的適配器,第二個(gè)參數(shù)數(shù)觸發(fā)的視圖,第三個(gè)參數(shù)是適配器中項(xiàng)的位置下標(biāo),第四個(gè)參數(shù)是ListView項(xiàng)下標(biāo)。

?

2.SimpleAdapter:具有很好擴(kuò)展性的適配器,可以顯示自定義內(nèi)容。

SimpleAdapter類(lèi)官方文檔地址:http://developer.android.com/reference/android/widget/SimpleAdapter.html

?

修改前面Demo的list_item.xml和MainActivity.class文件

<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="horizontal" ><ImageViewandroid:id="@+id/iv"android:layout_width="80dp"android:layout_height="80dp"android:layout_marginLeft="8dp"android:layout_marginRight="8dp" /><LinearLayoutandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:orientation="vertical" ><TextViewandroid:id="@+id/bigtv"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_marginTop="3dp"android:textSize="20sp" /><TextViewandroid:id="@+id/smalltv"android:layout_width="wrap_content"android:layout_height="wrap_content"android:textSize="12sp" /></LinearLayout></LinearLayout> list_item.xml package com.example.hiyou;import java.util.ArrayList; import java.util.HashMap; import java.util.List;import android.app.Activity; import android.os.Bundle; import android.view.View; import android.widget.AdapterView; import android.widget.AdapterView.OnItemClickListener; import android.widget.ListView; import android.widget.SimpleAdapter; import android.widget.Toast;public class MainActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);initView();}/*** 初始化組件*/private void initView() {// 創(chuàng)建動(dòng)態(tài)數(shù)組數(shù)據(jù)源 List<HashMap<String, Object>> data = new ArrayList<HashMap<String, Object>>();// 實(shí)例化一個(gè)列表數(shù)據(jù)容器HashMap<String, Object> map1 = new HashMap<String, Object>();// 往列表容器中添加數(shù)據(jù)/** map.put(String key,Object value) 第一個(gè)參數(shù)用于初始化適配器時(shí)需要映射數(shù)據(jù)對(duì)應(yīng)的索引;* 第二個(gè)參數(shù)表示對(duì)應(yīng)自定義項(xiàng)布局中的組件數(shù)據(jù)* 進(jìn)行添加數(shù)據(jù)時(shí),每一個(gè)put()函數(shù)都對(duì)應(yīng)自定義ListView項(xiàng)中的一個(gè)組件;按鈕、復(fù)選框等組件是無(wú)法映射的。*/map1.put("item1_imageview", R.drawable.list1);map1.put("item1_bigtv", "一加手機(jī)發(fā)布:強(qiáng)調(diào)手感 ");map1.put("item1_smalltv", "國(guó)內(nèi)手機(jī)新品牌一加手機(jī)今日在北京發(fā)布其首款產(chǎn)品,這是一款強(qiáng)調(diào)設(shè)計(jì)的手機(jī)新品,配備驍龍801處理器。16GB版售價(jià)1999.99元。");// 將列表數(shù)據(jù)添加到列表容器中data.add(map1);HashMap<String, Object> map2 = new HashMap<String, Object>();map2.put("item1_imageview", R.drawable.list2);map2.put("item1_bigtv", " LG L90美國(guó)發(fā)售");map2.put("item1_smalltv", "今日,LG L90正式在美國(guó)以T-Mobile定制機(jī)的形式進(jìn)行發(fā)售,售價(jià)為228美元。");data.add(map2);// 綁定XML中的ListView,作為data的容器ListView listview = (ListView) findViewById(R.id.list);// 動(dòng)態(tài)數(shù)組數(shù)據(jù)源中與ListItem中每個(gè)顯示項(xiàng)對(duì)應(yīng)的Key String[] from = new String[] { "item1_imageview", "item1_bigtv", "item1_smalltv"}; // ListItem的XML文件里面的一個(gè)ImageView ID和兩個(gè)TextView ID int[] to = new int[] { R.id.iv, R.id.bigtv, R.id.smalltv }; // 將動(dòng)態(tài)數(shù)組數(shù)據(jù)源data中的數(shù)據(jù)填充到ListItem的XML文件list_item.xml中去 // 從動(dòng)態(tài)數(shù)組數(shù)據(jù)源data中,取出from數(shù)組中key對(duì)應(yīng)的value值,填充到to數(shù)組中對(duì)應(yīng)ID的控件中去 /** 實(shí)例化SimpleAdapter適配器構(gòu)造函數(shù)Simple(Contect context,List data,int resource,String[] from,int[] to)* context:當(dāng)前context對(duì)象* data:ListView各項(xiàng)數(shù)據(jù) * resource:ListView每一項(xiàng)的布局 * from:每一項(xiàng)布局中的數(shù)據(jù)映射索引數(shù)組* to:每一項(xiàng)中數(shù)據(jù)對(duì)應(yīng)的組件ID數(shù)組*/SimpleAdapter adapter = new SimpleAdapter(this, data, R.layout.list_item, from, to);listview.setAdapter(adapter);// 將適配器數(shù)據(jù)映射ListView上listview.setOnItemClickListener(new OnItemClickListener() {@Overridepublic void onItemClick(AdapterView<?> arg0, View arg1, int arg2,long arg3) {Toast.makeText(MainActivity.this,"當(dāng)前選中列表項(xiàng)的為第" + (arg2+1)+"列。", Toast.LENGTH_SHORT).show();}});}} MainActivity.class

3.SimpleCursorAdapter

SimpleCursorAdapter類(lèi)官方文檔地址:http://developer.android.com/reference/android/widget/SimpleCursorAdapter.html

?下面用SimpleCursorAdapter來(lái)實(shí)現(xiàn)上一節(jié)中用SimpleAdapter實(shí)現(xiàn)的同樣的效果,activity_main.xml文件和list_item.xml文件都不需要更改,只需要更改MainActivity.java代碼。

package com.example.hiyou;import android.app.Activity; import android.content.ContentValues; import android.database.Cursor; import android.os.Bundle; import android.view.View; import android.widget.AdapterView; import android.widget.AdapterView.OnItemClickListener; import android.widget.ListView; import android.widget.SimpleCursorAdapter; import android.widget.Toast;public class MainActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);initView();}/*** 初始化組件*/private void initView() {DBHelper dbHelper = new DBHelper(this); // 向數(shù)據(jù)庫(kù)中插入數(shù)據(jù) insertDataIntoDB(dbHelper); Cursor cursor = dbHelper.query(); // 綁定XML中的ListView,作為data的容器ListView listview = (ListView) findViewById(R.id.list);// 動(dòng)態(tài)數(shù)組數(shù)據(jù)源中與ListItem中每個(gè)顯示項(xiàng)對(duì)應(yīng)的Key,要與創(chuàng)建的數(shù)據(jù)庫(kù)列名一樣 String[] from = new String[] { "iv", "bigtv", "smalltv"}; // ListItem的XML文件里面的一個(gè)ImageView ID和兩個(gè)TextView ID int[] to = new int[] { R.id.iv, R.id.bigtv, R.id.smalltv }; // 將動(dòng)態(tài)數(shù)組數(shù)據(jù)源data中的數(shù)據(jù)填充到ListItem的XML文件list_item.xml中去 // 從動(dòng)態(tài)數(shù)組數(shù)據(jù)源data中,取出from數(shù)組中key對(duì)應(yīng)的value值,填充到to數(shù)組中對(duì)應(yīng)ID的控件中去 /** 實(shí)例化SimpleCursorAdapter適配器構(gòu)造函數(shù)SimpleCursorAdapter(context, layout, c, from, to)* context:當(dāng)前context對(duì)象* layout每一項(xiàng)的布局 * c:* from:每一項(xiàng)布局中的數(shù)據(jù)映射索引數(shù)組* to:每一項(xiàng)中數(shù)據(jù)對(duì)應(yīng)的組件ID數(shù)組*/SimpleCursorAdapter adapter = new SimpleCursorAdapter (this, R.layout.list_item,cursor, from, to);listview.setAdapter(adapter);// 將適配器數(shù)據(jù)映射ListView上listview.setOnItemClickListener(new OnItemClickListener() {@Overridepublic void onItemClick(AdapterView<?> arg0, View arg1, int arg2,long arg3) {Toast.makeText(MainActivity.this,"當(dāng)前選中列表項(xiàng)的為第" + (arg2+1)+"列。", Toast.LENGTH_SHORT).show();}});}private void insertDataIntoDB(DBHelper dbHelper) {dbHelper.clear();//向數(shù)據(jù)庫(kù)插入數(shù)據(jù)ContentValues values1 = new ContentValues();values1.put("iv", R.drawable.list1);values1.put("bigtv", "一加手機(jī)發(fā)布:強(qiáng)調(diào)手感 ");values1.put("smalltv","國(guó)內(nèi)手機(jī)新品牌一加手機(jī)今日在北京發(fā)布其首款產(chǎn)品,這是一款強(qiáng)調(diào)設(shè)計(jì)的手機(jī)新品,配備驍龍801處理器。16GB版售價(jià)1999.99元。");dbHelper.insert(values1);ContentValues values2 = new ContentValues();values2.put("iv", R.drawable.list2);values2.put("bigtv", "LG L90美國(guó)發(fā)售 ");values2.put("smalltv", "今日,LG L90正式在美國(guó)以T-Mobile定制機(jī)的形式進(jìn)行發(fā)售,售價(jià)為228美元。");dbHelper.insert(values2);} } MainActivity.class

?

這里通過(guò)DBHelper這個(gè)類(lèi)來(lái)實(shí)現(xiàn)數(shù)據(jù)庫(kù)的插入和查詢(xún)功能。

package com.example.hiyou;import android.content.ContentValues; import android.content.Context; import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import android.database.sqlite.SQLiteOpenHelper;public class DBHelper extends SQLiteOpenHelper { public DBHelper(Context context) { super(context, "testDB", null, 1); } @Override public void onCreate(SQLiteDatabase db) { //如果數(shù)據(jù)庫(kù)不存在創(chuàng)建數(shù)據(jù)庫(kù)tbl_testString createTableSQL = "create table IF NOT EXISTS tbl_test " + "(_id integer primary key autoincrement, iv int, " + "bigtv text, smalltv text)"; db.execSQL(createTableSQL); } //數(shù)據(jù)新增操作public void insert(ContentValues values) { SQLiteDatabase db = getWritableDatabase(); db.insert("tbl_test", null, values); } //游標(biāo)查詢(xún)數(shù)據(jù)庫(kù)public Cursor query() { SQLiteDatabase db = getWritableDatabase(); Cursor cursor = db.query("tbl_test", null, null, null, null, null, null); return cursor; } //清除數(shù)據(jù)庫(kù)中的數(shù)據(jù)public void clear() { SQLiteDatabase db = getWritableDatabase(); db.delete("tbl_test", null, null); } //關(guān)閉讀取數(shù)據(jù)庫(kù)public void close() { SQLiteDatabase db = getWritableDatabase(); db.close(); } @Override public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) { } } DBHelper.class

?

自定義Adapter

使用android提供的adapter來(lái)繪制列表的話,列表的每一項(xiàng)的顯示都是一樣的。而且按鈕和復(fù)選框等這些事件的組件其實(shí)是無(wú)法將數(shù)據(jù)映射在ListView上的。所以如果要監(jiān)聽(tīng)和響應(yīng)按鈕、復(fù)選框等組件的事件時(shí),則需要進(jìn)行自定義適配器來(lái)完成。

下面示例實(shí)現(xiàn)獲取SD卡內(nèi)的MP3格式歌曲信息通過(guò)ListView顯示歌曲專(zhuān)輯圖片、歌曲名稱(chēng)、歌手名,并且ListView的單雙行不同顏色顯示,這需要自定義adapter的子類(lèi)。adapter的常用子類(lèi)有BaseAdapter、ArrayAdapter、SimpleAdapter等,下面介紹自定義BaseAdapter和ArrayAdapter的實(shí)現(xiàn)。

1.自定義BaseAdapter

?

為了實(shí)現(xiàn)ListView的單雙行不同顏色顯示,需要自定義adapter的子類(lèi),下面我們實(shí)現(xiàn)自定義的MusicAdapter類(lèi)。MusicAdapter類(lèi)繼承自BaseAdapter類(lèi),BaseAdapter為抽象類(lèi),繼承它需要實(shí)現(xiàn)如下方法,因此具有較高的靈活性。

public class MusicAdapter extends BaseAdapter { @Override public int getCount() { return 0; } @Override public Object getItem(int arg0) { return null; } @Override public long getItemId(int position) { return 0; } //實(shí)例化布局和組件以及設(shè)置組件數(shù)據(jù)//getView(int position, View convertView, ViewGroup parent) //position:繪制的行數(shù)//convertView:繪制的視圖,這里指的是ListView中的每一項(xiàng)布局//parent:view的合集 @Override public View getView(int position, View convertView, ViewGroup parent) { // TODO Auto-generated method stub return null; } }

ListView在繪制時(shí)首先會(huì)調(diào)用getCount()方法得到繪制次數(shù),然后通過(guò)getView()方法一層一層進(jìn)行繪制,所以我們可以在getView()方法中根據(jù)position(當(dāng)前繪制的ID)來(lái)的修改繪制內(nèi)容。而getItem()和getItemId()則在需要處理和取得Adapter中的數(shù)據(jù)時(shí)調(diào)用。

package com.example.hiyou;import java.util.ArrayList; import java.util.List; import android.content.ContentResolver; import android.content.Context; import android.database.Cursor; import android.graphics.Bitmap; import android.provider.MediaStore; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.BaseAdapter; import android.widget.ImageView; import android.widget.TextView;public class MusicAdapter extends BaseAdapter { private int[] colors = new int[] { 0xff3cb371, 0xffa0a0a0 }; // 用來(lái)獲得ContentProvider(共享數(shù)據(jù)庫(kù))public ContentResolver cr;// 用來(lái)裝查詢(xún)到的音樂(lè)文件數(shù)據(jù)public Cursor cur;// 歌曲信息列表public List<MusicInfo> musicList;public Context context;public MusicAdapter(Context context) {this.context = context;// 取得數(shù)據(jù)庫(kù)對(duì)象cr = context.getContentResolver();musicList = new ArrayList<MusicInfo>();String[] mString = new String[] { MediaStore.Audio.Media.DISPLAY_NAME,MediaStore.Audio.Media.ALBUM, MediaStore.Audio.Media.ARTIST,MediaStore.Audio.Media.DURATION, MediaStore.Audio.Media.SIZE,MediaStore.Audio.Media.ALBUM_ID, MediaStore.Audio.Media.DATA,MediaStore.Audio.Media._ID };// 查詢(xún)所有音樂(lè)信息cur = cr.query(MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, mString,null, null, null);if (cur != null) {// 移動(dòng)游標(biāo)到第一個(gè) cur.moveToFirst();int j = 1;for (int i = 0; i < cur.getCount(); i++) {if (cur.getString(0).endsWith(".mp3")) {// 過(guò)濾獲取MP3文件MusicInfo mInfo = new MusicInfo();String musicName = cur.getString(0).substring(0,cur.getString(0).lastIndexOf(".mp3"));mInfo.setMusicIndex(j++);mInfo.setMusicName(musicName);mInfo.setMusicAlubm(cur.getString(1));mInfo.setMusicSinger(cur.getString(2));mInfo.setMusicTime(cur.getInt(3));mInfo.setMusicSize(cur.getInt(4));mInfo.setMusicAlubmId(cur.getInt(5));mInfo.setMusicPath(cur.getString(6));mInfo.setMusicId(cur.getInt(7));musicList.add(mInfo);}cur.moveToNext();}}}@Overridepublic int getCount() {return musicList.size();//返回ListView項(xiàng)的長(zhǎng)度 }@Overridepublic Object getItem(int arg0) {return musicList.get(arg0);}@Overridepublic long getItemId(int arg0) {return arg0;}//實(shí)例化布局和組件以及設(shè)置組件數(shù)據(jù)//getView(int position, View convertView, ViewGroup parent) //position:繪制的行數(shù)//convertView:繪制的視圖,這里指的是ListView中的每一項(xiàng)布局//parent:view的合集 @Overridepublic View getView(int position, View convertView, ViewGroup parent) {ViewHolder holder = null; if (convertView == null) { holder = new ViewHolder(); //將布局通過(guò)LayoutInflater對(duì)象實(shí)例化為一個(gè)viewconvertView = LayoutInflater.from(context).inflate( R.layout.list_item, null); holder.songImage = (ImageView) convertView.findViewById(R.id.listImage); holder.singerName = (TextView) convertView.findViewById(R.id.list_Singer); holder.songName = (TextView) convertView.findViewById(R.id.listName); // 將holder綁定到convertView convertView.setTag(holder); }else { holder = (ViewHolder) convertView.getTag(); } // 向ViewHolder中填入的數(shù)據(jù) int mid = musicList.get(position).getMusicIndex();String musicName = musicList.get(position).getMusicName();String musciSinger = musicList.get(position).getMusicSinger();if (musciSinger.contains("<unknown>")) {musciSinger = "<未知>";}Bitmap img = MusicUtils.getArtwork(context,musicList.get(position).getMusicId(),musicList.get(position).getMusicAlubmId(), true);holder.songName.setText(mid + ". " + musicName);holder.singerName.setText(musciSinger);holder.songImage.setImageBitmap(img);int colorPos = position % colors.length; convertView.setBackgroundColor(colors[colorPos]); //控制背景顏色return convertView;}/** * ViewHolder類(lèi)用以?xún)?chǔ)存item中控件的引用 */ final class ViewHolder { ImageView songImage; TextView songName; TextView singerName; } } MusicAdapter.class

getView()方法用來(lái)獲得繪制每個(gè)item的View對(duì)象,如果每次getView()被執(zhí)行都new出一個(gè)View對(duì)象,長(zhǎng)此以往會(huì)產(chǎn)生很大的消耗,特別當(dāng)item中還有Bitmap等,甚至?xí)斐蒓OM的錯(cuò)誤導(dǎo)致程序崩潰。從上面的代碼可以看到getView()有一個(gè)convertView參數(shù),這個(gè)參數(shù)用來(lái)緩存View對(duì)象。當(dāng)ListView滑動(dòng)的過(guò)程中,會(huì)有item被滑出屏幕而不再被使用,這時(shí)候Android會(huì)回收這個(gè)item的view,這個(gè)view也就是這里的convertView。這樣如果convertView不為null,就不用new出一個(gè)新的View對(duì)象,只用往convertView中填充新的item,這樣就省去了new View的大量開(kāi)銷(xiāo)。

在上面的代碼中,在緩存convertView減少new View開(kāi)銷(xiāo)的同時(shí),通過(guò)setTag()方法將數(shù)據(jù)結(jié)構(gòu)ViewHolder綁定到convertView,從而利用ViewHolder存儲(chǔ)convertView中控件對(duì)象的引用,這樣避免每次調(diào)用findViewById()方法。

相關(guān)類(lèi):

package com.example.hiyou;import android.app.Activity; import android.os.Bundle; import android.widget.ListView;public class MainActivity extends Activity {public MusicAdapter mAdapter;private ListView mListView;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);initView();}/*** 初始化組件*/private void initView() {// 綁定XML中的ListView,作為Item的容器mListView = (ListView) findViewById(R.id.list);mAdapter = new MusicAdapter(MainActivity.this);mListView.setAdapter(mAdapter);}} MainActivity.class package com.example.hiyou;import java.io.FileDescriptor; import java.io.FileNotFoundException; import java.io.IOException; import java.io.InputStream; import android.content.ContentResolver; import android.content.ContentUris; import android.content.Context; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.net.Uri; import android.os.ParcelFileDescriptor; /**** * @author Jerryc*音樂(lè)助手類(lèi)*/ public class MusicUtils {private static final Uri sArtworkUri = Uri.parse("content://media/external/audio/albumart");private static final BitmapFactory.Options sBitmapOptions = new BitmapFactory.Options();private static Bitmap mCachedBit = null; //獲取音樂(lè)文件專(zhuān)輯圖片public static Bitmap getArtwork(Context context, long song_id,long album_id, boolean allowdefault) {if (album_id < 0) {// This is something that is not in the database, so get the album// art directly// from the file.if (song_id >= 0) {Bitmap bm = getArtworkFromFile(context, song_id, -1);if (bm != null) {return bm;}}if (allowdefault) {return getDefaultArtwork(context);}return null;}ContentResolver res = context.getContentResolver();Uri uri = ContentUris.withAppendedId(sArtworkUri, album_id);if (uri != null) {InputStream in = null;try {in = res.openInputStream(uri);return BitmapFactory.decodeStream(in, null, sBitmapOptions);} catch (FileNotFoundException ex) {// The album art thumbnail does not actually exist. Maybe the// user deleted it, or// maybe it never existed to begin with.Bitmap bm = getArtworkFromFile(context, song_id, album_id);if (bm != null) {if (bm.getConfig() == null) {bm = bm.copy(Bitmap.Config.RGB_565, false);if (bm == null && allowdefault) {return getDefaultArtwork(context);}}} else if (allowdefault) {bm = getDefaultArtwork(context);}return bm;} finally {try {if (in != null) {in.close();}} catch (IOException ex) {}}}return null;}private static Bitmap getArtworkFromFile(Context context, long songid,long albumid) {Bitmap bm = null;byte[] art = null;String path = null;if (albumid < 0 && songid < 0) {throw new IllegalArgumentException("Must specify an album or a song id");}try {if (albumid < 0) {Uri uri = Uri.parse("content://media/external/audio/media/"+ songid + "/albumart");ParcelFileDescriptor pfd = context.getContentResolver().openFileDescriptor(uri, "r");if (pfd != null) {FileDescriptor fd = pfd.getFileDescriptor();bm = BitmapFactory.decodeFileDescriptor(fd);}} else {Uri uri = ContentUris.withAppendedId(sArtworkUri, albumid);ParcelFileDescriptor pfd = context.getContentResolver().openFileDescriptor(uri, "r");if (pfd != null) {FileDescriptor fd = pfd.getFileDescriptor();bm = BitmapFactory.decodeFileDescriptor(fd);}}} catch (FileNotFoundException ex) {}if (bm != null) {mCachedBit = bm;}return bm;}private static Bitmap getDefaultArtwork(Context context) {BitmapFactory.Options opts = new BitmapFactory.Options();opts.inPreferredConfig = Bitmap.Config.RGB_565;return BitmapFactory.decodeStream(context.getResources().openRawResource(R.drawable.album), null, opts);} } MusicUtils.class <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="wrap_content"android:orientation="horizontal" ><LinearLayoutandroid:layout_width="50sp"android:layout_height="50sp"android:orientation="vertical" android:gravity="center" ><ImageViewandroid:id="@+id/listImage"android:layout_width="40sp"android:layout_height="40sp" /></LinearLayout><LinearLayoutandroid:layout_width="match_parent"android:layout_height="50sp"android:orientation="vertical" ><TextViewandroid:id="@+id/listName"android:layout_width="fill_parent"android:layout_height="match_parent"android:layout_weight="1"android:gravity="center_vertical"android:paddingLeft="10dp"android:singleLine="true"android:textSize="16sp" /><TextViewandroid:id="@+id/list_Singer"android:layout_width="match_parent"android:layout_height="match_parent"android:layout_weight="1"android:gravity="center_vertical"android:paddingLeft="10dp"android:singleLine="true"android:textSize="13sp" /></LinearLayout></LinearLayout> list_item.xml package com.example.hiyou;/*** 歌曲信息類(lèi)*/ public class MusicInfo {private int musicIndex; //排序號(hào)private int songId;//歌曲IDprivate int musicAlubmId;//專(zhuān)輯IDprivate String musicName;// 歌曲名private String musicSinger;// 歌手名private int musicTime;// 歌曲時(shí)間長(zhǎng)度private String musicAlubm;// 專(zhuān)輯名稱(chēng)private int musicSize;// 曲歌大小private String musicPath;// 歌曲路徑public int getMusicIndex() {return musicIndex;}public void setMusicIndex(int musicIndex) {this.musicIndex = musicIndex;}public int getMusicId() {return songId;}public void setMusicId(int songId) {this.songId = songId;}public int getMusicAlubmId() {return musicAlubmId;}public void setMusicAlubmId(int musicAlubmId) {this.musicAlubmId = musicAlubmId;}public String getMusicName() {return musicName;}public void setMusicName(String musicName) {this.musicName = musicName;}public String getMusicSinger() {return musicSinger;}public void setMusicSinger(String musicSinger) {this.musicSinger = musicSinger;}public int getMusicTime() {return musicTime;}public void setMusicTime(int musicTime) {this.musicTime = musicTime;}public String getMusicAlubm() {return musicAlubm;}public void setMusicAlubm(String musicAlubm) {this.musicAlubm = musicAlubm;}public int getMusicSize() {return musicSize;}public void setMusicSize(int musicSize) {this.musicSize = musicSize;}public String getMusicPath() {return musicPath;}public void setMusicPath(String musicPath) {this.musicPath = musicPath;}} MusicInfo.class

2.自定義ArrayAdapter<T>

在開(kāi)發(fā)中需要將對(duì)象顯示在listview中,這時(shí)候使用ArrayAdapter<T>來(lái)顯示指定對(duì)象類(lèi)型。下面自定義ArrayAdapter<T>實(shí)現(xiàn)上一節(jié)中自定義BaseAdapter實(shí)現(xiàn)的同樣的效果,首先定義要顯示的對(duì)象,代碼參照前面的MusicInfo.class

MainActivity.java代碼修改如下:

package com.example.hiyou;import java.util.ArrayList; import android.app.Activity; import android.content.ContentResolver; import android.database.Cursor; import android.os.Bundle; import android.provider.MediaStore; import android.widget.ListView;public class MainActivity extends Activity {public MyArrayAdapter mAdapter;private ListView mListView;// 用來(lái)獲得ContentProvider(共享數(shù)據(jù)庫(kù))public ContentResolver cr;// 用來(lái)裝查詢(xún)到的音樂(lè)文件數(shù)據(jù)public Cursor cur;// 歌曲信息列表public ArrayList<MusicInfo> musicList;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);initView();}/*** 初始化組件*/private void initView() {// 取得數(shù)據(jù)庫(kù)對(duì)象cr = getContentResolver();musicList = new ArrayList<MusicInfo>();String[] mString = new String[] { MediaStore.Audio.Media.DISPLAY_NAME,MediaStore.Audio.Media.ALBUM, MediaStore.Audio.Media.ARTIST,MediaStore.Audio.Media.DURATION, MediaStore.Audio.Media.SIZE,MediaStore.Audio.Media.ALBUM_ID, MediaStore.Audio.Media.DATA,MediaStore.Audio.Media._ID };// 查詢(xún)所有音樂(lè)信息cur = cr.query(MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, mString,null, null, null);if (cur != null) {// 移動(dòng)游標(biāo)到第一個(gè) cur.moveToFirst();int j = 1;for (int i = 0; i < cur.getCount(); i++) {if (cur.getString(0).endsWith(".mp3")) {// 過(guò)濾獲取MP3文件MusicInfo mInfo = new MusicInfo();String musicName = cur.getString(0).substring(0,cur.getString(0).lastIndexOf(".mp3"));mInfo.setMusicIndex(j++);mInfo.setMusicName(musicName);mInfo.setMusicAlubm(cur.getString(1));mInfo.setMusicSinger(cur.getString(2));mInfo.setMusicTime(cur.getInt(3));mInfo.setMusicSize(cur.getInt(4));mInfo.setMusicAlubmId(cur.getInt(5));mInfo.setMusicPath(cur.getString(6));mInfo.setMusicId(cur.getInt(7));musicList.add(mInfo);}cur.moveToNext();}}// 綁定XML中的ListView,作為Item的容器mListView = (ListView) findViewById(R.id.list);mAdapter = new MyArrayAdapter(MainActivity.this, R.layout.list_item,musicList);mListView.setAdapter(mAdapter);}} MainActivity

接下來(lái)自定義繼承自ArrayAdapter<MusicInfo>的MyArrayAdapter類(lèi),繼承ArrayAdapter<MusicInfo>只需要重寫(xiě)getView()方法就可以實(shí)現(xiàn)與上一節(jié)相同的效果,并且不用保存List<MusicInfo>對(duì)象引用。

package com.example.hiyou;import java.util.List; import android.content.Context; import android.graphics.Bitmap; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.ArrayAdapter; import android.widget.ImageView; import android.widget.TextView;public class MyArrayAdapter extends ArrayAdapter<MusicInfo> {private int[] colors = new int[] { 0xff3cb371, 0xffa0a0a0 };private Context mContext;private int resource;public MyArrayAdapter(Context context, int resource,List<MusicInfo> musicList) {super(context, resource,musicList);this.mContext = context;this.resource = resource;}@Overridepublic View getView(int position, View convertView, ViewGroup parent) {ViewHolder holder = null; if (convertView == null) { holder = new ViewHolder(); convertView = LayoutInflater.from(mContext).inflate( resource, null); holder.songImage = (ImageView) convertView.findViewById(R.id.listImage); holder.singerName = (TextView) convertView.findViewById(R.id.list_Singer); holder.songName = (TextView) convertView.findViewById(R.id.listName); // 將holder綁定到convertView convertView.setTag(holder); }else { holder = (ViewHolder) convertView.getTag(); } // 向ViewHolder中填入的數(shù)據(jù) int mid =getItem(position).getMusicIndex();String musicName = getItem(position).getMusicName();String musciSinger =getItem(position).getMusicSinger();if (musciSinger.contains("<unknown>")) {musciSinger = "<未知>";}Bitmap img = MusicUtils.getArtwork(mContext,getItem(position).getMusicId(),getItem(position).getMusicAlubmId(), true);holder.songName.setText(mid + ". " + musicName);holder.singerName.setText(musciSinger);holder.songImage.setImageBitmap(img);int colorPos = position % colors.length; convertView.setBackgroundColor(colors[colorPos]); //控制背景顏色return convertView;}/** * ViewHolder類(lèi)用以?xún)?chǔ)存item中控件的引用 */ final class ViewHolder { ImageView songImage; TextView songName; TextView singerName; } } MyArrayAdapter

?

總結(jié)

以上是生活随笔為你收集整理的【读书笔记《Android游戏编程之从零开始》】6.Android 游戏开发常用的系统控件(TabHost、ListView)...的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。