Android应用开发:数据存储和界面展现-2
1. pull解析XML文件
Android推薦使用pull解析XML文件,與SAX解析XML文件類似,都是事件驅(qū)動類型的解析方式。
示例:獲取天氣信息
res\layout\activity_main.xml
<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" ><Button android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="獲取天氣信息"android:onClick="click"/></RelativeLayout>src/weather.xml
<?xml version="1.0" encoding="utf-8"?> <weather><city><name>西安</name><temp>22°</temp><pm25>100</pm25></city><city><name>上海</name><temp>24°</temp><pm25>120</pm25></city><city><name>北京</name><temp>30°</temp><pm25>800</pm25></city> </weather>src/cn.itcast.pullParser.domain/City.java
package cn.itcast.pullparser.domain;public class City {private String name;private String temp;private String pm25;public String getName() {return name;}public void setName(String name) {this.name = name;}public String getTemp() {return temp;}public void setTemp(String temp) {this.temp = temp;}public String getPm25() {return pm25;}public void setPm25(String pm25) {this.pm25 = pm25;}@Overridepublic String toString() {return "City [name=" + name + ", temp=" + temp + ", pm25=" + pm25 + "]";} }src/cn.itcast.pullparser/MainActivity.java
package cn.itcast.pullparser;import java.io.InputStream; import java.util.ArrayList; import java.util.List; import org.xmlpull.v1.XmlPullParser; import android.app.Activity; import android.os.Bundle; import android.util.Xml; import android.view.View; import cn.itcast.pullparser.domain.City;public class MainActivity extends Activity {List<City> cityList;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);}public void click(View v){//獲取xml文件InputStream is = getClassLoader().getResourceAsStream("weather.xml");//獲取pull解析器//XmlPullParser是一個接口,使用Xml.newPullParser()獲取XmlPullParser對象XmlPullParser xp = Xml.newPullParser();//初始化//Android系統(tǒng)默認(rèn)編碼是“utf-8”try {xp.setInput(is,"utf-8");//開始解析//獲取事件類型,通過對事件類型的判斷,直到當(dāng)前解析的是什么節(jié)點int type = xp.getEventType();City city = null;while(type != XmlPullParser.END_DOCUMENT){switch(type){case XmlPullParser.START_TAG://獲取當(dāng)前節(jié)點的名字if("weather".equals(xp.getName())){cityList = new ArrayList<City>();}else if("city".equals(xp.getName())){city = new City();}else if("name".equals(xp.getName())){//獲取當(dāng)前節(jié)點的下一個節(jié)點的文本String name = xp.nextText();city.setName(name);}else if("temp".equals(xp.getName())){String temp = xp.nextText();city.setTemp(temp);}else if("pm25".equals(xp.getName())){String pm25 = xp.nextText();city.setPm25(pm25);}break;case XmlPullParser.END_TAG:if("city".equals(xp.getName())){cityList.add(city);}break;}//把指針移動至下一個節(jié)點,并返回該節(jié)點的事件類型type = xp.next();}} catch (Exception e) {e.printStackTrace();}for(City c:cityList){System.out.println(c.toString());}} }運行結(jié)果:
1.1 用debug查看pull解析流程
首先,雙擊代碼左側(cè),打一個斷點。
點擊debug “01_pull解析XML文件”。
等待debug初始化,跳出對話框,不要點“Force close”!等待一會,對話框就會消失。
點擊“獲取天氣信息”按鈕。
點擊“yes”,程序進(jìn)入debug模式。
通過“step over”即可以查看代碼執(zhí)行的每一步。
2. 測試概念
按崗位分:
- 黑盒測試:測試業(yè)務(wù)邏輯
- 白盒測試:測試邏輯方法
按測試粒度分:
- 方法測試 function
- 單元測試 unit:多個方法集成一個單元,測試
- 集成測試 intergration:多個單元集成,測試
- 系統(tǒng)測試 system:服務(wù)器端、客戶端端聯(lián)動,測試整個系統(tǒng)
按暴力程度分:
- 冒煙測試 smoke
- 壓力測試 pressure:針對服務(wù)器端的測試。
目前的智能手機都有測試框架,例如,Android自帶的monkey,不過需要在Android命令行中使用。
示例:隨機點擊模擬器一千次
進(jìn)入測試狀態(tài)…
2.1 單元測試
創(chuàng)建一個類,繼承AndroidTestCase
src/cn.itcast.junit/Test.java
package cn.itcast.junit;import cn.itcast.junit.tools.Tools; import android.test.AndroidTestCase;public class Test extends AndroidTestCase {//定義在測試框架中的方法可以直接運行public void test(){int result = Tools.damage(8, 3);//斷言,對比實際結(jié)果與預(yù)期是否一致assertEquals(5, result);} }src/cn.itcast.junit.tools/Tools.java
package cn.itcast.junit.tools;public class Tools {public static int damage(int i,int j){return i + j;} }使用測試框架需要在清單文件中配置指令集和使用類庫。
兩種方式啟動測試:方法一:在代碼區(qū)內(nèi)選擇需要測試的方法–>右擊Run As–>Android JUnit Test。
方法二:在Outline內(nèi)選擇需要測試的方法–>右擊Run As–>Android JUnit Test。
通過下圖,可以看到,報錯,結(jié)果與斷言不相符。
修改Tools.java中代碼。
通過下圖,可以看到,測試運行成功,結(jié)果與斷言相符。
通過Console可以看到測試過程。
雖然,測試過程中,Android項目并沒有啟動,整個測試過程可以看到模擬器上根本沒有前臺界面。但是,測試需要模擬器運行測試框架。因為,測試的項目是Android代碼,必須運行在Android設(shè)備上,可以是模擬器也可以是真機。沒有返回值的方法測試步驟相同,只是不需要斷言。
測試后,報錯,提示出現(xiàn)異常。
修改代碼,測試成功。
3. 創(chuàng)建數(shù)據(jù)庫
Android中存儲數(shù)據(jù)用的最多的還是數(shù)據(jù)庫,SQLite(Android自帶的輕量級數(shù)據(jù)庫)。
數(shù)據(jù)庫保存在內(nèi)部存儲空間。所以,在備份數(shù)據(jù)的時候,不會寫在數(shù)據(jù)庫里。一旦應(yīng)用刪除,數(shù)據(jù)庫就沒了,備份的數(shù)據(jù)也就全沒了。
示例:創(chuàng)建數(shù)據(jù)庫
創(chuàng)建一個類,MyOpenHelper,繼承SQLiteOpenHelper。
src/cn.itcast.sqlite/MyOpenHelper.java
package cn.itcast.sqlite;import android.content.Context; import android.database.sqlite.SQLiteDatabase; import android.database.sqlite.SQLiteDatabase.CursorFactory; import android.database.sqlite.SQLiteOpenHelper;public class MyOpenHelper extends SQLiteOpenHelper {public MyOpenHelper(Context context, String name, CursorFactory factory,int version) {//第一個參數(shù):用來打開和創(chuàng)建數(shù)據(jù)庫的上下文//第二個參數(shù):數(shù)據(jù)庫文件名字//第三個參數(shù):游標(biāo)工廠,用來創(chuàng)建游標(biāo)對象的工廠。如果傳null,表示使用默認(rèn)的游標(biāo)工廠//第四個參數(shù):數(shù)據(jù)庫的版本,最低為1,如果數(shù)據(jù)庫做升級,版本值只能升,不能降super(context, name, factory, version);}//數(shù)據(jù)庫創(chuàng)建時,此方法調(diào)用@Overridepublic void onCreate(SQLiteDatabase db) {}//數(shù)據(jù)庫升級時,此方法調(diào)用@Overridepublic void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {} }創(chuàng)建一個測試類,繼承AndroidTestCase。
src/cn.itcast.sqlite/MyOpenHelper.java
配置清單文件:
測試運行結(jié)果:
生成數(shù)據(jù)庫成功。
導(dǎo)出people.db,拖入SQLite Expert,可以看到自動生成了一張表,此表不能修改,不用管它。
3.1 創(chuàng)建表
創(chuàng)建數(shù)據(jù)庫完畢后會調(diào)用onCreate方法,數(shù)據(jù)庫升級后會調(diào)用onUpgrade方法。
示例:src/cn.itcast.sqlite/MyOpenHelper.java
package cn.itcast.sqlite;import android.content.Context; import android.database.sqlite.SQLiteDatabase; import android.database.sqlite.SQLiteDatabase.CursorFactory; import android.database.sqlite.SQLiteOpenHelper;public class MyOpenHelper extends SQLiteOpenHelper {public MyOpenHelper(Context context, String name, CursorFactory factory,int version) {super(context, name, factory, version);}//數(shù)據(jù)庫創(chuàng)建完之后,才會調(diào)用這個函數(shù)@Overridepublic void onCreate(SQLiteDatabase db) {System.out.println("數(shù)據(jù)庫創(chuàng)建");}@Overridepublic void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {System.out.println("數(shù)據(jù)庫升級");} }運行結(jié)果:
首先,刪除創(chuàng)建好的數(shù)據(jù)庫,重新創(chuàng)建數(shù)據(jù)庫。
可以看到onCreate方法被調(diào)用。
然后,升級數(shù)據(jù)庫版本。
可以看到onUpgrade方法被調(diào)用。
示例:創(chuàng)建表
src/cn.itcast.sqlite/MyOpenHelper.java
src/cn.itcast.sqlite/Test.java
package cn.itcast.sqlite;import android.database.sqlite.SQLiteDatabase; import android.test.AndroidTestCase;public class Test extends AndroidTestCase {public void createDataBase(){MyOpenHelper oh = new MyOpenHelper(getContext(), "people.db", null, 1);SQLiteDatabase db = oh.getWritableDatabase();} }測試運行結(jié)果:
首先,刪除數(shù)據(jù)庫,重新創(chuàng)建數(shù)據(jù)庫,創(chuàng)建表。然后導(dǎo)出people.db,拖入SQLite Expert,刷新。可以看到,生成了新的person表。
表格中的RecNo是不存在的字段,只是為了讓開發(fā)人員能夠方便地看到記錄行數(shù)。
3.2 使用SQL語句插入刪除
示例:src/cn.itcast.sqlite/Test.java
package cn.itcast.sqlite;import android.database.sqlite.SQLiteDatabase; import android.test.AndroidTestCase;public class Test extends AndroidTestCase {private MyOpenHelper oh;private SQLiteDatabase db;//測試框架初始化完畢后,測試方法執(zhí)行前,此方法調(diào)用@Overrideprotected void setUp() throws Exception {super.setUp();oh = new MyOpenHelper(getContext(), "people.db", null, 1);db = oh.getWritableDatabase();}public void createDataBase(){MyOpenHelper oh = new MyOpenHelper(getContext(), "people.db", null, 1);SQLiteDatabase db = oh.getWritableDatabase();}public void insert(){db.execSQL("insert into person(name,phone,salary) values(?,?,?)",new Object[]{"云鶴",138438,"13000"});db.execSQL("insert into person(name,phone,salary) values(?,?,?)",new Object[]{"云鶴兒子",138438,"13000"});db.execSQL("insert into person(name,phone,salary) values(?,?,?)",new Object[]{"云鶴孫子",138438,"13000"});db.execSQL("insert into person(name,phone,salary) values(?,?,?)",new Object[]{"云鶴曾孫",138438,"13000"});db.execSQL("insert into person(name,phone,salary) values(?,?,?)",new Object[]{"王亞聰",138438,"13000"});}public void delete(){db.execSQL("delete from person where name = ?",new Object[]{"王亞聰"});}//測試方法執(zhí)行完畢后,執(zhí)行。@Overrideprotected void tearDown() throws Exception {super.tearDown();db.close();} }測試運行結(jié)果:
首先,測試insert方法,插入數(shù)據(jù),然后導(dǎo)出people.db,拖入SQLite Expert,刷新。可以看到,插入數(shù)據(jù)成功。
然后,測試delete方法,刪除一條數(shù)據(jù),導(dǎo)出people.db,拖入SQLite Expert,刷新。可以看到,刪除數(shù)據(jù)成功。
如果將代碼調(diào)整,如下圖,測試insert方法或delete方法,就會報空指針異常。
原因分析:測試框架需要先被系統(tǒng)創(chuàng)建出來,然后再對它做各個參數(shù)的初始化,所有初始化做完之后,虛擬上下文才會存在。但是,private MyOpenHelper oh = new MyOpenHelper(getContext(), “people.db”, null, 1);這條語句是在測試框架構(gòu)造方法被調(diào)用之前調(diào)用的,因此,虛擬上下文(getContext方法返回值)根本不存在,獲取不到。所以,MyOpenHelper根本new不出來。如此,才會報空指針異常。但是,setUp方法是在測試框架初始化完畢后,測試方法執(zhí)行前,此方法才被調(diào)用。因此,虛擬上下文已經(jīng)存在,沒有任何問題。
3.3 使用SQL語句修改查詢
示例:src/cn.itcast.sqlite/Test.java
package cn.itcast.sqlite;import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import android.test.AndroidTestCase;public class Test extends AndroidTestCase {private MyOpenHelper oh;private SQLiteDatabase db;@Overrideprotected void setUp() throws Exception {super.setUp();oh = new MyOpenHelper(getContext(), "people.db", null, 1);db = oh.getWritableDatabase();}public void update(){db.execSQL("update person set salary = ? where name = ?",new Object[]{10000,"云鶴"});}public void select(){Cursor cursor = db.rawQuery("select name,salary from person where name = ?", new String[]{"云鶴"});//從游標(biāo)中取出數(shù)據(jù),方法與結(jié)果集類似while(cursor.moveToNext()){//只能傳入索引值,索引值根據(jù)sql語句中查詢內(nèi)容所在的順序而定,例如,上面的sql語句,name在第一位,索引即為0String name = cursor.getString(0);//可以通過字段名稱獲取列索引,再通過列索引獲取內(nèi)容String salary = cursor.getString(cursor.getColumnIndex("salary"));System.out.println(name + ":" + salary);}}@Overrideprotected void tearDown() throws Exception {super.tearDown();db.close();} }測試運行結(jié)果:
測試update方法,更新一條數(shù)據(jù),導(dǎo)出people.db,拖入SQLite Expert,刷新。可以看到,更新數(shù)據(jù)成功。
測試select方法,查詢數(shù)據(jù),打印出來。可以看到,查詢成功。
3.4 使用API完成插入
使用SQL語句不太方便,可以使用API完成增刪改查操作。
示例:src/cn.itcast.sqlite/Test.java
package cn.itcast.sqlite;import android.content.ContentValues; import android.database.sqlite.SQLiteDatabase; import android.test.AndroidTestCase;public class Test extends AndroidTestCase {private MyOpenHelper oh;private SQLiteDatabase db;@Overrideprotected void setUp() throws Exception {super.setUp();oh = new MyOpenHelper(getContext(), "people.db", null, 1);db = oh.getWritableDatabase();}public void insertApi(){ContentValues values = new ContentValues();values.put("name","亞聰和云鶴");values.put("phone","138888");values.put("salary","99999999");//第一個參數(shù):表名//第二個參數(shù):nullColumnHack,當(dāng)?shù)谌齻€參數(shù)ContentValues對象為null,或者里面沒有任何數(shù)據(jù)時,第二個參數(shù)才會有效。開發(fā)中,ContentValues都是有值的,所以直接傳入null即可//第三個參數(shù):通過鍵值對把要插入的數(shù)據(jù)封裝至ContentValues對象中,鍵名必須是字段名//db.insert返回值為行id,如果為-1,表示插入失敗long l1 = db.insert("person", null, values);//ContentValues下次使用的時候記得清空values.clear();values.put("name","亞聰和云鶴的兒子");values.put("phone","138888");values.put("salary","99999999");long l2 = db.insert("person", null, values);System.out.println(l1 + ":" + l2);}@Overrideprotected void tearDown() throws Exception {super.tearDown();db.close();} }測試運行結(jié)果:
測試insertApi方法,插入一條數(shù)據(jù),導(dǎo)出people.db,拖入SQLite Expert,刷新。可以看到,插入數(shù)據(jù)成功。
3.5 使用API完成刪改查
示例:src/cn.itcast.sqlite/Test.java
package cn.itcast.sqlite;import android.content.ContentValues; import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import android.test.AndroidTestCase;public class Test extends AndroidTestCase {private MyOpenHelper oh;private SQLiteDatabase db;@Overrideprotected void setUp() throws Exception {super.setUp();oh = new MyOpenHelper(getContext(), "people.db", null, 1);db = oh.getWritableDatabase();}public void deleteApi(){//返回被刪除的行的數(shù)量int i = db.delete("person", "name = ?", new String[]{"云鶴兒子"});System.out.println(i);}public void updateApi(){ContentValues values = new ContentValues();values.put("salary",10500);int i = db.update("person", values, "name = ?", new String[]{"云鶴"});System.out.println(i);}public void selectApi(){//第二個參數(shù)如果傳入null,表示查詢所有字段Cursor cursor = db.query("person", null, null, null, null, null, null, null);while(cursor.moveToNext()){String name = cursor.getString(1);String phone = cursor.getString(2);String salary = cursor.getString(3);System.out.println(name + ":" + phone + ":" + salary);}}@Overrideprotected void tearDown() throws Exception {super.tearDown();db.close();} }測試運行結(jié)果:
測試deleteApi方法,刪除一條數(shù)據(jù),導(dǎo)出people.db,拖入SQLite Expert,刷新。可以看到,刪除數(shù)據(jù)成功。
測試updateApi方法,更新一條數(shù)據(jù),導(dǎo)出people.db,拖入SQLite Expert,刷新。可以看到,更新數(shù)據(jù)成功。
測試selectApi方法,查詢數(shù)據(jù),打印出來。可以看到,查詢成功。
3.6 SQLite的事務(wù)
示例:src/cn.itcast.sqlite/Test.java
package cn.itcast.sqlite;import android.content.ContentValues; import android.database.sqlite.SQLiteDatabase; import android.test.AndroidTestCase;public class Test extends AndroidTestCase {private MyOpenHelper oh;private SQLiteDatabase db;@Overrideprotected void setUp() throws Exception {super.setUp();oh = new MyOpenHelper(getContext(), "people.db", null, 1);db = oh.getWritableDatabase();}public void transaction(){try{//開啟事務(wù)db.beginTransaction();ContentValues values = new ContentValues();values.put("salary", 11000);db.update("person", values, "name = ?", new String[]{"云鶴"});values.put("salary", 12500);db.update("person", values, "name = ?", new String[]{"云鶴孫子"});//如果下面這行代碼執(zhí)行,說明事務(wù)執(zhí)行成功db.setTransactionSuccessful();}catch(Exception e){e.printStackTrace();}finally{//關(guān)閉事務(wù),這時候就提交了,不用再次提交//關(guān)閉時候的時候,如果發(fā)現(xiàn)db.setTransactionSuccessful();語句未執(zhí)行,那么就會回滾事務(wù)db.endTransaction();}}@Overrideprotected void tearDown() throws Exception {super.tearDown();db.close();} }測試運行結(jié)果;
測試transaction方法導(dǎo)出people.db,拖入SQLite Expert,刷新。可以看到,事務(wù)執(zhí)行成功。
如果對代碼作如下修改,事務(wù)執(zhí)行中出現(xiàn)異常。
可以看到,報異常,并且數(shù)據(jù)未發(fā)生任何變化。
3.7 把數(shù)據(jù)庫中的數(shù)據(jù)顯示至屏幕
src/cn.itcast.showdata/MyOpenHelper.java
package cn.itcast.showdata;import android.content.Context; import android.database.sqlite.SQLiteDatabase; import android.database.sqlite.SQLiteOpenHelper;public class MyOpenHelper extends SQLiteOpenHelper {public MyOpenHelper(Context context) {super(context, "people.db", null, 1);}@Overridepublic void onCreate(SQLiteDatabase db) {db.execSQL("create table person(_id integer primary key autoincrement,name char(10),phone char(20),salary integer(10))");}@Overridepublic void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {System.out.println("數(shù)據(jù)庫升級");} }AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android"package="cn.itcast.showdata"android:versionCode="1"android:versionName="1.0" ><uses-sdk android:minSdkVersion="8"android:targetSdkVersion="17" /><instrumentation android:name="android.test.InstrumentationTestRunner"android:targetPackage="cn.itcast.showdata"></instrumentation><application android:allowBackup="true"android:icon="@drawable/ic_launcher"android:label="@string/app_name"android:theme="@style/AppTheme" ><uses-library android:name="android.test.runner"/><activity android:name="cn.itcast.showdata.MainActivity"android:label="@string/app_name" ><intent-filter><action android:name="android.intent.action.MAIN" /><category android:name="android.intent.category.LAUNCHER" /></intent-filter></activity></application></manifest>src/cn.itcast.showdata/Test.java
package cn.itcast.showdata;import android.content.ContentValues; import android.database.sqlite.SQLiteDatabase; import android.test.AndroidTestCase;public class Test extends AndroidTestCase {private MyOpenHelper oh;private SQLiteDatabase db;@Overrideprotected void setUp() throws Exception {super.setUp();oh = new MyOpenHelper(getContext());db = oh.getWritableDatabase();}public void insertApi(){for(int i = 0; i < 50; i++){ContentValues values = new ContentValues();values.put("name","張" + i);values.put("phone","138" + i + i);values.put("salary","200" + i);db.insert("person", null, values);}}@Overrideprotected void tearDown() throws Exception {super.tearDown();db.close();} }src/cn.itcast.showdata.domain/Person.java
package cn.itcast.showdata.domain;public class Person {private int _id;private String name;private String phone;private int salary;public Person(int _id, String name, String phone, int salary) {super();this._id = _id;this.name = name;this.phone = phone;this.salary = salary;}public int get_id() {return _id;}public void set_id(int _id) {this._id = _id;}public String getName() {return name;}public void setName(String name) {this.name = name;}public String getPhone() {return phone;}public void setPhone(String phone) {this.phone = phone;}public int getSalary() {return salary;}public void setSalary(int salary) {this.salary = salary;}@Overridepublic String toString() {return "name=" + name + ", phone=" + phone+ ", salary=" + salary;}}res\layout\activity_main.xml
<ScrollView android:layout_width="match_parent"android:layout_height="match_parent"xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"><LinearLayout android:id="@+id/ll"android:layout_width="match_parent"android:layout_height="match_parent"tools:context=".MainActivity"android:orientation="vertical"></LinearLayout> </ScrollView>src/cn.itcast.showdata/MainActivity.java
package cn.itcast.showdata;import java.util.ArrayList; import java.util.List;import android.app.Activity; import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import android.os.Bundle; import android.widget.LinearLayout; import android.widget.TextView; import cn.itcast.sqlite.domain.Person;public class MainActivity extends Activity {List<Person> personList;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);personList = new ArrayList<Person>();//讀取數(shù)據(jù)庫MyOpenHelper oh = new MyOpenHelper(this);SQLiteDatabase db = oh.getWritableDatabase();Cursor cursor = db.query("person", null, null, null, null, null, null, null);while(cursor.moveToNext()){int _id = cursor.getInt(0);String name = cursor.getString(1);String phone = cursor.getString(2);int salary = cursor.getInt(3);Person p = new Person(_id, name, phone, salary);personList.add(p);}//獲取線性布局LinearLayout ll = (LinearLayout) findViewById(R.id.ll);for(Person p : personList){TextView tv = new TextView(this);tv.setText(p.toString());tv.setTextSize(16);//把tv設(shè)置為線性布局的子節(jié)點ll.addView(tv);}} }運行結(jié)果:
首先,在模擬器上運行應(yīng)用程序,在data/data目錄下創(chuàng)建應(yīng)用程序文件目錄。運行Test類中的insertApi方法,導(dǎo)出people.db文件,拖入SQLite Expert,可以看到插入數(shù)據(jù)成功。
然后,再次在模擬器上運行應(yīng)用程序,將數(shù)據(jù)庫中的數(shù)據(jù)通過TextView的形式展示在屏幕上。
使用ScrollView包裹LinearLayout,ScrollView表示可以上下滑動的View。如果線性布局過大,超出屏幕顯示范圍,那么就可以上下滑動了。
4. 使用ListView顯示數(shù)據(jù)
開發(fā)中不會按照如上方法顯示數(shù)據(jù)到屏幕上的。因為,如果待顯示數(shù)據(jù)太多,程序可能會崩潰。原因在于循環(huán)提取數(shù)據(jù),如果有10000條數(shù)據(jù),就需要創(chuàng)建10000個JavaBean對象和10000個TextView對象到內(nèi)存中,太耗內(nèi)存。解決方案之一是采用分頁查詢,只查詢一部分?jǐn)?shù)據(jù),通過SQL語句的limit控制讀取的條數(shù)。另一個Android方式的解決方案是動態(tài)創(chuàng)建TextView,也就是說屏幕只能顯示10條數(shù)據(jù),那么就創(chuàng)建10個JavaBean對象和10個TextView對象。如果用戶下滑屏幕顯示數(shù)據(jù),一方面緩存(內(nèi)存不足的時候會被摧毀,內(nèi)存充足的時候緩存,避免下次再顯示的時候重新創(chuàng)建,提升性能)屏幕上方已經(jīng)消失的TextView,一方面創(chuàng)建屏幕下方要顯示的TextView。這樣,內(nèi)容中只有10個JavaBean對象及10個TextView,避免內(nèi)存溢出。Android中用ListView實現(xiàn)此功能。
ListView專門用于顯示列表,每一行稱為一個條目(Item)
MVC
- M:模型層 Javabean personList
- V:視圖層 jsp ListView
- C:控制層 servlet Adapter(Adapter是適配器,用適配器控制要顯示哪些內(nèi)容)
ListAdapter是一個接口,需要實現(xiàn)的方法太多。因此,一般通過繼承BaseAdapter(抽象類)的方式創(chuàng)建ListAdapter對象。
所有在布局文件中定義的組件(包括LinearLayout),都可以在ListView中顯示。
示例:修改上個示例中的代碼如下:res\layout\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"android:orientation="vertical"><ListView android:id="@+id/lv"android:layout_width="match_parent"android:layout_height="match_parent"></ListView></LinearLayout>src/cn.itcast.listView/MainActivity.java
package cn.itcast.listView;import java.util.ArrayList; import java.util.List;import android.app.Activity; import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import android.os.Bundle; import android.view.View; import android.view.ViewGroup; import android.widget.BaseAdapter; import android.widget.ListView; import android.widget.TextView; import cn.itcast.listView.domain.Person;public class MainActivity extends Activity {List<Person> personList;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);personList = new ArrayList<Person>();//讀取數(shù)據(jù)庫MyOpenHelper oh = new MyOpenHelper(this);SQLiteDatabase db = oh.getWritableDatabase();Cursor cursor = db.query("person", null, null, null, null, null, null, null);while(cursor.moveToNext()){int _id = cursor.getInt(0);String name = cursor.getString(1);String phone = cursor.getString(2);int salary = cursor.getInt(3);Person p = new Person(_id, name, phone, salary);personList.add(p);}ListView lv = (ListView)findViewById(R.id.lv);lv.setAdapter(new MyAdapter());}class MyAdapter extends BaseAdapter{//獲取集合的元素數(shù)量@Overridepublic int getCount() {return personList.size();}//系統(tǒng)調(diào)用此方法,獲取一個View對象,該View對象會作為一個條目顯示至ListView中//position:getView返回的View對象會作為ListView的第幾個條目顯示,那么position的值就是多少@Overridepublic View getView(int position, View convertView, ViewGroup parent) {System.out.println("getView調(diào)用" + position);Person p = personList.get(position);TextView tv = new TextView(MainActivity.this);tv.setText(p.toString());tv.setTextSize(16);return tv;}@Overridepublic Object getItem(int position) {return null;}@Overridepublic long getItemId(int position) {return 0;}} }運行結(jié)果:
如果將屏幕向下滑動,結(jié)果如下:
如果將屏幕向上滑動,結(jié)果如下:
4.1 把布局文件填充成View對象
如果想要修改ListView的展示樣式,那么就需要將指定的布局文件填充為View。
示例:修改上個示例中的代碼,如下:res\layout\item_listview.xml
<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="wrap_content"><TextView android:id="@+id/tv_name"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="name"android:textSize="22sp"/><LinearLayout android:layout_width="wrap_content"android:layout_height="wrap_content"android:orientation="vertical"android:layout_alignParentRight="true"android:layout_centerVertical="true"><TextView android:id="@+id/tv_phone"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="phone" /><TextView android:id="@+id/tv_salary"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="salary" /></LinearLayout></RelativeLayout>src/cn.itcast.listView/MainActivity.java
package cn.itcast.listView;import java.util.ArrayList; import java.util.List;import android.app.Activity; import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import android.os.Bundle; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.BaseAdapter; import android.widget.ListView; import android.widget.TextView; import cn.itcast.listView.domain.Person;public class MainActivity extends Activity {List<Person> personList;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);personList = new ArrayList<Person>();MyOpenHelper oh = new MyOpenHelper(this);SQLiteDatabase db = oh.getWritableDatabase();Cursor cursor = db.query("person", null, null, null, null, null, null, null);while(cursor.moveToNext()){int _id = cursor.getInt(0);String name = cursor.getString(1);String phone = cursor.getString(2);int salary = cursor.getInt(3);Person p = new Person(_id, name, phone, salary);personList.add(p);}ListView lv = (ListView)findViewById(R.id.lv);lv.setAdapter(new MyAdapter());}class MyAdapter extends BaseAdapter{@Overridepublic int getCount() {return personList.size();}@Overridepublic View getView(int position, View convertView, ViewGroup parent) {System.out.println("getView調(diào)用" + position);Person p = personList.get(position);//把指定布局文件填充成View對象//第二個參數(shù)指定把哪個布局文件轉(zhuǎn)換成View對象,變成ListView條目顯示在屏幕上//第三個參數(shù)給布局文件指定父節(jié)點,不需要,設(shè)置nullView v = View.inflate(MainActivity.this, R.layout.item_listview, null);//第二種方式,不常用//LayoutInflater inflater = LayoutInflater.from(MainActivity.this);//View v = inflater.inflate(R.layout.item_listview, null);//第三種方式,不常用//LayoutInflater inflater = (LayoutInflater) getSystemService(LAYOUT_INFLATER_SERVICE);//View v = inflater.inflate(R.layout.item_listview, null);//修改View對象中各個組件的內(nèi)容//findViewById(R.id.tv_name);通過id尋找tv_name默認(rèn)是在activity_mian.xml布局文件中查找(由于本類中onCreate方法有一條語句setContentView(R.layout.activity_main);)。//v.findViewById(R.id.tv_name);就會在被填充的item_listview布局文件中查找tv_name。TextView tv_name = (TextView)v.findViewById(R.id.tv_name);tv_name.setText(p.getName());TextView tv_phone = (TextView)v.findViewById(R.id.tv_phone);tv_phone.setText(p.getPhone());TextView tv_salary = (TextView)v.findViewById(R.id.tv_salary);//參數(shù)一定要為字符串,如果是整數(shù),那么就會把它作為一個id,去尋找相應(yīng)的資源tv_salary.setText(p.getSalary() + "");return v;}//下面兩個方法,系統(tǒng)不會調(diào)用,程序員可以調(diào)用//返回條目,一般返回某個元素,例如:personList.get(position);@Overridepublic Object getItem(int position) {return null;}//返回條目的id,一般返回元素的主鍵@Overridepublic long getItemId(int position) {return 0;}} }運行結(jié)果:
4.2 ListView的優(yōu)化
上面的代碼,只要用戶一直上下滑屏,就會不斷填充新的View對象(View v = View.inflate(MainActivity.this, R.layout.item_listview, null);),一方面耗費內(nèi)存(內(nèi)存可能會溢出)、CPU,另一方面導(dǎo)致加載時間變長。
可以使用listView的緩存機制解決這個問題。任何一個條目如果被滑出屏幕,那么就緩存起來,下次再顯示,不用重新創(chuàng)建,直接從緩存中提取就好。ListView對條目的緩存。在顯示新的條目時,只要內(nèi)存中有緩存,就會拿來用,不管是哪個條目的緩存。
示例:修改上面的示例代碼,如下:src/cn.itcast.listView/MainActivity.java
package cn.itcast.listView;import java.util.ArrayList; import java.util.List;import android.app.Activity; import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import android.os.Bundle; import android.view.View; import android.view.ViewGroup; import android.widget.BaseAdapter; import android.widget.ListView; import android.widget.TextView; import cn.itcast.listView.domain.Person;public class MainActivity extends Activity {List<Person> personList;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);personList = new ArrayList<Person>();MyOpenHelper oh = new MyOpenHelper(this);SQLiteDatabase db = oh.getWritableDatabase();Cursor cursor = db.query("person", null, null, null, null, null, null, null);while(cursor.moveToNext()){int _id = cursor.getInt(0);String name = cursor.getString(1);String phone = cursor.getString(2);int salary = cursor.getInt(3);Person p = new Person(_id, name, phone, salary);personList.add(p);}ListView lv = (ListView)findViewById(R.id.lv);lv.setAdapter(new MyAdapter());}class MyAdapter extends BaseAdapter{@Overridepublic int getCount() {return personList.size();}//convertView就是被緩存起來的View對象,系統(tǒng)每次調(diào)用getView方法,都會把convertView傳遞進(jìn)來@Overridepublic View getView(int position, View convertView, ViewGroup parent) {System.out.println("getView調(diào)用" + position);Person p = personList.get(position);View v = null;System.out.println(convertView);if(convertView == null){//如果沒有緩存,填充新的View對象v = View.inflate(MainActivity.this, R.layout.item_listview, null);}else{//如果有緩存,復(fù)用緩存v = convertView;}//緩存并不是指給每個單獨的條目緩存,系統(tǒng)是只要系統(tǒng)中有對象被緩存就會拿來使用。//所以緩存的是View對象,并不是內(nèi)容。也就是說系統(tǒng)中緩存的對象個數(shù)等于同屏能夠顯示的條目+1個。TextView tv_name = (TextView)v.findViewById(R.id.tv_name);tv_name.setText(p.getName());TextView tv_phone = (TextView)v.findViewById(R.id.tv_phone);tv_phone.setText(p.getPhone());TextView tv_salary = (TextView)v.findViewById(R.id.tv_salary);tv_salary.setText(p.getSalary() + "");return v;}@Overridepublic Object getItem(int position) {return null;}@Overridepublic long getItemId(int position) {return 0;}} }運行結(jié)果:
可以看到,緩存了11個View對象之后,后來不會再創(chuàng)建新的的View對象了,而都是通過緩存獲取的。
4.3 ArrayAdapter和SimpleAdapter
ListAdapter,我們用的最多的是BaseAdapter,因為它的四個方法是我們自己實現(xiàn)的,自由度比較大,做復(fù)雜列表的時候比較容易。但列表不是那么復(fù)雜的時候,ListAdapter有一些封裝度更高的Adapter提供給我們使用,用起來會更方便,其中的getCount、getView都不需要我們自己寫代碼。
示例1:res\layout\activity_main.xml
<?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" ><ImageView android:id="@+id/iv"android:layout_width="40dp"android:layout_height="40dp"android:src="@drawable/ic_launcher"/><TextView android:id="@+id/tv"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="name"android:textSize="25sp"android:layout_gravity="center_vertical"/></LinearLayout>src/cn.itcast.simplearray/MainActivity.java
package cn.itcast.simplearray;import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map;import android.app.Activity; import android.os.Bundle; import android.widget.ArrayAdapter; import android.widget.ListView; import android.widget.SimpleAdapter;public class MainActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);String[] objects = new String[]{"白吃","沙比","亞聰"};ListView lv = (ListView)findViewById(R.id.lv);//使用ArrayAdapter只能處理文本數(shù)據(jù),圖片數(shù)據(jù)無法個性化單獨設(shè)置,所以如果只是簡單的文本顯示,就是用ArrayAdapter。//第二個參數(shù)為資源文件,填充ListView條目的布局文件//第三個參數(shù)指定文本顯示至哪一個textView中lv.setAdapter(new ArrayAdapter<String>(this, R.layout.item_listview, R.id.tv, objects));} }運行結(jié)果:
示例2:src/cn.itcast.simplearray/MainActivity.java
package cn.itcast.simplearray;import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map;import android.app.Activity; import android.os.Bundle; import android.widget.ArrayAdapter; import android.widget.ListView; import android.widget.SimpleAdapter;public class MainActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);ListView lv = (ListView)findViewById(R.id.lv);//集合中的每一個元素都是一個條目要顯示的數(shù)據(jù),因為有多種數(shù)據(jù)類型,// 所以List不能使用單一泛型,先把所有類型的數(shù)據(jù)都封裝至map,然后把map存入List。List<Map<String,Object>> data = new ArrayList<Map<String,Object>>();Map<String,Object> map1= new HashMap<String,Object>();map1.put("image", R.drawable.ic_launcher);map1.put("name", "白吃");data.add(map1);Map<String,Object> map2= new HashMap<String,Object>();map2.put("image", R.drawable.photo1);map2.put("name", "沙比");data.add(map2);Map<String,Object> map3= new HashMap<String,Object>();map3.put("image", R.drawable.photo2);map3.put("name", "亞聰");data.add(map3);lv.setAdapter(new SimpleAdapter(this, data, R.layout.item_listview,new String[]{"image","name"}, new int[]{R.id.iv,R.id.tv}));} }運行結(jié)果:
案例1:SQLite數(shù)據(jù)庫
public class MyOpenHelper extends SQLiteOpenHelper {public MyOpenHelper(Context context, String name, CursorFactory factory,int version) {super(context, name, factory, version);// TODO Auto-generated constructor stub}//數(shù)據(jù)庫創(chuàng)建時,此方法會調(diào)用@Overridepublic void onCreate(SQLiteDatabase db) {db.execSQL("create table person(_id integer primary key autoincrement, name char(10), salary char(20), phone integer(20))");}//數(shù)據(jù)庫升級時,此方法會調(diào)用@Overridepublic void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {System.out.println("數(shù)據(jù)庫升級了");} } public class TestCase extends AndroidTestCase {//此時測試框架還沒有初始化完畢,沒有虛擬上下文對象 // private MyOpenHelper oh = new MyOpenHelper(getContext(), "people.db", null, 1);private MyOpenHelper oh;private SQLiteDatabase db;public void test(){//getContext():獲取一個虛擬的上下文MyOpenHelper oh = new MyOpenHelper(getContext(), "people.db", null, 1);//如果數(shù)據(jù)庫不存在,先創(chuàng)建數(shù)據(jù)庫,再獲取可讀可寫的數(shù)據(jù)庫對象,如果數(shù)據(jù)庫存在,就直接打開SQLiteDatabase db = oh.getWritableDatabase();//如果存儲空間滿了,那么返回只讀數(shù)據(jù)庫對象 // SQLiteDatabase db = oh.getReadableDatabase();}//測試框架初始化完畢之后,在測試方法執(zhí)行之前,此方法調(diào)用@Overrideprotected void setUp() throws Exception {super.setUp();oh = new MyOpenHelper(getContext(), "people.db", null, 1);db = oh.getWritableDatabase();}//測試方法執(zhí)行完畢之后,此方法調(diào)用@Overrideprotected void tearDown() throws Exception {// TODO Auto-generated method stubsuper.tearDown();db.close();}public void insert(){// db.execSQL("insert into person (name, salary, phone)values(?, ?, ?)", new Object[]{"小志的老婆[1]", "13000", 138438}); // db.execSQL("insert into person (name, salary, phone)values(?, ?, ?)", new Object[]{"小志的兒子", 14000, "13888"});db.execSQL("insert into person (name, salary, phone)values(?, ?, ?)", new Object[]{"小志", 14000, "13888"});}public void delete(){db.execSQL("delete from person where name = ?", new Object[]{"小志"});}public void update(){db.execSQL("update person set phone = ? where name = ?", new Object[]{186666, "小志的兒子"});}public void select(){Cursor cursor = db.rawQuery("select name, salary from person", null);while(cursor.moveToNext()){//通過列索引獲取列的值String name = cursor.getString(cursor.getColumnIndex("name"));String salary = cursor.getString(1);System.out.println(name + ";" + salary);}}public void insertApi(){//把要插入的數(shù)據(jù)全部封裝至ContentValues對象ContentValues values = new ContentValues();values.put("name", "游天龍");values.put("phone", "15999");values.put("salary", 16000);db.insert("person", null, values);}public void deleteApi(){int i = db.delete("person", "name = ? and _id = ?", new String[]{"小志的兒子", "3"});System.out.println(i);}public void updateApi(){ContentValues values = new ContentValues();values.put("salary", 26000);int i = db.update("person", values, "name = ?", new String[]{"游天龍"});System.out.println(i);}public void selectApi(){Cursor cursor = db.query("person", null, null, null, null, null, null, null);while(cursor.moveToNext()){String name = cursor.getString(cursor.getColumnIndex("name"));String phone = cursor.getString(cursor.getColumnIndex("phone"));String salary = cursor.getString(cursor.getColumnIndex("salary"));System.out.println(name + ";" + phone + ";" + salary);}}public void transaction(){try{//開啟事務(wù)db.beginTransaction();ContentValues values = new ContentValues();values.put("salary", 12000);db.update("person", values, "name = ?", new String[]{"小志"});values.clear();values.put("salary", 16000);db.update("person", values, "name = ?", new String[]{"小志的兒子"});int i = 3/0;//設(shè)置 事務(wù)執(zhí)行成功db.setTransactionSuccessful();}finally{//關(guān)閉事務(wù),同時提交,如果已經(jīng)設(shè)置事務(wù)執(zhí)行成功,那么sql語句就生效了,反之,sql語句回滾db.endTransaction();}} }案例2:把數(shù)據(jù)顯示至屏幕
public class MainActivity extends Activity {List<Person> personList;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);personList = new ArrayList<Person>();//把數(shù)據(jù)庫的數(shù)據(jù)查詢出來MyOpenHelper oh = new MyOpenHelper(this);SQLiteDatabase db = oh.getWritableDatabase();Cursor cursor = db.query("person", null, null, null, null, null, null, null);while(cursor.moveToNext()){String _id = cursor.getString(0);String name = cursor.getString(1);String salary = cursor.getString(2);String phone = cursor.getString(3);Person p = new Person(_id, name, phone, salary);personList.add(p);}LinearLayout ll = (LinearLayout) findViewById(R.id.ll);//把數(shù)據(jù)顯示至屏幕for (Person p : personList) {//1.集合中每有一條元素,就new一個textViewTextView tv = new TextView(this);//2.把人物的信息設(shè)置為文本框的內(nèi)容tv.setText(p.toString());tv.setTextSize(18);//3.把textView設(shè)置為線性布局的子節(jié)點ll.addView(tv);}}}案例3:使用ListView把數(shù)據(jù)顯示至屏幕
public class MainActivity extends Activity {List<Person> personList;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);personList = new ArrayList<Person>();//把數(shù)據(jù)庫的數(shù)據(jù)查詢出來MyOpenHelper oh = new MyOpenHelper(this);SQLiteDatabase db = oh.getWritableDatabase();Cursor cursor = db.query("person", null, null, null, null, null, null, null);while(cursor.moveToNext()){String _id = cursor.getString(0);String name = cursor.getString(1);String salary = cursor.getString(2);String phone = cursor.getString(3);Person p = new Person(_id, name, phone, salary);personList.add(p);}ListView lv = (ListView) findViewById(R.id.lv);lv.setAdapter(new MyAdapter());}class MyAdapter extends BaseAdapter{//系統(tǒng)調(diào)用,用來獲知集合中有多少條元素@Overridepublic int getCount() {return personList.size();}//由系統(tǒng)調(diào)用,獲取一個View對象,作為ListView的條目//position:本次getView方法調(diào)用所返回的View對象,在listView中是處于第幾個條目,那么position的值就是多少@Overridepublic View getView(int position, View convertView, ViewGroup parent) {Person p = personList.get(position);// TextView tv = new TextView(MainActivity.this);System.out.println("getView調(diào)用:" + position + ";" + convertView); // tv.setText(p.toString()); // tv.setTextSize(18);View v = null;//判斷條目是否有緩存if(convertView == null){//把布局文件填充成一個View對象v = View.inflate(MainActivity.this, R.layout.item_listview, null);}else{v = convertView;}//獲取布局填充器對象 // LayoutInflater inflater = LayoutInflater.from(MainActivity.this); // 使用布局填充器填充布局文件 // View v2 = inflater.inflate(R.layout.item_listview, null);// LayoutInflater inflater2 = (LayoutInflater) getSystemService(LAYOUT_INFLATER_SERVICE); // View v3 = inflater2.inflate(R.layout.item_listview, null);//通過資源id查找組件,注意調(diào)用的是View對象的findViewByIdTextView tv_name = (TextView) v.findViewById(R.id.tv_name);tv_name.setText(p.getName());TextView tv_phone = (TextView) v.findViewById(R.id.tv_phone);tv_phone.setText(p.getPhone());TextView tv_salary = (TextView) v.findViewById(R.id.tv_salary);tv_salary.setText(p.getSalary());return v;}@Overridepublic Object getItem(int position) {return null;}@Overridepublic long getItemId(int position) {return 0;}} }案例4:ArrayAdapter&SimpleAdapter
public class MainActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);String[] objects = new String[]{"小志","小志的兒子","萌萌"};ListView lv = (ListView) findViewById(R.id.lv); // lv.setAdapter(new ArrayAdapter<String>(this, R.layout.item_listview, R.id.tv_name, objects));//集合中每個元素都包含ListView條目需要的所有數(shù)據(jù),該案例中每個條目需要一個字符串和一個整型,所以使用一個map來封裝這兩種數(shù)據(jù)List<Map<String, Object>> data = new ArrayList<Map<String,Object>>();Map<String, Object> map1 = new HashMap<String, Object>();map1.put("photo", R.drawable.photo1);map1.put("name", "小志的兒子");data.add(map1);Map<String, Object> map2 = new HashMap<String, Object>();map2.put("photo", R.drawable.photo2);map2.put("name", "小志");data.add(map2);Map<String, Object> map3 = new HashMap<String, Object>();map3.put("photo", R.drawable.photo3);map3.put("name", "趙帥哥");data.add(map3);lv.setAdapter(new SimpleAdapter(this, data, R.layout.item_listview,new String[]{"photo", "name"}, new int[]{R.id.iv_photo, R.id.tv_name}));}}案例5:對話框
public class MainActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);}public void click1(View v){AlertDialog.Builder builder = new Builder(this);//設(shè)置圖標(biāo)builder.setIcon(android.R.drawable.alert_dark_frame);//設(shè)置標(biāo)題builder.setTitle("欲練此功必先自宮");//設(shè)置文本builder.setMessage("李志平,想清楚哦");//設(shè)置確定按鈕builder.setPositiveButton("確定", new OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {Toast.makeText(MainActivity.this, "感謝使用本軟件,再見", 0).show();}});//設(shè)置取消按鈕builder.setNegativeButton("取消", new OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {Toast.makeText(MainActivity.this, "若不自宮,一定不成功", 0).show();}});//使用創(chuàng)建器,生成一個對話框?qū)ο?/span>AlertDialog ad = builder.create();ad.show();}public void click2(View v){AlertDialog.Builder builder = new Builder(this);builder.setTitle("請選擇性別");final String[] items = new String[]{"男","女"};builder.setSingleChoiceItems(items, -1, new OnClickListener() {//which:用戶所選的條目的下標(biāo)//dialog:觸發(fā)這個方法的對話框@Overridepublic void onClick(DialogInterface dialog, int which) {Toast.makeText(MainActivity.this, "您選擇的是:" + items[which], 0).show();//關(guān)閉對話框dialog.dismiss();}});builder.show();}public void click3(View v){AlertDialog.Builder builder = new Builder(this);builder.setTitle("請選擇您覺得帥的人");final String[] items = new String[]{"侃哥","趙帥哥","趙老師","趙師兄"};final boolean[] checkedItems = new boolean[]{true,true,false,false};builder.setMultiChoiceItems(items, checkedItems, new OnMultiChoiceClickListener() {//which:用戶點擊的條目的下標(biāo)//isChecked:用戶是選中該條目還是取消該條目@Overridepublic void onClick(DialogInterface dialog, int which, boolean isChecked) {checkedItems[which] = isChecked;}});//設(shè)置一個確定按鈕builder.setPositiveButton("確定", new OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {String text = "";for(int i = 0; i < 4; i++){text += checkedItems[i]? items[i] + "," : "";}Toast.makeText(MainActivity.this, text, 0).show();dialog.dismiss();}});builder.show();} }總結(jié)
以上是生活随笔為你收集整理的Android应用开发:数据存储和界面展现-2的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Android应用开发:数据存储和界面展
- 下一篇: Android应用开发-快速入门