《第一行代码:Android》读书笔记——第10章 Android网络编程

(一)WebView的用法

1、WebView也是一个普通的控件。

2、常用用法:

 1 WebView webView = (WebView)findViewById(R.id.web_view);
 2 webView.getSettings( ).setJavaScriptEnabled(true);    //让webView支持javascript脚本
 3 webView.setWebViewClient(new WebViewClient( ){
 4     @Override
 5     public boolean shouldOverrideUrlLoading(WebView view, String url){
 6            view.loadUrl(url);    //根据传入的参数再去加载新的网页
 7            return true;    //表示当前WebView可以处理打开新网页的请求,不用借助系统浏览器
 8     }
 9 });
10 webView.loadUrl("http://www.baidu.com");

3、使用任何网络功能的程序都要申请权限:

1 <uses-permission android:name="android.permission.INTERNET" />

(二)使用HttpURLConnection访问网络

1、步骤如下:

(1)创建一个URL对象,然后使用其openConnection创建一个HttpURLConnection对象:

1 URL url = new URL("http://www.baidu.com");
2 connection = (HttpURLConnection) url.openConnection();

(2)设置HttpURLConnection是GET方法还是POST方法:

1 connection.setRequestMethod("GET");

(3)对HttpURLConnection进行其他的设置:

1 connection.setConnectTimeout(8000);    //设置连接超时的毫秒数
2 connection.setReadTimeout(8000);    //设置读取超时的毫秒数

(4)用HttpURLConnection对象的getInputStream方法获取服务器的返回输入流InputStream对象:

1 InputStream in = connection.getInputStream();

(5)对输入流进行读取:

1 BufferedReader reader = new BufferedReader(
2 new InputStreamReader(in));
3 StringBuilder response = new StringBuilder();
4 String line;
5 while ((line = reader.readLine()) != null) {
6     response.append(line);
7 }

(6)用disconnect方法关闭这个HTTP连接:

1 connection.disconnect();

2、注意:

(1)网络请求要放在子线程里。

(2)在子线程里网络请求获取返回数据后,如果要进行UI操作,则要采用异步消息处理方法。

(3)要申请网络权限。

(三)使用HttpClient访问网络

1、HttpClient是一个接口类,是Apache提供的HTTP网络访问接口,从一开始就被引入到了Android API中。

2、使用步骤:

(1)创建HttpClient实例:

1 HttpClient httpClient = new DefaultHttpClient();

(2)根据发起请求的类型不同,下面的第(2)步的步骤也不同:

①GET请求:

1 HttpGet httpGet = new HttpGet("http://10.0.2.2:8081/get_data.xml");

②POST请求:

1 HttpPost httpPost = new HttpPost("http://www.baidu.com");
2 List<NameValuePair> params = new ArrayList<NameValuePair>();
3 params.add(new BasicNameValuePair("username","admin"));
4 params.add(new BasicNameValuePair("password","123456"));
5 UrlEncodedFormEntity entity = new UrlEncodedFormEntity(params, "utf-8");
6 httpPost.setEntity(entity);

(3)获取服务器返回值:

1 HttpResponse httpResponse = httpClient.execute(httpGet);

(4)判断返回状态码,如果等于200就表示请求和响应都成功了:

1 if (httpResponse.getStatusLine().getStatusCode() == 200) {
2     HttpEntity entity = httpResponse.getEntity();
3     String response = EntityUtils.toString(entity, "utf-8");
4     ...    //其他操作
5 }

3、注意:HttpClient访问网络同样要放在子线程里、申请网络权限。

(四)使用回调机制封装HttpURLConnection操作来创建HttpUtil类

1、创建HttpCallbackListener接口:

1 public interface HttpCallbackListener {
2     void onFinish(String response);    //在服务器成功响应请求时调用
3     void onError(Exception e);    //进行网络操作出错时调用
4 }

