當(dāng)前位置:首頁 > 嵌入式 > 嵌入式軟件
[導(dǎo)讀] 注:本文來自友盟-安卓巴士教程大賽第一名獲獎作品,作者安卓巴士的ID為liupeinye。推薦給所有剛剛開始接觸Android開發(fā)的朋友們!本文面向Android初級開發(fā)者,有一定的Java

 注:本文來自友盟-安卓巴士教程大賽第一名獲獎作品,作者安卓巴士的ID為liupeinye。推薦給所有剛剛開始接觸Android開發(fā)的朋友們!

本文面向Android初級開發(fā)者,有一定的Java和Android知識即可。

文章覆蓋知識點:HttpWatch抓包,HttpClient模擬POST請求,Jsoup解析HTML代碼,動態(tài)更新ListView

背景介紹:客戶端(Client)或稱為用戶端,是指與服務(wù)器相對應(yīng),為客戶提供本地服務(wù)的程序。而android系統(tǒng)上的90%客戶端軟件都有一個共性,就是為了改善網(wǎng)頁在android系統(tǒng)上體驗不佳而生,最具有影響力的軟件有:新浪微博、人人網(wǎng)、淘寶等,這類軟件最突出的特點就是,先有網(wǎng)站再有軟件。由于網(wǎng)絡(luò)技術(shù)發(fā)展的多樣性,手機(jī)瀏覽器往往無法跟隨它的步伐,為改善用戶體驗,網(wǎng)站客戶端軟件印運(yùn)而生。

以下內(nèi)容100%原創(chuàng),并在安卓巴士論壇首發(fā),如需轉(zhuǎn)載,請注明作者和出處。謝謝合作。

開發(fā)Android網(wǎng)站客戶端通常有兩種方法:第一種,通過服務(wù)端的開放平臺,調(diào)用提供的API接口來開發(fā),比如說open sina;第二種,服務(wù)端沒有提供任何接口,你也沒有服務(wù)端任何數(shù)據(jù)庫訪問權(quán)限,就是一個純純粹粹的網(wǎng)站,要你做客戶端。今天,我要和大家分享的正是第二種情況。

這是一個簡單的示意圖,告訴我們,數(shù)據(jù)是由網(wǎng)頁從數(shù)據(jù)庫中取出,我們要為這個系統(tǒng)做客戶端,我們就應(yīng)該這樣去改造它。

通過這樣間接的方法來訪問數(shù)據(jù)庫,只要網(wǎng)頁能看到的內(nèi)容,我們的客戶端都能獲取到,而UI是由你自行制作,就可以使使用體驗上一個臺階。

既然網(wǎng)頁是我們的數(shù)據(jù)樞紐,我們就從網(wǎng)頁分析著手。

該教程來自本人項目-掌上民大-真實經(jīng)驗,所以用項目中的”掌上圖書館”板塊來示范。

該項目任務(wù)為中南民族大學(xué)圖書館圖書查詢功能制作客戶端。

首先打開該網(wǎng)址http://www.lib.scuec.edu.cn/,我們會看到主界面

正中間就是查詢?nèi)肟?,我們輸?rdquo;android”進(jìn)行查詢

得到結(jié)果的網(wǎng)頁,但我們能發(fā)現(xiàn),這個頁面是ILAS圖書管理系統(tǒng),所以真正的入口并不是上面紅圈的地方。

所以我繼續(xù)找,經(jīng)過短暫的觀察,發(fā)現(xiàn)入口在這里

我們點擊進(jìn)入

果然就是這貨,我們點擊書目查詢

就是它了。真正的入口就在這里,這時我們打開HttpWatch軟件,點”Record”,在搜索框里輸入”android”,點擊查詢,抓取以”android”為關(guān)鍵字搜索時瀏覽器的所有包。待結(jié)果界面載入完成后,HttpWatch上就會出現(xiàn)如下信息

我們先看Summary選項卡,我們可以初步了解,這是一個POST請求(Http請求中的一種,另一種是GET),POST到的網(wǎng)址是http://coin.lib.scuec.edu.cn/cgi-bin/IlaswebBib。

