兼职做设计的网站,怎么搭建自己的网站后台,安卓app用什么开发,成都旅游攻略景点必去文章目录 Android解析XML格式数据搭建Web服务器Pull解析方式SAX解析方式 Android解析XML格式数据
通常情况下#xff0c;每个需要访问网络的应用程序都会有一个自己的服务器#xff0c;我们可以向服务器提交数据#xff0c;也可以从服务器上获取数据。不过这个时候就出现了… 文章目录 Android解析XML格式数据搭建Web服务器Pull解析方式SAX解析方式 Android解析XML格式数据
通常情况下每个需要访问网络的应用程序都会有一个自己的服务器我们可以向服务器提交数据也可以从服务器上获取数据。不过这个时候就出现了一个问题这些数据到底要以什么样的格式在网络上传输呢随便传递一段文本肯定是不行的因为另一方根本就不会知道这段文本的用途是什么。因此一般我们都会在网络上传输一些格式化后的数据这种数据会有一定的结构规格和语义当另一方收到数据消息之后就可以按照相同的结构规格进行解析从而取出他想要的那部分内容。
在网络上传输数据时最常用的格式有两种XML和JSON下面我们就来一个一个地进行学习本节首先学习一下如何解析XML格式的数据。
搭建Web服务器
在开始之前我们还需要先解决一个问题就是从哪儿才能获取一段XML格式的数据呢这里我准备教你搭建一个最简单的Web服务器在这个服务器上提供一段XML文本然后我们在程序里去访问这个服务器再对得到的XML文本进行解析。
搭建Web服务器其实非常简单有很多的服务器类型可供选择这里我准备使用Apache服务器。首先你需要去下载一个Apache服务器的安装包官方下载地址是http://httpd.apache.org/download.cgi。如果你在这个网址中找不到Windows版的安装包也可以直接在百度上搜索“Apache服务器下载”将会找到很多下载链接。
下载完成后双击就可以进行安装了如图所示: 然后一直点击Next会提示让你输入自己的域名我们随便填一个域名就可以了如图所示: 接着继续一直点击Next会提示让你选择程序安装的路径这里我选择安装到C:\Apache目录下之后再继续点击Next就可以完成安装了。安装成功后服务器会自动启动起来你可以打开电脑的浏览器来验证一下。在地址栏输入127.0.0.1如果出现了如图9.5所示的界面就说明服务器已经启动成功了。 接下来进入到C:\Apache\htdocs目录下在这里新建一个名为get_data.xml的文件然后编辑这个文件并加入如下XML格式的内容。
apps
app
id1/id
nameGoogle Maps/name
version1.0/version
/app
app
id2/id
nameChrome/name
version2.1/version
/app
app
id3/id
nameGoogle Play/name
version2.3/version
/app
/apps这时在浏览器中访问这个网址就应该出现如图9.6所示的内容。 好了准备工作到此结束接下来就让我们在Android程序里去获取并解析这段XML数据吧。
Pull解析方式
解析XML格式的数据其实也有挺多种方式的本节中我们学习比较常用的两种Pull解析和SAX解析。那么简单起见这里仍然是在NetworkTest项目的基础上继续开发这样我们就可以重用之前网络通信部分的代码从而把工作的重心放在XML数据解析上。
既然XML格式的数据已经提供好了现在要做的就是从中解析出我们想要得到的那部分内容。修改MainActivity中的代码如下所示
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
...
private void sendRequestWithOkHttp() {
new Thread(new Runnable() {
Override
public void run() {
try {
OkHttpClient client new OkHttpClient();
Request request new Request.Builder()
//指定访问的服务器地址是电脑本机
.url(http://10.0.2.2/get_data.xml)
.build();
Response response client.newCall(request).execute();
String responseData response.body().string();
parseXMLWithPull(responseData);
} catch (Exception e) {
e.printStackTrace();
}
}
}).start();
}
...
private void parseXMLWithPull(String xmlData) {
try {
XmlPullParserFactory factory XmlPullParserFactory.newInstance();
XmlPullParser xmlPullParser factory.newPullParser();
xmlPullParser.setInput(new StringReader(xmlData));
int eventType xmlPullParser.getEventType();
String id ;
String name ;
String version ;
while (eventType ! XmlPullParser.END_DOCUMENT) {
String nodeName xmlPullParser.getName();
switch (eventType) {
//开始解析某个节点
case XmlPullParser.START_TAG: {
if (id.equals(nodeName)) {
id xmlPullParser.nextText();
} else if (name.equals(nodeName)) {
name xmlPullParser.nextText();
} else if (version.equals(nodeName)) {
version xmlPullParser.nextText();
}
break;
}
//完成解析某个节点
case XmlPullParser.END_TAG: {
if (app.equals(nodeName)) {
Log.d(MainActivity, id is id);
Log.d(MainActivity, name is name);
Log.d(MainActivity, version is version);
}
break;
}
default:
break;
}
eventType xmlPullParser.next();
}
} catch (Exception e) {
e.printStackTrace();
}
}
}可以看到这里首先是将HTTP请求的地址改成了10.0.2.2对于模拟器来说就是电脑本机的IP地址。在得到了服务器返回的数据后我们并不再直接将其展示而是调用了parseXMLWithPull()方法来解析服务器返回的数据。
下面就来仔细看下parseXMLWithPull()方法中的代码吧。这里首先要获取到一个XmlPullParserFactory的实例并借助这个实例得到XmlPullParser对象然后调用XmlPullParser的setInput()方法将服务器返回的XML数据设置进去就可以开始解析了。解析的过程也非常简单通过getEventType()可以得到当前的解析事件然后在一个while循环中不断地进行解析如果当前的解析事件不等于XmlPullParser.END_DOCUMENT说明解析工作还没完成调用next()方法后可以获取下一个解析事件。
在while循环中我们通过getName()方法得到当前节点的名字如果发现节点名等于id、name或version就调用nextText()方法来获取节点内具体的内容每当解析完一个app节点后就将获取到的内容打印出来。
好了整体的过程就是这么简单下面就让我们来测试一下吧。运行NetworkTest项目然后点击Send Request按钮观察logcat中的打印日志可以看到我们已经将XML数据中的指定内容成功解析出来了。
SAX解析方式
Pull解析方式虽然非常好用但它并不是我们唯一的选择。SAX解析也是一种特别常用的XML解析方式虽然它的用法比Pull解析要复杂一些但在语义方面会更加清楚。
通常情况下我们都会新建一个类继承自DefaultHandler并重写父类的5个方法如下所示
public class MyHandler extends DefaultHandler {
Override
public void startDocument() throws SAXException {
}
Override
public void startElement(String uri, String localName, String qName, Attributes
attributes) throws SAXException {
}
Override
public void characters(char[] ch, int start, int length) throws SAXException {
}
Override
public void endElement(String uri, String localName, String qName) throws
SAXException {
}
Override
public void endDocument() throws SAXException {
}
}这5个方法一看就很清楚吧startDocument()方法会在开始XML解析的时候调用startElement()方法会在开始解析某个节点的时候调用characters()方法会在获取节点中内容的时候调用endElement()方法会在完成解析某个节点的时候调用endDocument()方法会在完成整个XML解析的时候调用。其中startElement()、characters()和endElement()这3个方法是有参数的从XML中解析出的数据就会以参数的形式传入到这些方法中。需要注意的是在获取节点中的内容时characters()方法可能会被调用多次一些换行符也被当作内容解析出来我们需要针对这种情况在代码中做好控制。
那么下面就让我们尝试用SAX解析的方式来实现和上一小节中同样的功能吧。新建一个ContentHandler类继承自DefaultHandler并重写父类的5个方法如下所示
public class ContentHandler extends DefaultHandler {
private String nodeName;
private StringBuilder id;
private StringBuilder name;
private StringBuilder version;
Override
public void startDocument() throws SAXException {
id new StringBuilder();
name new StringBuilder();
version new StringBuilder();
}
Override
public void startElement(String uri, String localName, String qName, Attributes
attributes) throws SAXException {
//记录当前节点名
nodeName localName;
}
Override
public void characters(char[] ch, int start, int length) throws SAXException {
//根据当前的节点名判断将内容添加到哪一个StringBuilder对象中
if (id.equals(nodeName)) {
id.append(ch, start, length);
} else if (name.equals(nodeName)) {
name.append(ch, start, length);
} else if (version.equals(nodeName)) {
version.append(ch, start, length);
}
}
Override
public void endElement(String uri, String localName, String qName) throws
SAXException {
if (app.equals(localName)) {
Log.d(ContentHandler, id is id.toString().trim());
Log.d(ContentHandler, name is name.toString().trim());
Log.d(ContentHandler, version is version.toString().trim());
//最后要将StringBuilder清空掉
id.setLength(0);
name.setLength(0);
version.setLength(0);
}
}
Override
public void endDocument() throws SAXException {
super.endDocument();
}
}可以看到我们首先给id、name和version节点分别定义了一个StringBuilder对象并在startDocument()方法里对它们进行了初始化。每当开始解析某个节点的时候startElement()方法就会得到调用其中localName参数记录着当前节点的名字这里我们把它记录下来。接着在解析节点中具体内容的时候就会调用characters()方法我们会根据当前的节点名进行判断将解析出的内容添加到哪一个StringBuilder对象中。最后在endElement()方法中进行判断如果app节点已经解析完成就打印出id、name和version的内容。需要注意的是目前id、name和version中都可能是包括回车或换行符的因此在打印之前我们还需要调用一下trim()方法并且打印完成后还要将StringBuilder的内容清空掉不然的话会影响下一次内容的读取。
接下来的工作就非常简单了修改MainActivity中的代码如下所示
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
...
private void sendRequestWithOkHttp() {
new Thread(new Runnable() {
Override
public void run() {
try {
OkHttpClient client new OkHttpClient();
Request request new Request.Builder()
//指定访问的服务器地址是电脑本机
.url(http://10.0.2.2/get_data.xml)
.build();
Response response client.newCall(request).execute();
String responseData response.body().string();
parseXMLWithSAX(responseData);
} catch (Exception e) {
e.printStackTrace();
}
}
}).start();
}
...
private void parseXMLWithSAX(String xmlData) {
try {
SAXParserFactory factory SAXParserFactory.newInstance();
XMLReader xmlReader factory.newSAXParser().getXMLReader();
ContentHandler handler new ContentHandler();
//将ContentHandler的实例设置到XMLReader中
xmlReader.setContentHandler(handler);
//开始执行解析
xmlReader.parse(new InputSource(new StringReader(xmlData)));
} catch (Exception e) {
e.printStackTrace();
}
}
}在得到了服务器返回的数据后我们这次去调用parseXMLWithSAX()方法来解析XML数据。parseXMLWithSAX()方法中先是创建了一个SAXParserFactory的对象然后再获取到XMLReader对象接着将我们编写的ContentHandler的实例设置到XMLReader中最后调用parse()方法开始执行解析就好了。
现在重新运行一下程序点击Send Request按钮后观察logcat中的打印日志你会看到和Pull解析一样的结果。
如果对你有帮助就一键三连呗关注点赞收藏我会持续更新更多干货~~