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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > Android >内容正文

Android

Android应用开发:数据存储和界面展现-2

發布時間:2025/4/16 Android 32 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Android应用开发:数据存储和界面展现-2 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

1. pull解析XML文件

Android推薦使用pull解析XML文件,與SAX解析XML文件類似,都是事件驅動類型的解析方式。

示例:獲取天氣信息

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系統默認編碼是“utf-8”try {xp.setInput(is,"utf-8");//開始解析//獲取事件類型,通過對事件類型的判斷,直到當前解析的是什么節點int type = xp.getEventType();City city = null;while(type != XmlPullParser.END_DOCUMENT){switch(type){case XmlPullParser.START_TAG://獲取當前節點的名字if("weather".equals(xp.getName())){cityList = new ArrayList<City>();}else if("city".equals(xp.getName())){city = new City();}else if("name".equals(xp.getName())){//獲取當前節點的下一個節點的文本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;}//把指針移動至下一個節點,并返回該節點的事件類型type = xp.next();}} catch (Exception e) {e.printStackTrace();}for(City c:cityList){System.out.println(c.toString());}} }

運行結果:

1.1 用debug查看pull解析流程

首先,雙擊代碼左側,打一個斷點。

點擊debug “01_pull解析XML文件”。

等待debug初始化,跳出對話框,不要點“Force close”!等待一會,對話框就會消失。

點擊“獲取天氣信息”按鈕。

點擊“yes”,程序進入debug模式。

通過“step over”即可以查看代碼執行的每一步。

2. 測試概念

  • 按崗位分:

    • 黑盒測試:測試業務邏輯
    • 白盒測試:測試邏輯方法
  • 按測試粒度分:

    • 方法測試 function
    • 單元測試 unit:多個方法集成一個單元,測試
    • 集成測試 intergration:多個單元集成,測試
    • 系統測試 system:服務器端、客戶端端聯動,測試整個系統
  • 按暴力程度分:

    • 冒煙測試 smoke
    • 壓力測試 pressure:針對服務器端的測試。

目前的智能手機都有測試框架,例如,Android自帶的monkey,不過需要在Android命令行中使用。

示例:隨機點擊模擬器一千次

進入測試狀態…

2.1 單元測試

創建一個類,繼承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);//斷言,對比實際結果與預期是否一致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;} }

使用測試框架需要在清單文件中配置指令集和使用類庫。

兩種方式啟動測試:方法一:在代碼區內選擇需要測試的方法–>右擊Run As–>Android JUnit Test。

方法二:在Outline內選擇需要測試的方法–>右擊Run As–>Android JUnit Test。

通過下圖,可以看到,報錯,結果與斷言不相符。

修改Tools.java中代碼。

通過下圖,可以看到,測試運行成功,結果與斷言相符。

通過Console可以看到測試過程。

雖然,測試過程中,Android項目并沒有啟動,整個測試過程可以看到模擬器上根本沒有前臺界面。但是,測試需要模擬器運行測試框架。因為,測試的項目是Android代碼,必須運行在Android設備上,可以是模擬器也可以是真機。沒有返回值的方法測試步驟相同,只是不需要斷言。

測試后,報錯,提示出現異常。

修改代碼,測試成功。

3. 創建數據庫

Android中存儲數據用的最多的還是數據庫,SQLite(Android自帶的輕量級數據庫)。
數據庫保存在內部存儲空間。所以,在備份數據的時候,不會寫在數據庫里。一旦應用刪除,數據庫就沒了,備份的數據也就全沒了。

示例:創建數據庫