這樣我們的思路就清晰了,我們的客戶端需要模擬瀏覽器,向上述地址POST一個包,那個地址肯定會返回一個Content給我們,不出意外的話,Content里面就是我們要的書目信息。那么,瀏覽器POST上去的內(nèi)容是什么呢?我們點擊這條POST請求,看詳細(xì)信息,

由于是POST請求,我們先看POST DATA,里面是以鍵值對的形式存儲的,這里顯示了我們?yōu)g覽器在我們搜索”android”時,POST的所有數(shù)據(jù)。那這些鍵值對又代表了什么呢,我們打開這個網(wǎng)頁的源碼來一探究竟。

從這段可以看出v_index是表示查找途徑的它有TITLE,AUTHOR,SUBJECT,CLASSNO,ISBN,CALLNO六種值

FLD_DAT_BEG和FLD_DAT_END分別是開始和結(jié)束年份

v_value表示用戶在搜索框中輸入的內(nèi)容

v_paggnum表示每頁顯示的書目條數(shù),有10 15 20三種

v_seldatabases是檢索庫 有0 1 2三種值v_LogicSrch是檢索方式 有0 1兩種值[!--empirenews.page--]

Submit是查詢或重填,有 查 詢 和 重 填 兩種值

至此,我們弄清楚了POST Data里所有內(nèi)容的含義和取值可能。但我們模擬POST請求為什么,其實就是為了得到搜索的書目信息,所以我們看一下返回的Content是不是我們要的東西

果然,就是我們搜到的書目信息,就以String的形式放在Content里面。最后我們查看一下Stream,截圖,以防等下我們需要這里面的東西

好了,這個頁面的工作原理我們已經(jīng)弄清楚了:用戶在網(wǎng)頁中輸入搜索內(nèi)容后,點擊查詢,瀏覽器會POST一個Data到目標(biāo)網(wǎng)址,該網(wǎng)址的返回信息就是搜到的書目。

我們開始編寫代碼,模擬這個過程,先打開eclipse建立一個Java項目(注意是Java項目,因為Java項目可以完美移植到Android項目中且調(diào)試方便,并且模擬Http請求這一過程沒有用到任何Android功能)。

導(dǎo)入HttpClient的4個包commons-codec、commons-httpclient、commons-logging、log4j。

//實例化HttpClient

HttpClient client = new HttpClient();

//Stream頁面里面有Host地址 端口是80

client.getHostConfiguration().setHost("http://coin.lib.scuec.edu.cn", 80);

//用目標(biāo)地址 實例一個POST方法

PostMethod post = new PostMethod("http://coin.lib.scuec.edu.cn/cgi-bin/IlaswebBib");

//將需要的鍵值對寫出來

NameValuePair beg = new NameValuePair("FLD_DAT_BEG" , “”);

NameValuePair end = new NameValuePair("FLD_DAT_END" , “”);

NameValuePair submit = new NameValuePair("submit" , "查 詢"));

NameValuePair vIndex = new NameValuePair("v_index" , “TITLE”);

NameValuePair vLogicSrch = new NameValuePair("v_LogicSrch" , "0");

NameValuePair vPagenum = new NameValuePair("v_pagenum" , "10");

NameValuePair vSeldatabase = new NameValuePair("v_seldatabase" , "0");

NameValuePair vValue = new NameValuePair("v_value" ,”android”);

//給POST方法加入上述鍵值對

post.setRequestBody(new NameValuePair[] {beg , end , submit , vIndex , vLogicSrch , vPagenum , vSeldatabase , vValue});

//執(zhí)行POST方法

client.executeMethod(post);

//將POST返回的數(shù)據(jù)以流的形式讀入,再把輸入流流至一個buff緩沖字節(jié)數(shù)組

//StreamTool類是我自己寫的一個工具類,其內(nèi)容將在下文附出

byte[] buff = StreamTool.readInputStream(post.getResponseBodyAsStream());

