android软件开发--天气预报

    这两天开发了一个天气预报软件。基本上用到了不少以前学习的内容,而后发现,只有实践,才能发现更加多的问题,也才能了解其中的原理,甚至能够辨别你之前的知识是不是正确。 java

    原本我想把源码发上来的,可是发现没有添加附件的功能。只有经过代码分享了。http://www.oschina.net/code/snippet_1016021_21811 android

    界面比较简单,主要是实现功能。 web


    程序说明 数据库

    一、进入程序以后,能够经过点击城市的名字来设置当前城市。(一开始默认为广州) 编程

    二、进入设置城市界面以后,省市的选择为级联下拉列表。能够选择点击保存按钮,则会返回主界面,而且更新当前城市为你所选的值。也能够选择取消,则直接返回主界面。 数组

    三、点击Menu,能够进入设置界面对查询天气以及附带信息进行选择,或者能够选择退出程序。 网络

    四、点击查询按钮完成查询。 app

    学习要点 ide

1、android工程正确导入jar包(MyEclipse下) 学习

   这个工程要用到SOAP技术,因此要导入ksoap2-android-assembly-3.0.0-jar-with-dependencies。根据之前的作法,通常都是直接新建一个lib目录,把jar包复制进去,而后右键,接着Build path。可是在android工程中,这样作是不正确的。会出现红叉或者叹号。

     正确的作法是:

    一、右键工程, Build path

    二、点击“Add Libraries”

    三、选择“User library”,点击“下一步”

    四、点击“User librarys”按钮在出现的界面中点击“New..”按钮。在弹出的界面中随便起一个名字,点击“肯定”

    五、点击“Add jars”按钮选择第三方jar包,点击“肯定”完成操做。这样该jar包会被一块儿打包到apk中。

2、省市下拉列表实现二级级联

    首先将省市信息以<string-array>的形式保存到名为arrays.xml的文件中(我记得貌似必定要把文件名取为arrays.xml)。其中,name属性能够理解为数组名和ID名。这里要注意:省份的顺序要与对应拥有的城市顺序一致。即台湾为最后最后一个省,那么它的对应城市也要写在最后。(下面会有解释)

<resources>
    <string-array name="provinces">
        <item>--请选择--</item>
        <item>北京</item>
        <item>天津</item>
        ...
        <item>海南省</item>
        <item>台湾省</item>
    </string-array>

    <string-array name="beijing_array">
        <item>北京</item>
    </string-array>
    ...
    <string-array name="taiwan_array">
        <item>台北</item>
        <item>高雄</item>
        <item>台中</item>
    </string-array>
</resources>
    而后,在选择城市界面对应的Activity中,经过下面代码将省份列表显示。其中R.array.provinces就是咱们上面定义的name属性值。


ArrayAdapter<CharSequence> adapter = ArrayAdapter.createFromResource(this,R.array.provinces,android.R.layout.simple_spinner_item);
provinceSpinner.setAdapter(adapter);
    接着,对省份下拉列表进行监听。这里有一个比较麻烦的地方,由于当你选择不一样的省份的时候,须要显示该省份对应的城市。面对那么多的省份,若是咱们经过if或者switch来操做的话,使得代码很冗长,也难以维护。我一开始想在网上找答案,可是没有发现好的想法,甚至连相关的实例都没有。不过,后来我发现这里是经过R.array.name这种形式来显示下拉列内容的。因而,我经过观察R文件,发现了必定的规律。R文件中的array类的int属性值,是根据咱们写入顺序,从0x7f050000开始,逐个+1造成的。即


public static final int provinces=0x7f050000;
public static final int beijing_array=0x7f050001;
public static final int tianjin_array=0x7f050002;
    可能R文件中没有按照此顺序排列,不过,不影响这一性质。因此我就想到了只要城市数组的顺序与省份一一对应(上面提到过),就能够经过所选省份的position,跟ID初始值 0x7f050000相加,得出所属城市的数组。具体看看代码


provinceSpinner.setOnItemSelectedListener(new OnItemSelectedListener() {
	public void onItemSelected(AdapterView<?> parent, View view,
					int position, long id) {
		if(position != 0){//选择了省份,position=0时,为“--请选择--”
			/*这是一个小技巧
			*0x7f050000为R文件中省份数组对应的id值,只要加上position,便可得到对应选项(省份)的城市
			*若是不是用这个方法,可能就要用一大堆的判断语句来级联城市
			*/
			int cityID = 0x7f050000 + position;
					
			ArrayAdapter<CharSequence> adapter = ArrayAdapter.createFromResource(getApplicationContext(), cityID, android.R.layout.simple_spinner_item);
			citySpinner.setAdapter(adapter);
		}
	}
	public void onNothingSelected(AdapterView<?> parent) {
	}
});
    我不知道还有没有更加方便方法,不过个人这个方法在我这边确实可行。造成界面为上方图二。


