目录
一、准备工作
EasyWebServer
二、Pull解析方式
三、SAX解析方式
我们可以向服务器提交数据,也可以获取数据,但是数据交换的不仅仅是内容,还要对数据的属性、作用进行描述,当另一方收到数据消息后可以按照相同的结构规格进行解析,从而取出想要的部分内容。
在网络上传输数据时最常用的格式有两种:XML和JSON。
一、准备工作
接下来我们需要获取一段XML格式的数据,先搭建一个简单的服务器,在服务器上获取XML文本,我们这里使用EasyWebServer搭建一个小型服务器。
EasyWebServer
EasyWebServer是一款小型的Web服务器软件。它可以很快速地在您的PC上创建一个站点,而无需IIS等庞大复杂的工具。
下载地址:【EasyWebServer网站服务器下载】2022年最新官方正式版EasyWebServer网站服务器免费下载 - 腾讯软件中心官网
下载完成后打开软件,
点击设置,主目录是共享文件夹,在文件里建立一个HTML文件。
然后打开手机的热点,笔记本连上手机热点,查看笔记本获取的ip,然后启动服务器,再使用手机浏览器打开笔记本获取的ip,如果成功可以看到共享文件夹里的HTML文件。
上面操作成功后,接下来在网页目录下创建一个get_data.xml文件,加入以下内容:
<apps>
<app>
<id>1</id>
<name>Google Maps</name>
<version>1.0</version>
</app>
<app>
<id>2</id>
<name>Chrome</name>
<version>2.1</version>
</app>
<app>
<id>3</id>
<name>Goole Play</name>
<version>2.3</version>
</app>
</apps>
准备工作已完成,下面在Android程序里获取并解析这段XML数据。
二、Pull解析方式
下面接着在NetworkTest项目上开发,可先移步这篇文章,学习HTTP访问网络
修改MainActivity代码如下:
首先将HTTP请求的地址改成你的ip+get_data.xml,每个人不一样注意修改!然后调用parseXMLWithPull(responseData)方法解析服务器返回的数据。
parseXMLWithPull()方法的代码,首先获取一个XmlPullParserFactory实例,并得到XmlPullParse对象,然后调用这个对象的setInput()方法将服务器返回的XML数据设置进去就可以解析了。
解析过程:通过getEventType()可以得到当前的解析事件,然后在循环里解析,如果当前的解析事件不等于XmlPullParser.END_DOCUMENT说明还没解析完成,调用next()方法可以解析下一个事件。
在循环中,通过getName()得到当前节点名字,如果节点名字等于id、name或version就调用nextText()方法获取具体内容。
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
TextView responseText;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Button sendRequest = (Button) findViewById(R.id.send_request);
responseText = (TextView) findViewById(R.id.response_text);
sendRequest.setOnClickListener(this);
}
@Override
public void onClick(View v) {
if (v.getId() == R.id.send_request) {
sendRequestWithOkHttp();
}
}
private void sendRequestWithOkHttp() {
new Thread(new Runnable() {
@Override
public void run() {
try {
OkHttpClient client = new OkHttpClient();
Request request = new Request.Builder()
.url("http://192.168.30.175/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();
}
}
}
效果如下:
点击按钮后查看日志
三、SAX解析方式
SAX解析用法比Pull解析更复杂一些,但在语义方面会更清楚。
新建一个ContentHandler类继承自DefaultHand,并重写父类5个方法,如下:
startDocument()方法会在开始XML解析的时候调用,startElement()方法会在开始解析某个节点的时候调用,characters()会在获取节点中内容的时候调用,endElement()会在完成解析某个节点的时候调用,endDocument()会在完成整个XML解析的时候调用。
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();
}
}
修改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://192.168.30.175/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();
}
}
}
效果和Pull方式一样。