2、创建HttpUtil类:

 1 import java.io.BufferedReader;
 2 import java.io.InputStream;
 3 import java.io.InputStreamReader;
 4 import java.net.HttpURLConnection;
 5 import java.net.URL;
 6
 7 public class HttpUtil {
 8     public static void sendHttpRequest(final String address,final HttpCallbackListener listener) {
 9         new Thread(new Runnable() {
10             @Override
11             public void run() {
12                 HttpURLConnection connection = null;
13
14                 try {
15                     URL url = new URL(address);
16                     connection = (HttpURLConnection) url.openConnection();
17
18                     connection.setRequestMethod("GET");
19                     connection.setConnectTimeout(8000);
20                     connection.setReadTimeout(8000);
21                     connection.setDoInput(true);
22                     connection.setDoOutput(true);
23
24                     InputStream in = connection.getInputStream();
25                     BufferedReader reader = new BufferedReader(
26                             new InputStreamReader(in));
27                     StringBuilder response = new StringBuilder();
28                     String line;
29
30                     while ((line = reader.readLine()) != null) {
31                         response.append(line);
32                     }
33
34                     if (listener != null) {
35                         // 回调onFinish方法
36                         listener.onFinish(response.toString());
37                     }
38
39                 } catch (Exception e) {
40                     if (listener != null) {
41                         listener.onError(e);
42                     }
43                 } finally {
44                     if (connection != null) {
45                         connection.disconnect();
46                     }
47                 }
48             }
49         }).start();
50     }
51 }

3、使用时这样使用:

 1 HttpUtil.sendHttpRequest("http://www.baidu.com",new HttpCallBackListener(){
 2     @Override
 3     public void onFinish(String response){
 4         //在这里根据返回内容执行具体的逻辑
 5     }
 6
 7     @Override
 8     public void onError(Exception e){
 9         //在这里对异常情况进行处理
10     }
11 });

(五)解析XML数据

1、安装Apache服务器:下载安装包,然后安装成功后启动服务器,在浏览器里输入127.0.0.1,会看到Apache的字样。

2、在Apache安装目录:...\Apache\htdocs目录下,可以新建一个xml文件,命名为get_data.xml,加入内容,然后在浏览器里输入:127.0.0.1/get_data.xml(在手机模拟器里要输入:10.0.2.2/get_data.xml),就会显示出该文件的内容。

3、编写的XML数据的格式如下:

 1 <apps>
 2     <app>
 3         <id>1</id>
 4         <name>Google Maps</name>
 5         <version>1.0</version>
 6     </app>
 7     <app>
 8         <id>2</id>
 9         <name>Chrome</name>
10         <version>1.8</version>
11     </app>
12     <app>
13         <id>3</id>
14         <name>Google Play</name>
15         <version>3.2</version>
16     </app>
17 </apps>

4、用Pull方式解析XML数据:

 1 private void parseXMLWithPull(String xmlData) {
 2         try {
 3             XmlPullParserFactory factory = XmlPullParserFactory.newInstance();
 4             XmlPullParser xmlPullParse = factory.newPullParser();
 5             xmlPullParse.setInput(new StringReader(xmlData));
 6             int eventType = xmlPullParse.getEventType();
 7
 8             String id = "";
 9             String name = "";
10             String version = "";
11
12             while (eventType != XmlPullParser.END_DOCUMENT) {
13                 String nodeName = xmlPullParse.getName();
14                 switch (eventType) {
15                 // 开始解析某个结点
16                 case XmlPullParser.START_TAG: {
17                     if ("id".equals(nodeName)) {
18                         id = xmlPullParse.nextText();
19                     } else if ("name".equals(nodeName)) {
20                         name = xmlPullParse.nextText();
21                     } else if ("version".equals(nodeName)) {
22                         version = xmlPullParse.nextText();
23                     }
24                 }
25                     break;
26                 // 完成解析某个结点
27                 case XmlPullParser.END_TAG: {
28                     if ("app".equals(nodeName)) {
29                         Log.d("MainActivity", "id is " + id);
30                         Log.d("MainActivity", "name is " + name);
31                         Log.d("MainActivity", "version is " + version);
32                     }
33                 }
34                     break;
35                 default:
36                     break;
37                 }
38
39                 eventType = xmlPullParse.next();
40             }
41         } catch (Exception e) {
42             e.printStackTrace();
43         }
44     }