創建一個類,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) {//第一個參數:用來打開和創建數據庫的上下文//第二個參數:數據庫文件名字//第三個參數:游標工廠,用來創建游標對象的工廠。如果傳null,表示使用默認的游標工廠//第四個參數:數據庫的版本,最低為1,如果數據庫做升級,版本值只能升,不能降super(context, name, factory, version);}//數據庫創建時,此方法調用@Overridepublic void onCreate(SQLiteDatabase db) {}//數據庫升級時,此方法調用@Overridepublic void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {} }

創建一個測試類,繼承AndroidTestCase。
src/cn.itcast.sqlite/MyOpenHelper.java

package cn.itcast.sqlite;import android.database.sqlite.SQLiteDatabase; import android.test.AndroidTestCase;public class Test extends AndroidTestCase {public void createDataBase(){//創建數據庫//1. 創建打開幫助器//測試框架提供了getContext()方法,獲取虛擬上下文MyOpenHelper oh = new MyOpenHelper(getContext(), "people.db", null, 1);//2. 創建并打開數據庫//如果數據庫不存在,先創建,后打開。如果數據庫存在,直接打開//getWritableDatabase創建的數據庫可讀可寫。//getReadableDatabase返回的數據庫也是可讀可寫。但是,在某些情況下,例如,數據庫滿了(但是現在,手機內部存儲器都比較大,基本上不會出現這種情況),getReadableDatabase就會返回一個只讀的數據庫。SQLiteDatabase db = oh.getWritableDatabase();} }

配置清單文件:

測試運行結果:

生成數據庫成功。

導出people.db,拖入SQLite Expert,可以看到自動生成了一張表,此表不能修改,不用管它。

3.1 創建表

創建數據庫完畢后會調用onCreate方法,數據庫升級后會調用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);}//數據庫創建完之后,才會調用這個函數@Overridepublic void onCreate(SQLiteDatabase db) {System.out.println("數據庫創建");}@Overridepublic void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {System.out.println("數據庫升級");} }

運行結果:
首先,刪除創建好的數據庫,重新創建數據庫。

可以看到onCreate方法被調用。

然后,升級數據庫版本。

可以看到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);}@Overridepublic void onCreate(SQLiteDatabase db) {//執行sql語句//Android所有自帶的數據庫主鍵都帶下劃線,入鄉隨俗//所有字段類型對于SQLite都是varchar類型,sql語句寫數據類型是寫給程序員看的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("數據庫升級");} }

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();} }

測試運行結果:
首先,刪除數據庫,重新創建數據庫,創建表。然后導出people.db,拖入SQLite Expert,刷新??梢钥吹?#xff0c;生成了新的person表。

表格中的RecNo是不存在的字段,只是為了讓開發人員能夠方便地看到記錄行數。

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;//測試框架初始化完畢后,測試方法執行前,此方法調用@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[]{"王亞聰"});}//測試方法執行完畢后,執行。@Overrideprotected void tearDown() throws Exception {super.tearDown();db.close();} }

測試運行結果:
首先,測試insert方法,插入數據,然后導出people.db,拖入SQLite Expert,刷新??梢钥吹?#xff0c;插入數據成功。

然后,測試delete方法,刪除一條數據,導出people.db,拖入SQLite Expert,刷新??梢钥吹?#xff0c;刪除數據成功。

如果將代碼調整,如下圖,測試insert方法或delete方法,就會報空指針異常。

原因分析:測試框架需要先被系統創建出來,然后再對它做各個參數的初始化,所有初始化做完之后,虛擬上下文才會存在。但是,private MyOpenHelper oh = new MyOpenHelper(getContext(), “people.db”, null, 1);這條語句是在測試框架構造方法被調用之前調用的,因此,虛擬上下文(getContext方法返回值)根本不存在,獲取不到。所以,MyOpenHelper根本new不出來。如此,才會報空指針異常。但是,setUp方法是在測試框架初始化完畢后,測試方法執行前,此方法才被調用。因此,虛擬上下文已經存在,沒有任何問題。

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[]{"云鶴"});//從游標中取出數據,方法與結果集類似while(cursor.moveToNext()){//只能傳入索引值,索引值根據sql語句中查詢內容所在的順序而定,例如,上面的sql語句,name在第一位,索引即為0String name = cursor.getString(0);//可以通過字段名稱獲取列索引,再通過列索引獲取內容String salary = cursor.getString(cursor.getColumnIndex("salary"));System.out.println(name + ":" + salary);}}@Overrideprotected void tearDown() throws Exception {super.tearDown();db.close();} }

測試運行結果:
測試update方法,更新一條數據,導出people.db,拖入SQLite Expert,刷新。可以看到,更新數據成功。

測試select方法,查詢數據,打印出來。可以看到,查詢成功。

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");//第一個參數:表名//第二個參數:nullColumnHack,當第三個參數ContentValues對象為null,或者里面沒有任何數據時,第二個參數才會有效。開發中,ContentValues都是有值的,所以直接傳入null即可//第三個參數:通過鍵值對把要插入的數據封裝至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();} }