//將返回的內(nèi)容格式化為String存在html中

String html = new String(buff);

//任務(wù)完成了,釋放連接

post.releaseConnection();

//StreamTool類如下

public class StreamTool {

/**

* 從輸入流中獲取數(shù)據(jù)

* @param inputStream 輸入流

* @return 字節(jié)數(shù)組

* @throws Exception

*/

public static byte[] readInputStream(InputStream inputStream) throws Exception

{

//實例化一個輸出流

ByteArrayOutputStream outputStream = new ByteArrayOutputStream();

//一個1024字節(jié)的緩沖字節(jié)數(shù)組

byte[] buffer = new byte[1024];

int len = 0;

//讀流的基本知識

while ((len=inputStream.read(buffer)) != -1) {

outputStream.write(buffer, 0, len);

}

//用完要關(guān),大家都懂的

inputStream.close();

return outputStream.toByteArray();

}

}

現(xiàn)在,我們得到了POST方法返回的String,我們輸出到控制臺看看是什么

System.out.println(html);

沒錯,就是我們上文看到的HttpWatch 抓到的返回Content,也就是一段HTML代碼,這說明,我們模擬瀏覽器POST請求成功了!

我們再試試別的搜索內(nèi)容,來一個”android開發(fā)”(即將v_value鍵值對的值改成”android開發(fā)”),這時運(yùn)行后,我們卻從控制臺得到了這樣的結(jié)果:

經(jīng)過幾次試驗后,發(fā)現(xiàn)一個規(guī)律,只要搜索內(nèi)容中包括中文,就搜不到。

所以可以判定是中文編碼的問題,(在開發(fā)這類客戶端時候,中文編碼往往是個很具困難的問題。安卓巴士開發(fā)3群的某群友提到:服務(wù)器交流用的編碼是”ISO-8859-1”,跟我起初用到的編碼一致,但真實性仍需考證)所以我們修改上面的代碼,將代表搜索內(nèi)容的v_value對應(yīng)的值編碼為”ISO-8859-1”

就將上段代碼中的

NameValuePair vValue = new NameValuePair("v_value" ,”android”);

改為

NameValuePair vValue = new NameValuePair("v_value" , new String(“android開發(fā)”.getBytes(),"ISO-8859-1"));[!--empirenews.page--]

這時再運(yùn)行,控制臺成功輸出以” android開發(fā)”為關(guān)鍵字的Content。

至此,我們POST請求才真正完成。 觀察控制臺的HTML后發(fā)現(xiàn),我們需要的書目信息就在里面,只不過被一些HTML標(biāo)簽包裹住了,下一步我們就要解放這些信息,存儲到容器里。

這里我們要用到Jsoup,一個Java開源HTML解析器(來自org.jsoup包)。

我們直接上代碼,逐行解釋(大家最好對應(yīng)上面的HTML代碼來理解)

首先我們建一個容器來裝這些解析到的數(shù)據(jù),由于我的項目是將這些數(shù)據(jù)以ListView呈現(xiàn)給用戶,而ListView的數(shù)據(jù)是由Adapter提供,Adapter需要傳一個特殊容器-包含HashMap的ArrayList(Android基礎(chǔ)知識)

//所以有

List

//開始使用Jsoup

//Jsoup支援一個Document類 將剛才的html轉(zhuǎn)化成Document

Document document = Jsoup.parse(html);

//一個Document又由elements組成 我們選擇”tr”開頭的標(biāo)簽,存入 trs元素群中

Elements trs = document.select("tr");

//得到整個HTML中包含tr的標(biāo)簽的個數(shù)

int totalTrs = trs.size();

//我們可以觀察上面沒有搜索結(jié)果的那個HTML。發(fā)現(xiàn),如果totalTrs<=3就表示沒結(jié)果。

//只要有書目結(jié)果totalTrs必定大于3,于是

if(totalTrs > 3)

for(int i = 0;i < totalTrs – 3;i++)