5、用SAX方式解析XML数据步骤:

(1)创建ContentHandler 类继承自DefaultHandler并重写5个方法:

 1 import org.xml.sax.Attributes;
 2 import org.xml.sax.SAXException;
 3 import org.xml.sax.helpers.DefaultHandler;
 4
 5 import android.util.Log;
 6
 7 public class ContentHandler extends DefaultHandler {
 8     private String nodeName;
 9     private StringBuilder id;
10     private StringBuilder name;
11     private StringBuilder version;
12
13     @Override
14     public void startDocument() throws SAXException {
15         id = new StringBuilder();
16         name = new StringBuilder();
17         version = new StringBuilder();
18     }
19
20     @Override
21     public void startElement(String uri, String localName, String qName,
22             Attributes attributes) throws SAXException {
23         // 记录当前结点名
24         nodeName = localName;
25     }
26
27     @Override
28     public void characters(char[] ch, int start, int length)
29             throws SAXException {
30         // 根据当前结点名判断将内容添加到哪一个StringBuilder对象中
31         if ("id".equals(nodeName)) {
32             id.append(ch, start, length);
33         } else if ("name".equals(nodeName)) {
34             name.append(ch, start, length);
35         } else if ("version".equals(nodeName)) {
36             version.append(ch, start, length);
37         }
38     }
39
40     @Override
41     public void endElement(String uri, String localName, String qName)
42             throws SAXException {
43         // 用trim方法去掉空白字符
44         if ("app".equals(localName)) {
45             Log.d("MainActivity", "id is " + id.toString().trim());
46             Log.d("MainActivity", "name is " + name.toString().trim());
47             Log.d("MainActivity", "version is " + version.toString().trim());
48
49             // 将StringBuilder清空
50             id.setLength(0);
51             name.setLength(0);
52             version.setLength(0);
53         }
54     }
55
56     @Override
57     public void endDocument() throws SAXException {
58     }
59 }

(2)写具体方法:

 1 private void parseXMLWithSAX(String xmlData) {
 2         try {
 3             SAXParserFactory factory = SAXParserFactory.newInstance();
 4             XMLReader xmlReader = factory.newSAXParser().getXMLReader();
 5             ContentHandler handler = new ContentHandler();
 6
 7             xmlReader.setContentHandler(handler);
 8
 9             // 开始执行解析
10             xmlReader.parse(new InputSource(new StringReader(xmlData)));
11         } catch (Exception e) {
12             e.printStackTrace();
13         }
14 }

(六)解析Json数据

  见我的博客园文章:用GSON解析Json格式数据,这里不再详写。

 

  【本章结束】

时间: 2024-10-20 09:17:13

《第一行代码:Android》读书笔记——第10章 Android网络编程的相关文章

《第一行代码》读书笔记

一: 1:项目文件 开发中经常用到的有: app目录:存放项目代码.资源文件 build.gradle:项目全局gradle脚本. 2:app目录 libs:放到libs目录下的第三方jar包会自动添加到项目构建路径. java:存放java代码的地方. res:资源文件目录,包括:图片.布局.字符串.颜色.样式.菜单等. AndroidManifest.xml:项目配置文件.多用于注册四大组.添加权限等. proguard-rules.pro:项目代码混淆规则. 3:build.gradle逐

阅读郭林《第一行代码》的笔记——第6章 数据存储全方案,详解持久化技术