測試運行結果:
測試insertApi方法,插入一條數據,導出people.db,拖入SQLite Expert,刷新。可以看到,插入數據成功。

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(){//返回被刪除的行的數量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(){//第二個參數如果傳入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();} }

測試運行結果:

測試deleteApi方法,刪除一條數據,導出people.db,拖入SQLite Expert,刷新??梢钥吹?#xff0c;刪除數據成功。

測試updateApi方法,更新一條數據,導出people.db,拖入SQLite Expert,刷新??梢钥吹?#xff0c;更新數據成功。

測試selectApi方法,查詢數據,打印出來。可以看到,查詢成功。

3.6 SQLite的事務

示例: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{//開啟事務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[]{"云鶴孫子"});//如果下面這行代碼執行,說明事務執行成功db.setTransactionSuccessful();}catch(Exception e){e.printStackTrace();}finally{//關閉事務,這時候就提交了,不用再次提交//關閉時候的時候,如果發現db.setTransactionSuccessful();語句未執行,那么就會回滾事務db.endTransaction();}}@Overrideprotected void tearDown() throws Exception {super.tearDown();db.close();} }

測試運行結果;
測試transaction方法導出people.db,拖入SQLite Expert,刷新??梢钥吹?#xff0c;事務執行成功。

如果對代碼作如下修改,事務執行中出現異常。

可以看到,報異常,并且數據未發生任何變化。

3.7 把數據庫中的數據顯示至屏幕

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("數據庫升級");} }

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>();//讀取數據庫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設置為線性布局的子節點ll.addView(tv);}} }

運行結果:

首先,在模擬器上運行應用程序,在data/data目錄下創建應用程序文件目錄。運行Test類中的insertApi方法,導出people.db文件,拖入SQLite Expert,可以看到插入數據成功。

然后,再次在模擬器上運行應用程序,將數據庫中的數據通過TextView的形式展示在屏幕上。

使用ScrollView包裹LinearLayout,ScrollView表示可以上下滑動的View。如果線性布局過大,超出屏幕顯示范圍,那么就可以上下滑動了。

4. 使用ListView顯示數據

開發中不會按照如上方法顯示數據到屏幕上的。因為,如果待顯示數據太多,程序可能會崩潰。原因在于循環提取數據,如果有10000條數據,就需要創建10000個JavaBean對象和10000個TextView對象到內存中,太耗內存。解決方案之一是采用分頁查詢,只查詢一部分數據,通過SQL語句的limit控制讀取的條數。另一個Android方式的解決方案是動態創建TextView,也就是說屏幕只能顯示10條數據,那么就創建10個JavaBean對象和10個TextView對象。如果用戶下滑屏幕顯示數據,一方面緩存(內存不足的時候會被摧毀,內存充足的時候緩存,避免下次再顯示的時候重新創建,提升性能)屏幕上方已經消失的TextView,一方面創建屏幕下方要顯示的TextView。這樣,內容中只有10個JavaBean對象及10個TextView,避免內存溢出。Android中用ListView實現此功能。

ListView專門用于顯示列表,每一行稱為一個條目(Item)

MVC

  • M:模型層 Javabean personList
  • V:視圖層 jsp ListView
  • C:控制層 servlet Adapter(Adapter是適配器,用適配器控制要顯示哪些內容)

ListAdapter是一個接口,需要實現的方法太多。因此,一般通過繼承BaseAdapter(抽象類)的方式創建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>();//讀取數據庫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();}//系統調用此方法,獲取一個View對象,該View對象會作為一個條目顯示至ListView中//position:getView返回的View對象會作為ListView的第幾個條目顯示,那么position的值就是多少@Overridepublic View getView(int position, View convertView, ViewGroup parent) {System.out.println("getView調用" + 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;}} }

運行結果:

如果將屏幕向下滑動,結果如下:

如果將屏幕向上滑動,結果如下:

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調用" + position);Person p = personList.get(position);//把指定布局文件填充成View對象//第二個參數指定把哪個布局文件轉換成View對象,變成ListView條目顯示在屏幕上//第三個參數給布局文件指定父節點,不需要,設置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對象中各個組件的內容//findViewById(R.id.tv_name);通過id尋找tv_name默認是在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);//參數一定要為字符串,如果是整數,那么就會把它作為一個id,去尋找相應的資源tv_salary.setText(p.getSalary() + "");return v;}//下面兩個方法,系統不會調用,程序員可以調用//返回條目,一般返回某個元素,例如:personList.get(position);@Overridepublic Object getItem(int position) {return null;}//返回條目的id,一般返回元素的主鍵@Overridepublic long getItemId(int position) {return 0;}} }

