使用JACKSON解析JSON(HttpClient 3处理请求)

    在 上一篇中,我们使用了XStream来解析XML(HttpClient 4发送请求)制作了一个电子商务小应用,本篇我们来解析另外一种使用广泛的数据交换技术JSON。
    本文意在说明JACKSON组件解析JSON格式文本,并结合中央气象台的天气预报API来展示。天气预报的请求地址是:http://m.weather.com.cn/data/101070201.html,这个文本就是城市代码,101070201代表大连市,执行URL,得到一个返回文本,是JSON格式的,如下:
{"weatherinfo":{"city":"大连","city_en":"dalian","date_y":"2010年11月21日","date":"庚寅年十月十六","week":"星期日","fchh":"18","cityid":"101070201","temp1":"0℃~6℃","temp2":"1℃~9℃","temp3":"4℃~7℃","temp4":"-1℃~6℃","temp5":"-1℃~7℃","temp6":"-2℃~3℃","tempF1":"32℉~42.8℉","tempF2":"33.8℉~48.2℉","tempF3":"39.2℉~44.6℉","tempF4":"30.2℉~42.8℉","tempF5":"30.2℉~44.6℉","tempF6":"28.4℉~37.4℉","weather1":"多云转晴","weather2":"晴","weather3":"多云","weather4":"晴","weather5":"晴转多云","weather6":"多云转晴","img1":"1","img2":"0","img3":"0","img4":"99","img5":"1","img6":"99","img7":"0","img8":"99","img9":"0","img10":"1","img11":"1","img12":"0","img_single":"0","img_title1":"多云","img_title2":"晴","img_title3":"晴","img_title4":"晴","img_title5":"多云","img_title6":"多云","img_title7":"晴","img_title8":"晴","img_title9":"晴","img_title10":"多云","img_title11":"多云","img_title12":"晴","img_title_single":"晴","wind1":"西北风5-6级转北风4-5级","wind2":"北风转南风4-5级","wind3":"南风4-5级转北风5-6级","wind4":"北风5-6级转4-5级","wind5":"北风转南风4-5级","wind6":"南风转北风4-5级","fx1":"西北风","fx2":"北风","fl1":"5-6级转4-5级","fl2":"4-5级","fl3":"4-5级转5-6级","fl4":"5-6级转4-5级","fl5":"4-5级","fl6":"4-5级","index":"凉","index_d":"天气凉,建议着厚外套加毛衣等春秋服装。年老体弱者宜着大衣、呢外套加羊毛衫。","index48":"凉","index48_d":"天气凉,建议着厚外套加毛衣等春秋服装。年老体弱者宜着大衣、呢外套加羊毛衫。","index_uv":"中等","index48_uv":"弱","index_xc":"较适宜","index_tr":"一般","index_co":"较不舒适","st1":"2","st2":"-4","st3":"7","st4":"1","st5":"5","st6":"1","index_cl":"较不宜","index_ls":"基本适宜"}}

    字段言简意赅,城市名,英文名,日期,农历日期,星期,预报时间,城市代码,6个预报温度(华氏温度,摄氏温度),6个预报天气和风力,剩下的就是一些气象指数什么的了,本例我们仅仅拿当日的信息来说明。
    这段文本是我们直接通过浏览器请求来的,但是在程序中如何进行?需要借助网络api的帮助,可以直接使用java网络api,也可以使用第三方类库,这里我们使用Apache的Commons组件中的HttpClient3来说明。首先建立项目工程,使用Maven管理:
使用JACKSON解析JSON(HttpClient 3处理请求)
    引入的必要的依赖,如下:
使用JACKSON解析JSON(HttpClient 3处理请求)
    Weather类是封装必要数据的Bean,很简单,如下:
package weather;
public class Weather {
	private String city;// 城市名
	private String date;// 日期:yyyy年MM月d日
	private String lunarDate;// 农历日期/当日有
	private String week;// 星期
	private String fcTime;// 预报时间:24制小时数/当日有
	private String temperature;// 当日气温
	private String weather;// 天气
	private String wind;// 风力
	// 省略了getter和setter方法
	@Override
	public String toString() {
		return "Weather [city=" + city + ", date=" + date + ", fcTime="
				+ fcTime + ", lunarDate=" + lunarDate + ", temperature="
				+ temperature + ", weather=" + weather + ", week=" + week
				+ ", wind=" + wind + "]";
	}
}

    下面就是示例代码了,很简单:
package weather;
import java.util.HashMap;
import org.apache.commons.httpclient.HttpClient;
import org.apache.commons.httpclient.HttpStatus;
import org.apache.commons.httpclient.methods.GetMethod;
import org.codehaus.jackson.JsonFactory;
import org.codehaus.jackson.JsonParser;
import org.codehaus.jackson.JsonToken;
import org.codehaus.jackson.map.MappingJsonFactory;
/**
 * 天气预报服务,解析JSON
 * 
 * @author Nanlei
 * 
 */
public class Demo {
	private static String URL = "http://m.weather.com.cn/data/101070201.html";// 请求的地址
	private static HttpClient client;
	private static GetMethod getMethod;
	/**
	 * 静态块初始化所需对象
	 */
	static {
		client = new HttpClient();
		getMethod = new GetMethod(URL);
	}
	/**
	 * 获取获得的Json结果
	 * 
	 * @return
	 */
	public static String getJsonText() {
		String jsonText = null;
		try {
			int status = client.executeMethod(getMethod);
			if (status == HttpStatus.SC_OK) {// HTTP 200 OK
				jsonText = getMethod.getResponseBodyAsString();// 获取字符串形式的结果,建议使用流式结果
			}
		} catch (Exception e) {
			System.err.println(e);
		}
		return jsonText;
	}
	/**
	 * 处理Json结果并封装Bean
	 * 
	 * @param jsonText
	 * @return
	 */
	public static Weather getWeatherBean(String jsonText) {
		// Weather对象
		Weather weather = new Weather();
		try {
			JsonFactory jsonFactory = new MappingJsonFactory();
			// Json解析器
			JsonParser jsonParser = jsonFactory.createJsonParser(jsonText);
			// 跳到结果集的开始
			jsonParser.nextToken();
			// 接受结果的HashMap
			HashMap<String, String> map = new HashMap<String, String>();
			// while循环遍历Json结果
			while (jsonParser.nextToken() != JsonToken.END_OBJECT) {
				// 跳转到Value
				jsonParser.nextToken();
				// 将Json中的值装入Map中
				map.put(jsonParser.getCurrentName(), jsonParser.getText());
			}
			// 将数据封装
			weather.setCity(map.get("city"));
			weather.setDate(map.get("date_y"));
			weather.setLunarDate(map.get("date"));
			weather.setWeek(map.get("week"));
			weather.setFcTime(map.get("fchh"));
			weather.setTemperature(map.get("temp1"));
			weather.setWeather(map.get("weather1"));
			weather.setWind(map.get("wind1"));
		} catch (Exception e) {
			System.err.println(e);
		}
		return weather;
	}
	/**
	 * 主函数
	 * 
	 * @param args
	 */
	public static void main(String[] args) {
		System.out.println(getWeatherBean(getJsonText()));
	}
}

    其中提供了获取JSON和处理JSON结果的方法,用法都很简单,这样就可以得到天气预报的结果了:
Weather [city=大连, date=2010年11月21日, fcTime=18, lunarDate=庚寅年十月十六, temperature=0℃~6℃, weather=多云转晴, week=星期日, wind=西北风5-6级转北风4-5级]

    Jackson处理JSON非常简单,但要了解JSON文本的格式,这样就可以遍历了。到这里我们已经可以通过HttpClient和Jackson组件获取到了我们需要的信息,可以在应用中提供天气预报服务了,但是我们最开始使用的是一个城市代码,比如代表大连的101070201,这个数字是网站自己定义的,而没有其它任何实际含义(比如电话区号,行政代码等),那么要么我们在网站直接获得该城市的代码,否则就无法得到城市代码而去查询API了。其实网站也为我们获取城市代码提供了查询方法,如下:
访问http://m.weather.com.cn/data5/city.xml?level=0,(后面level参数可省略)得到一级列表(省、直辖市、自治区),结果用逗号隔开,id和城市名称使用竖线“|”隔开;结果示例如下:
01|北京,02|上海,03|天津,04|重庆,05|黑龙江,06|吉林,07|辽宁,08|内蒙古,09|河北,10|山西,11|陕西,12|山东,13|新疆,14|西藏,15|青海,16|甘肃,17|宁夏…

    之后我们继续访问http://m.weather.com.cn/data5/city02.xml?level=1,(后面level参数可省略)得到二级列表。其中02是一级省市的id,结果格式和上一层相同,示例如下(河南):
1801|郑州,1802|安阳,1803|新乡,1804|许昌,1805|平顶山,1806|信阳,1807|南阳,1808|开封,1809|洛阳,1810|商丘,1811|焦作,1812|鹤壁,1813|濮阳,1814|周口,1815|漯河,1816|驻马店,1817|三门峡,1818|济源

    继续访问http://m.weather.com.cn/data5/city1809.xml?level=3,(后面level参数可省略)得到后一级的id,是区域的id,示例如下(洛阳市):
180901|洛阳,180902|新安,180903|孟津,180904|宜阳,180905|洛宁,180906|伊川,180907|嵩县,180908|偃师,180909|栾川,180910|汝阳
   
    继续访问http://m.weather.com.cn/data5/city180908.xml,获得偃师市的代码:180908|101180908,那么通过101180908就可以获得偃师市的天气预报。
    知道这个规则后就可以在页面使用ajax方式请求地址获取到城市代码然后执行拿到天气预报,而在程序中,我们就需要一级一级的请求来获取代码了。
    下面就展示一下这个根据字符串地址获取天气的程序:
package weather;
import java.io.InputStream;
import java.util.HashMap;
import org.apache.commons.httpclient.HttpClient;
import org.apache.commons.httpclient.HttpStatus;
import org.apache.commons.httpclient.methods.GetMethod;
import org.apache.commons.lang.StringUtils;
import org.codehaus.jackson.JsonFactory;
import org.codehaus.jackson.JsonParser;
import org.codehaus.jackson.JsonToken;
import org.codehaus.jackson.map.MappingJsonFactory;
public class Demo2 {
	public static void main(String[] args) throws Exception {
		String name = "河南洛阳偃师";
		StringBuffer cityCode = new StringBuffer();
		String provinceName = null;// 省份名称
		String cityName = null;// 城市名称
		String districtName = null;// 区县名称
		if (name.length() >= 6) {
			provinceName = name.substring(0, 2);
			cityName = name.substring(2);
			districtName = name.substring(4);
		} else if (name.length() == 5) {
			provinceName = name.substring(0, 2);
			cityName = name.substring(2);
			districtName = "";
		} else if (name.length() <= 4) {
			provinceName = name.substring(0, 2);
			cityName = "";
			districtName = name.substring(2);
		} else {
			System.out.println("城市名称错误");
			System.exit(0);
		}
		// 下面开始根据这三个名称来获取weather.com.cn的城市代码
		HttpClient client = new HttpClient();
		GetMethod get = new GetMethod("http://m.weather.com.cn/data5/city.xml");// 获取省份
		String provinceStr = null;
		if (client.executeMethod(get) == HttpStatus.SC_OK) {
			provinceStr = get.getResponseBodyAsString();
		}
		String[] provinceArray = provinceStr.split(",");
		String provinceCode = null;
		for (int i = 0; i < provinceArray.length; i++) {// 遍历省份
			if (provinceArray[i].substring(3).indexOf(provinceName) != -1) {
				provinceCode = provinceArray[i].substring(0, 2);
				cityCode.append(provinceCode);// 获得省份编码
			}
		}
		// 城市代码
		if (Integer.parseInt(provinceCode) <= 4
				|| Integer.parseInt(provinceCode) >= 32) {// 直辖市和港澳台
			get = new GetMethod("http://m.weather.com.cn/data5/city"
					+ provinceCode + "01.xml");
			if (client.executeMethod(get) == HttpStatus.SC_OK) {
				String[] districtArray = get.getResponseBodyAsString().split(
						",");
				for (int i = 0; i < districtArray.length; i++) {
					if (districtName.indexOf(districtArray[i].substring(7)) != -1) {
						cityCode.append("01").append(
								districtArray[i].substring(4, 6));// 直辖市或港澳台代码获取结束,6位
					}
				}
				if (cityCode.length() < 6) {
					cityCode.append("0101");
				}
			}
		} else {// 省市
			get = new GetMethod("http://m.weather.com.cn/data5/city"
					+ provinceCode + ".xml");
			String[] cityArray = null;
			if (client.executeMethod(get) == HttpStatus.SC_OK) {
				cityArray = get.getResponseBodyAsString().split(",");// 获取到了城市数组
			}
			if (StringUtils.isEmpty(cityName)) {// 没有取到城市名
				for (int i = 0; i < cityArray.length; i++) {
					if (districtName.indexOf(cityArray[i].substring(5)) != -1) {
						cityCode.append(cityArray[i].substring(2, 4)).append(
								"01");// 省直管地区代码获取结束,6位
					}
				}
			} else if (StringUtils.isNotEmpty(cityName)) {// 取得了城市名
				for (int i = 0; i < cityArray.length; i++) {
					if (cityName.indexOf(cityArray[i].substring(5)) != -1) {
						cityCode.append(cityArray[i].substring(2, 4));
					}
				}
				get = new GetMethod("http://m.weather.com.cn/data5/city"
						+ cityCode.toString() + ".xml");// 请求4位城市代码
				String[] districtArray = null;
				if (client.executeMethod(get) == HttpStatus.SC_OK) {
					districtArray = get.getResponseBodyAsString().split(",");
				}
				for (int i = 0; i < districtArray.length; i++) {
					if (districtName.indexOf(districtArray[i].substring(7)) != -1) {
						cityCode.append(districtArray[i].substring(4, 6));
					}
				}
				if (cityCode.toString().length() == 4) {
					cityCode.append("01");
				}
			}
		}
		if (cityCode.length() == 6) {
			// 获取城市代码结束,获取天气页面代号
			GetMethod weatherCodeGet = new GetMethod(
					"http://m.weather.com.cn/data5/city" + cityCode.toString()
							+ ".xml");
			String weatherCode = "";
			if (client.executeMethod(weatherCodeGet) == HttpStatus.SC_OK) {
				weatherCode = weatherCodeGet.getResponseBodyAsString()
						.substring(7);
			}
			GetMethod weatherGet = new GetMethod(
					"http://m.weather.com.cn/data/" + weatherCode + ".html");
			InputStream weatherInfo = null;
			if (client.executeMethod(weatherGet) == HttpStatus.SC_OK) {
				weatherInfo = weatherGet.getResponseBodyAsStream();
				//
				HashMap<String, String> map = new HashMap<String, String> ();
				JsonFactory jsonFactory = new MappingJsonFactory();
				JsonParser jsonParser = jsonFactory
						.createJsonParser(weatherInfo);
				// 跳到结果集的开始
				jsonParser.nextToken();
				while (jsonParser.nextToken() != JsonToken.END_OBJECT) {
					// 跳转到Value
					jsonParser.nextToken();
					map.put(jsonParser.getCurrentName(), jsonParser.getText());
				}
				// 封装VO,向页面传递数据
				Weather todayWeather = new Weather();
				// 当日数据
				todayWeather.setCity(map.get("city"));
				todayWeather.setDate(map.get("date_y"));
				todayWeather.setLunarDate(map.get("date"));
				todayWeather.setWeek(map.get("week"));
				todayWeather.setFcTime(map.get("fchh"));
				todayWeather.setTemperature(map.get("temp1"));
				todayWeather.setWeather(map.get("weather1"));
				todayWeather.setWind(map.get("wind1"));
				System.out.println(todayWeather);
			}
		} else {
			System.out.println("没有获取到天气信息");
		}
	}
}

    程序中对城市名称做了一点限制,直辖市等级为省,没有城市名,有城市名可以不用输入到区县,又如:
String name = "河南洛阳";
String name = "河南洛阳涧西";

    这都是可以的,程序中对城市名称的处理可能不很准确,真实环境中这省市区这三个字段应该分别提供,截字符串可能会有偏差,本程序测试了大多数情况,是可行的。
    本文系作者的实践探索,欢迎交流,希望对使用者有用。

你可能感兴趣的:(apache,maven,json,xml,网络应用)