3、SQLite保存城市数据

   使用SQLite而不使用Intent传递参数,是由于当用户下次打开程序时,当前城市应该为TA最后一次的选择。关于SQLite的使用,网上有不少文章,好比:http://52android.blog.51cto.com/2554429/478368 以前也学习过一些,但发现看懂跟实际编程仍是有很大差距的。特别是数据库的关闭以及Cursor的关闭,出现了好些问题。在这一块要仔细一点。

4、PreferenceActivity做为设置界面

    参照Android系统的设置,用PreferenceActivity来对系统进行信息配置和管理。这里我也采用PreferenceActivity做为设置界面。(上方图三)

    首先,编写xml文件。PreferenceCategory:类别(用于分组)。key:惟一标识(获取信息时使用)。title:显示标题。summary:小标题。还有defaultValue:默认值。我这里值用到了CheckBoxPreference,它还有EditTextPreperenceRingtonePreference,ListPreference,Preference等

<?xml version="1.0" encoding="utf-8"?>
<PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android" >
    <PreferenceCategory android:title="信息选择" >
        <CheckBoxPreference
            android:key="threeDay"
            android:summary="今明后三天的天气预报,若是不选,则只有当天的天气"
            android:title="三天预报" />
        <CheckBoxPreference
            android:key="cityInfo"
            android:summary="关于当前城市的简要介绍"
            android:title="城市简介" />
    </PreferenceCategory>
</PreferenceScreen>
    而后,新建Activity继承 PreferenceActivity ,重写onCreate方法,经过 addPreferencesFromResource(R.xml.xx); 加载Preference。

public class SetupActivity extends PreferenceActivity {
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		addPreferencesFromResource(R.xml.setup);
	}
}
    最后,获取preference数据。可 经过下面三种方式:

    一、getPreferences():能够获取同一activity中的preference;

    二、getSharedPreferences():能够获取应用级别的preferences,即封装在同一app中,使用SharePreferences prefs = getSharedPreferences(packName+name ,0)

    三、getDefaultSharedPreferences():经过Android的管理器来获取其所管理的preferences。

   因为这里不是同一个Activity,因此不能使用getPreferences()。我这里只有一个preference,所以使用PreferenceManager.getDefaultSharedPreferences(this);来获取较方便。

5、经过WebService获取天气信息

   WebService获取天气的网址为:http://www.webxml.com.cn/webservices/weatherwebservice.asmx 上面有较为详细的介绍,以及相关图标的下载。

    我这里经过SOAP技术获取天气信息。

// 保存获取到的信息
SoapObject detail = null;

// 1.实例化SoapObject对象
SoapObject soapObject = new SoapObject(NAMESPACE, METHOD_NAME);
// 2.若是方法须要参数,设置参数
soapObject.addProperty("theCityName", cityName);

// 3.设置Soap的请求信息,得到序列化envelope,参数部分为Soap协议的版本号
SoapSerializationEnvelope envelope = new SoapSerializationEnvelope(
				SoapEnvelope.VER11);
envelope.bodyOut = soapObject;
envelope.dotNet = true;
envelope.setOutputSoapObject(soapObject);

// 4.构建传输对象
int timeout = 10000;// 设置超时为10秒
MyAndroidHttpTransport httpTransportSE = new MyAndroidHttpTransport(URL, timeout);
httpTransportSE.debug = true;
// 5.访问WebService,第一个参数为命名空间 + 方法名,第二个参数为Envelope对象
httpTransportSE.call(SOAP_ACTION, envelope);

detail = (SoapObject) envelope.getResponse();// 获取详细天气信息
if (detail != null) {// 当前城市有天气信息
	return parseWeather(detail);//解析天气
}
    这里要注意一下,代码18行 MyAndroidHttpTransport为继承了HttpTransportSE的内部类。虽然ksoap2版本中的HttpTransportSE已经能够设置timeout(超时时间),可是运行后发现没有效果。查找资料后,才知道HttpTransportSE的源码中并无把timeout做为参数传递给ServiceConnectionSE。所以咱们须要建立一个类,使得timeout起做用。