瞬时数据是指那些存储在内存当中,有可能会因为程序关闭或其他原因导致内存被回收而丢失的数据.这对于一些关键性的数据信息来说是绝对不能容忍的,谁都不希望自己刚发出去的一条微博,刷新一下就没了吧.那么怎样才能保证让一些关键性的数据不会丢失呢?这就需要用到数据持久化技术了. 持久化技术简介 数据持久化就是指将那些内存中的瞬时数据保存到存储设备中,保证即使在手机或电脑关机的情况下,这些数据仍然不会丢失.保存在内存中的数据是处于瞬时状态的,而保存在存储设备中的数据是处于持久状态的,持久化技术则是提供了一种机

《第一行代码》读书笔记-2

gravity的对齐方式有top,bottom,left,right,center可以用"|"同时选取. wrap_content 适配内容,保证显示完整. match_parent:父布局决定大小 默认人提示性文字:hint xml中给imageview添加图片:src= activity中设置图片,imageview.setimageresource(id); progressbar visible:可见 invisible:隐身  gone:不在屏幕上 设置方法:setvisib

使用Intent在活动之间穿梭(《第一行代码》读书笔记)

以下全是个人理解//瞎扯 其实活动理解理解起来就像一个个函数 那么Intent就是调用函数和参数传递 可以有无参,仅仅是调用 Intent intent = new Intent(A.this, B.class); startActivity(intent); 由活动A调用活动B,无参,无返回值,当然B调用结束,要回到A. 可以有参数 A---Intent intent = new Intent(A.this, B.class); intent.putExtra( "这里是参数的键"

《第一行代码》读书笔记-1

assets文件夹存放随程序打包的文件 libs存放第三方Jar包 R.string.xxx获得对该字符串的引用 Log.vdiwe->i---info XM文件中引用一个id:@id/id_name 定义一个id:@+id/id_name 隐藏标题栏在activity中:requestwindowfeature(window.feature_no_title);在setcontentview之前执行. menu intent的action配合category属性区分intent的目的 inte

阅读郭林《第一行代码》的笔记——第2章 先从看得到的入手,探究活动

一.活动是什么,活动的基本用法 活动(Activity)是最容易吸引到用户的地方了,它是一种可以包含用户界面的组件,主要用于和用户进行交互.一个应用程序中可以包含零个或多个活动,但不包含任何活动的应用程序很少见,谁也不想让自己的应用永远无法被用户看到吧? Android程序的设计讲究逻辑和视图分离,最好每一个活动都能对应一个布局,布局就是用来显示界面内容的. 创建和加载布局 @Override protected void onCreate(Bundle savedInstanceState)

《第一行代码》学习笔记 第 2 章

第 2 章 先从看得到的入手,探究活动 知识点1:在活动中使用 Menu 在 res 目录下新建一个 menu 文件夹,右击 menu文件夹→New→Android XML File,文件名输入 main 在 main.xml 中添加如下代码: <menu xmlns:android="http://schemas.android.com/apk/res/android" > <item android:id="@+id/add_item" and

《浪潮之巅》读书笔记——第10章 惠普

第10章 惠普  公司发展    硅谷最早的公司    1939 Hewlett和Packard创办    进驻斯坦福工业园    90年代前发现一帆风顺    衰落      领导者的错误        产品线太长.内部混乱        将仪器部门剥离上市(安捷伦)        与亏损的康柏合并        卡莉.菲奥莉娜          1999 最差CEO?          主持朗讯与飞利浦的合并          2002 与康柏合并      日本.中国制造冲击      从

Android学习笔记—第四章 Android开发组件2

第四章 Android开发组件2 列表类组件 (1)ListView组件:以垂直列表的形式列出需要显示的列表项 相关属性: a. android:divider  用于为列表视图设置分隔条,可以用颜色或者图片资源 b. android:dividerHeight  设置分隔条的高度 c. android:entries  通过数组资源为ListView指定列表项 d. android:footerDividersEnabled  设置是否在footerView之前绘制分隔条,默认为true. e