運行結果:

4.2 ListView的優化

上面的代碼,只要用戶一直上下滑屏,就會不斷填充新的View對象(View v = View.inflate(MainActivity.this, R.layout.item_listview, null);),一方面耗費內存(內存可能會溢出)、CPU,另一方面導致加載時間變長。

可以使用listView的緩存機制解決這個問題。任何一個條目如果被滑出屏幕,那么就緩存起來,下次再顯示,不用重新創建,直接從緩存中提取就好。ListView對條目的緩存。在顯示新的條目時,只要內存中有緩存,就會拿來用,不管是哪個條目的緩存。

示例:修改上面的示例代碼,如下: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對象,系統每次調用getView方法,都會把convertView傳遞進來@Overridepublic View getView(int position, View convertView, ViewGroup parent) {System.out.println("getView調用" + 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{//如果有緩存,復用緩存v = convertView;}//緩存并不是指給每個單獨的條目緩存,系統是只要系統中有對象被緩存就會拿來使用。//所以緩存的是View對象,并不是內容。也就是說系統中緩存的對象個數等于同屏能夠顯示的條目+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;}} }

運行結果:

可以看到,緩存了11個View對象之后,后來不會再創建新的的View對象了,而都是通過緩存獲取的。

4.3 ArrayAdapter和SimpleAdapter

ListAdapter,我們用的最多的是BaseAdapter,因為它的四個方法是我們自己實現的,自由度比較大,做復雜列表的時候比較容易。但列表不是那么復雜的時候,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只能處理文本數據,圖片數據無法個性化單獨設置,所以如果只是簡單的文本顯示,就是用ArrayAdapter。//第二個參數為資源文件,填充ListView條目的布局文件//第三個參數指定文本顯示至哪一個textView中lv.setAdapter(new ArrayAdapter<String>(this, R.layout.item_listview, R.id.tv, objects));} }

運行結果:

示例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);//集合中的每一個元素都是一個條目要顯示的數據,因為有多種數據類型,// 所以List不能使用單一泛型,先把所有類型的數據都封裝至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}));} }

運行結果:

案例1:SQLite數據庫

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}//數據庫創建時,此方法會調用@Overridepublic void onCreate(SQLiteDatabase db) {db.execSQL("create table person(_id integer primary key autoincrement, name char(10), salary char(20), phone integer(20))");}//數據庫升級時,此方法會調用@Overridepublic void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {System.out.println("數據庫升級了");} } 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);//如果數據庫不存在,先創建數據庫,再獲取可讀可寫的數據庫對象,如果數據庫存在,就直接打開SQLiteDatabase db = oh.getWritableDatabase();//如果存儲空間滿了,那么返回只讀數據庫對象 // SQLiteDatabase db = oh.getReadableDatabase();}//測試框架初始化完畢之后,在測試方法執行之前,此方法調用@Overrideprotected void setUp() throws Exception {super.setUp();oh = new MyOpenHelper(getContext(), "people.db", null, 1);db = oh.getWritableDatabase();}//測試方法執行完畢之后,此方法調用@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(){//把要插入的數據全部封裝至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{//開啟事務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;//設置 事務執行成功db.setTransactionSuccessful();}finally{//關閉事務,同時提交,如果已經設置事務執行成功,那么sql語句就生效了,反之,sql語句回滾db.endTransaction();}} }

案例2:把數據顯示至屏幕

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()){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);//把數據顯示至屏幕for (Person p : personList) {//1.集合中每有一條元素,就new一個textViewTextView tv = new TextView(this);//2.把人物的信息設置為文本框的內容tv.setText(p.toString());tv.setTextSize(18);//3.把textView設置為線性布局的子節點ll.addView(tv);}}}