class MyAndroidHttpTransport extends HttpTransportSE {
	private int timeout = 20000; // 默认超时时间为20s
	public MyAndroidHttpTransport(String url) {
		super(url);
	}
	public MyAndroidHttpTransport(String url, int timeout) {
		super(url);
		this.timeout = timeout;
	}
	//此方法使得超时有效
	public ServiceConnection getServiceConnection() throws IOException {
	ServiceConnectionSE serviceConnection = new ServiceConnectionSE(this.url,timeout);
		return serviceConnection;
	}
}

6、开启service处理联网操做

    我一开始直接在MainActivity中联网加载天气信息,不过,发如今点击查询按钮到信息显示出来以前,界面时卡死的,并且容易出现ANR(程序没法响应异常)。因此我就想经过service来处理。在个人记忆里,service是做为后台运行的,并且网上大部分的文章,都有提到耗时操做要使用service。能够当我真的这么作的时候,我才发现,界面依旧卡住了。

    我当时就纳闷了,怎么会这样呢?难道service开启后,会在主线程运行?查了资料后发现,果然是如此。这是才发现,在service里面还要开启一个线程来执行耗时操做。咦?若是是这样的话,那我还不如直接在MainActivity中另开线程,这样不是更加方便吗,还省去了Activity与Service之间的通讯的麻烦。网上有些大神说service有它的生命周期,更加方便管理,以及还有其余一些优势。恩恩,确实吧。不过就个人程序而言,我以为直接在MainActivity中另开线程获取天气,而后经过Handler更新UI显示天气可能会更加方便。(我最后仍是使用了service,由于能够学到更多的东西)

    如今来讲建立service的过程

     一、新建类继承Service;

    二、必须重写onBind方法(若是你经过bindService方法启动service,则在这个方法内执行操做

    三、重写onStart方法(因为本程序中,每次点击查询按钮,service就要进行联网操做,所以我经过startService方法启动service,则每次startService,都会执行onStart方法。注意:在service中止前,onCreate只会执行一次

    四、在AndroidManifest.xml文件中添加

<service android:name="className" >
    <intent-filter >
       <action android:name="serviceName" />
    </intent-filter>
</service>
    className为类名全称:如vaint.wyt .service.WeatherService。若是跟MainActivity在同一个包,能够直接写 .WeatherService。

    serviceName为startService(new Intent(String action))的action,bindService相似。

    五、在须要启动service的地方,添加一下代码

Intent intent =new Intent("WeatherService");
//传递数据,能够由onStart接收
intent.putExtra("city", city);
this.startService(intent);

    六、若是是经过bindService启动service,则能够不执行unbindService。由于只要程序退出,service也将被摧毁。可是,若是是经过startService启动service,则必须经过stopService将其中止,不然即便程序退出,service依旧在运行。咱们能够在MainActivity的onDestroy中执行stopService。

protected void onDestroy() {
	//中止service
	stopService(new Intent("WeatherService"));
	super.onDestroy();
}

7、用BroadcastReceiver实现从service到Activity的通讯

    这只是其中一种方法而已。

    一、建立广播接收器。(能够直接在MainActivity中做为内部类建立)重写onReceive方法,接收从service传递过来的天气信息。

//定义一个广播接收器,用于接收Service得到的天气信息
class MyBroadcastRecever extends BroadcastReceiver{
	@Override
	public void onReceive(Context context, Intent intent) {
		String[] weatherInfo = intent.getStringArrayExtra("weather");
		if(weatherInfo==null){
			Toast.makeText(MainActivity.this, "没有当前城市的天气信息", 1000).show();
		}else if(weatherInfo.length==1){//即weatherInfo = new String[]{"timeOut"};
			Toast.makeText(MainActivity.this, "链接超时,请检查网络", 1000).show();
		}else{
			showWeather(weatherInfo);
		}
	}
}
    二、经过代码动态注册广播接收器。(也能够在AndroidManifest中添加<receiver>属性

//注册广播接收器
IntentFilter filter = new IntentFilter();
myBroadcastRecever = new MyBroadcastRecever();
//设置接收广播的类型,这里要和Service里设置的类型匹配,还能够在AndroidManifest.xml文件中注册 
//BROADCAST_ACTION=“某个自定义字符串”。若是有多个广播,则要惟一
filter.addAction(BROADCAST_ACTION);
registerReceiver(myBroadcastRecever, filter);
    三、经过广播发送消息

Intent i = new Intent();
i.putExtra("weather", weather);
//BROADCAST_ACTION与注册时的字符串一致
i.setAction(BROADCAST_ACTION);
sendBroadcast(i);
相关文章
相关标签/搜索