和之前的内容一样,Android简易打地鼠,也是通过小软件的方式来巩固android知识
先看下效果,其实主要就一张图,但是设计到的知识点还是很多的
应用介绍:
一个相对比较简单的天气预报项目,涉及两个界面,主界面用于显示当日天气和后5天的天气情况,点击搜索会进入添加城市界面, 该界面仅有 一个输入框,输入城市后再次点击搜索会返回到主界面,并根据输入的城市信息进行数据更新。点击更新则会刷新当 前数据,由于应用目前没有添加本地保存的功能,如果添加的话,应该会采用SharedPreferences将选择的城市对应的编码保存到本地(之前的Android简易老虎机就是使用这个保存游戏数据),这样在每次打开软件的时候先读取本地保存信息并进行相应更新,如果没读取到,则默认为显示武汉的天气
涉及到的知识点:
首先是HTTP请求,因为需要从网络获取天气数据,采用的是HttpURLConnection,同时将获取到的天气信息转化为字符串 ,小结如下
1.new一个URL的对象,并传入目标网址
URL url = new URL("http://www.baidu.com");
2.使用URL对象开启连接,并转型为HttpURLConnection
HttpURLConnection connection = (HtttpURLConnection)url.openConnection();
3.设置HTTP的请求方法GET或者POST
connection.setRequestMethod("GET");
4.设置连接超时、读取超时的时间等
connection.setConnectionTimeout(8000); connection.setConnectionReadTimeout(8000);
5.连接服务器 connection.connect();
6.开启输入流,获得服务器返回的信息
InputStream in = connection.getInputStream();
7.使用disconnection()方法关闭HTTP连接 connection.disconection();
**使用JSON数据解析,使用JSONObject类将获取到的JSON型数据进行解析,得到所需的天气信息 **
json数据的格式:
{ }中包含的即为JSON对象, [ ]代表JSON对象的数组 ,里面可以存放一系列JSON对 象,如: [{ "name":"tom", "age":"20" }]
一般可能在 [ ] 前面还有一个字符串 比如 information :[ ]
在解析时,要到达 [ ] 里面开始解析
1.JSONObject rootObject = new JSONObject(dataStr); // 得到数据的根对象
2.JSONObject json = rootObject.getJSONObject("information"); //取出 information中的值
3.然后直接使用 json.getString(" key "); //key为对应数据的键,就可以得到对应的数据
**在子线程中更新界面天气信息,界面中间当日天气显示较为详细,后5天的天气放在一个HorizontalScrollView中,实现水平滚 动查看天气信息 **
**更新是为ImageView设置监听器,点击后会启动一个动画,即更新图标的转动,该处是采用了Animation来实现图标的转动动画, 动画开启后使用一个Handler对象的延时操作,在1s后关闭该动画,也就是说每次点击,理论上图标都会转动1s然后停止 **
new Handler().postDelayed(new Runnable(){ @Override public void run(){ //要执行的操作 } },delayedTime)
点击搜索后进入添加城市界面,目前只能手动输入城市名,采用的是在程序内部放置了包含城市名和对应的城市编号的XML文件, 所以自然需要使用到XML解析,本次使用的是pull解析,在进入添加城市界面后,系统会首先获取到所有的城市信息,即解析XML 数据,解析到的数据是采用键值对的方式放入到Map中进行存储的,城市名为键,城市编号为值,在输入正确的城市后,点击搜 索会返回到主界面,同时根据输入的城市名(键),获取到对应的城市编号(值),该处用到了回值传递,在返回到主界面的同 时则会把城市编码传递过来,并根据获取的城市编码重新通过HTTP连接获取到新的数据,并进行更新
使用pull解析
1.首先构建一个pull解析器工厂
XmlPullParserFactory factory =XmlPullParserFactory.newInstance();
2.从工厂中构建一个pull解析器
XmlPullParser parse= factory.newInstance();
3.设置输入信息 parse.setInput(is, "UTF-8");
其中is是一个输入流对象,当然可以通过getAssets().open("file");
打开资产文件中的文件以获得一个输入流对象
4.定义一个事件类型int eventType = parse.getEventType();
5.开始遍历XML,得设置一个while循环,只要事件类型不是结束的事件类型,那么一直读取
while(eventType != XmlPullParser.END_DOCUMENT){ }
6.在上面的循环中开始读取,得设置一个判断,判断当前读取的标签是否是开始标签
if(eventType == XmlPullPaser.START_TAG){ }
7.当上诉判断的是开始标签,即说明已经开始读取正文部分,此时应该定义一个字符串用于表示读取的标签的标识
String tag = parse.getName();
这时需要设置一个判断,用于确认标签的标识是包裹所需数据的标签
if(tag.equals("key")){ //假设此处的key是XML中的包裹所需数据的那个标签 String key = parse.nextText(); //由名字也可以看出来,nextText(),该标签下一位的文本,也就是数据了 }
8.可以把parse理解为一个游标吧,注意上面的问题,判断如果是包裹所需数据的标签,那么读取它下一位的文本数据,但是一般不会只有一个数据,所以还是得继续读取,为了循环读取,那么在当前数据读取后,游标应该往下再移动一位,同时更新eventType 即
paese.next(); eventType = parser.getEventType();
总代码:
XmlPullParserFactory factory = XmlPullParserFactory.newInstance(); XmlPullParser parse= factory.newInstance(); parse.setInput(is, "UTF-8"); int eventType = parse.getEventType(); while(eventType != XmlPullParser.END_DOCUMENT){ if(eventType == XmlPullPaser.START_TAG){ String tag = parse.getName(); if(tag.equals("key")){ String key = parse.nextText(); } } paese.next(); eventType = parser.getEventType(); }
遇到的问题:在建立assets文件夹时,把它建在了res目录下,结果总是读取不到XML文件对应的信息,以为是解析代码写错了,但是系统并没报任何错误,折腾好久才发现是这个问题,应该和res放置在同等位置
小结:这次的相比之前的小软件,主要就是涉及到了网络和数据解析,源码放在GitHubWeather ,当然了,肯定还是有不少缺陷的,希望有兴趣的童鞋一起改正一起进步哈