案例3:使用ListView把數據顯示至屏幕

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()){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{//系統調用,用來獲知集合中有多少條元素@Overridepublic int getCount() {return personList.size();}//由系統調用,獲取一個View對象,作為ListView的條目//position:本次getView方法調用所返回的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調用:" + 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查找組件,注意調用的是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條目需要的所有數據,該案例中每個條目需要一個字符串和一個整型,所以使用一個map來封裝這兩種數據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);//設置圖標builder.setIcon(android.R.drawable.alert_dark_frame);//設置標題builder.setTitle("欲練此功必先自宮");//設置文本builder.setMessage("李志平,想清楚哦");//設置確定按鈕builder.setPositiveButton("確定", new OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {Toast.makeText(MainActivity.this, "感謝使用本軟件,再見", 0).show();}});//設置取消按鈕builder.setNegativeButton("取消", new OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {Toast.makeText(MainActivity.this, "若不自宮,一定不成功", 0).show();}});//使用創建器,生成一個對話框對象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:用戶所選的條目的下標//dialog:觸發這個方法的對話框@Overridepublic void onClick(DialogInterface dialog, int which) {Toast.makeText(MainActivity.this, "您選擇的是:" + items[which], 0).show();//關閉對話框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:用戶點擊的條目的下標//isChecked:用戶是選中該條目還是取消該條目@Overridepublic void onClick(DialogInterface dialog, int which, boolean isChecked) {checkedItems[which] = isChecked;}});//設置一個確定按鈕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();} }

總結

以上是生活随笔為你收集整理的Android应用开发:数据存储和界面展现-2的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

主站蜘蛛池模板: 黄色一几片 | 91久久视频| 性中国古装videossex | a∨色狠狠一区二区三区 | 丁香花电影高清在线阅读免费 | 打屁股视频网站 | 亚洲人在线观看视频 | 九九热精品免费视频 | 成人免费在线小视频 | 黄色av国产 | 激情自拍偷拍 | 久久手机视频 | 淫语对白| 日本免费黄色 | 毛片久久久久久久 | 可以免费在线观看的av | 朝桐光av在线一区二区三区 | 在线观看高清视频 | 一区二区三区蜜桃 | 97成人人妻一区二区三区 | 伊人国产在线视频 | 日p免费视频 | 色综合久久久久久 | 天天操免费视频 | 在线国产精品一区 | 欧美激情视频二区 | 精品视频不卡 | 日本特级黄色录像 | 国产女人高潮的av毛片 | 性歌舞团一区二区三区视频 | 久久久久久国产精品免费免费 | 欧美日韩在线二区 | 亚洲区欧美 | av中文字幕亚洲 | 亚洲性生活网站 | 天天谢天天干 | 欧美色性视频 | 最新亚洲精品 | 99re热这里只有精品视频 | 少妇激情视频 | 国产成人无码一区二区三区在线 | 精品欧美| 99色婷婷| 日韩成人av在线播放 | 好吊妞操 | 一级视频免费观看 | 黄色一级大片免费看 | 美乳人妻一区二区三区 | 国产成人麻豆精品午夜在线 | 日韩欧美一区二区在线观看 | 国产成人av网 | 色姑娘天天操 | 婷婷午夜精品久久久久久性色av | 日韩有码中文字幕在线观看 | 一本色道久久综合亚洲精品酒店 | 久久精品国产亚洲av嫖农村妇女 | 中文字幕亚洲欧美 | 大陆一级片 | 国产日韩欧美精品在线观看 | 亚洲精品综合在线 | 明日花绮罗高潮无打码 | 欧美一区二区激情 | 欧美日韩国产中文字幕 | 伊人成年网 | 亚洲精品一区二区口爆 | 热久久久久久 | 亚洲经典av | 欧美你懂得 | 大乳女喂男人吃奶视频 | av网站在线看 | 亚洲AV乱码国产精品观看麻豆 | 久久人人精品 | 久久久久久亚洲av毛片大全 | 91浏览器在线观看 | 蕾丝视频污| eeuss鲁丝片一区二区三区 | 国产乱国产乱老熟300部视频 | 性欧美videos另类艳妇3d | 免费看的黄色网 | 天堂在线 | 日韩a在线| 国产情侣第一页 | 亚洲成人麻豆 | 红桃视频黄色 | 国产精品久久久久久一区二区三区 | 狠狠爱婷婷 | 九九热精 | 中文无码日韩欧 | 国产精品一级二级三级 | 成年人在线观看网站 | 综合亚洲色图 | 欧美日韩首页 | 亚洲AV无码国产精品播放在线 | 91精选国产| 337p粉嫩大胆噜噜噜亚瑟影院 | 人人妻人人玩人人澡人人爽 | 91黄色大片 | 国产免费一区二区三区在线观看 | 咪咪色影院 |