{

//觀察HTML,從第i+2個tr開始,包含的才是我們要的書目信息

//我們從每個tr中選出td標(biāo)簽元素群

Elements tds = trs.get(i + 2).select("td");

//得到每個tr中td的個數(shù)

int totalTds = tds.size();

//一個臨時的HashMap,里面是String-Object鍵值對

Map

//j是一個標(biāo)識數(shù)

for(int j =0;j < totalTds ;j++)

{

switch (j) {

//0表示第一個,即書名

//put方法即向map加入一條鍵值對

//html()方法就得到標(biāo)簽括起來的內(nèi)容

case 0:

map.put("book_title", tds.get(j).html().toString());

break;

case 1:

//1表示第二個,即作者

map.put("book_author", tds.get(j).html().toString());

break;

case 2:

//2表示第三個,即出版信息

map.put("book_press", tds.get(j).html().toString());

break;

case 3:

//3表示第四個,即頁數(shù)

map.put("book_page", tds.get(j).html().toString());

break;

case 4:

//4表示第五個,即價格

map.put("book_price", tds.get(j).html().toString());

break;

case 5:

//5表示第六個,即索取號

map.put("book_noFor", tds.get(j).html().toString());

break;

case 6:

//6表示第七個,即那段網(wǎng)址

//那段網(wǎng)址td中又包含一個a標(biāo)簽,a標(biāo)簽的href屬性的值就是網(wǎng)址

//attr(“href”)可以返回href屬性的值

map.put("book_detail", tds.get(j).select("a").attr("href").toString());

break;

default:

break;

}

}

list.add(map);

}

list就是我們需要的ArrayList啦

上面所有代碼調(diào)通后,我們只需一些簡單的復(fù)制粘貼,就可以放在我們的Android工程中,加上一段簡單的代碼就可以讓ListView顯示這個ArrayList。(由于沒有任何技術(shù)含量,以及該項目暫未上線,此段代碼不予以展示,敬請諒解)

接下來,我們一個頁面最多只包含10個書目信息,而我們校圖書館,光以”Java”為關(guān)鍵字的書就超過1000本,怎么來顯示完全呢,一次顯示所有的書肯定不現(xiàn)實。首先數(shù)據(jù)量太大,手機(jī)無法承受;消耗流量過大,用戶體驗極差。所以,我們就需要ListView能夠動態(tài)加載數(shù)據(jù),即一開始顯示十項,如果用戶此時拉動ListView顯示完十項之后,自動聯(lián)網(wǎng),再加載十項(如果還有十項的話),這樣的用戶體驗會非常順暢。

這個功能的核心是,我們的ListView需要實現(xiàn)OnScrollListener接口。

如果你的ListView所在的Activity繼承的是ListActivity的話,只需在extends ListActivity后面加上implements OnScrollListener,這時你需要復(fù)寫onScroll和onScrollStateChanged。如果你的ListView是從XMLgetView 得到的,你只需為它setOnScrollListener,也會需要你復(fù)寫onScroll和onScrollStateChanged。

不管你用哪種方法,我們只用修改onScroll方法

@Override

public void onScroll(AbsListView view, int firstVisibleItem,

int visibleItemCount, int totalItemCount) {

// TODO Auto-generated method stub

//關(guān)鍵的判斷代碼,這句話表示用戶將ListView拉至最底部

if(firstVisibleItem + visibleItemCount == totalItemCount)

//你只需要把繼續(xù)得到下面十項的代碼寫在這里,就可以實現(xiàn)上述功能了。

//同樣再使用一次POST方法,不再贅述

//代碼由于同樣原因不予以展示,敬請諒解

至此,文章開頭的幾個知識點已經(jīng)全部講解完畢,時間倉促,事物繁忙,可能會影響文章質(zhì)量,還請大家多多包涵。 如果有問題,可以直接回帖、發(fā)論壇信息或通過Email:anliupeinye@gmail.com聯(lián)系我。

項目成品展示:

看看這些信息是不是就是上面用網(wǎng)頁以"android"為關(guān)鍵字搜索到的?

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點: 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運(yùn)營商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團(tuán))股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