最近在编写一个基于Android 2.1 的手机应用程序,其中的一个功能是利用Google 的地图API接口实现足迹追踪,整个程序设计大概分为三个部分,UI设计、GoogleMapAPI接口调用以及后台Service所做的数据的采集和传输以及和服务器的通讯。
Android的UI设计和JAVA、MFC、C#.NET有些不同,毕竟是手持设备,硬件资源的限制要求它用尽量轻便的代码框架去完成功能,Android的用户界面是用XML来进行布局和管理的,支持直接拖拽,但是效果并不是很好。它的主活动界面是在main.xml中编写的,在这里可以定义一些按钮啊、文本框什么的。GoogleMapAPI的用法网上有很多教程,首先要去申请一个KEY,然后才能去调用API得到想要的数据,我们这里要获得GPS的实时数据,所以要用到 LocationListeninger 和 LocationManager 两个类,绘制地图可以调用Mapview类,里面有很多操作地图的方法,类似放大缩小、拖拽描点等都可以调用函数直接实现。
主要来说一下后台服务的编写吧,在Android中Service是用来进行后台数据处理的东西,类似于Linux下的后台进程,就是当前活动创建的Service在这个活动窗口退出后Service还是继续运行的,我们要对用户的行动进行跟踪,就要长时间采集GPS发送过来的地理位置信息,这样的东西写成一个Service再合适不过了。自己编写的Service要继承系统的Service类,然后Override其中的方法。
package server.track;
import java.util.Calendar;
import android.app.Service;
import android.content.Context;
import android.content.Intent;
import android.location.Location;
import android.location.LocationListener;
import android.location.LocationManager;
import android.os.Bundle;
import android.os.IBinder;
import android.util.Log;
import android.widget.Toast;
public class Track extends Service {
private static final String TAG = "Track";
private LocationManager lm;
private LocationListener locationListener;
final SDRW filerecord = new SDRW();
private String record ;
private String length;
private int headposition;
@Override
public IBinder onBind(Intent arg0) {
// Log.d(TAG, "onBind.");
return null;
}
public void onStart(Intent intent, int startId) {
Log.d(TAG, "onStart.");
Toast.makeText(getApplicationContext(), "启动服务",Toast.LENGTH_SHORT).show();
super.onStart(intent, startId);
// startDb();
// Bundle extras = intent.getExtras();
// if (extras != null) {
// track_id = extras.getInt(LocateDbAdapter.TRACKID);
// }
// Log.d(TAG, "track_id =" + track_id);
// ---use the LocationManager class to obtain GPS locations---
Calendar calendar = Calendar.getInstance();
headposition = 5;
record = "head: \r\n";
filerecord.write(record);
record = "user:"+"Yastand\r\n"+"date:"+calendar.get(Calendar.YEAR) + "-" +calendar.get(Calendar.MONTH) + "-" + calendar.get(Calendar.DAY_OF_MONTH) + " "
+ calendar.get(Calendar.HOUR_OF_DAY) + ":"
+ calendar.get(Calendar.MINUTE) + ":" + calendar.get(Calendar.SECOND)+"\r\n";
filerecord.write(record);
length = String.valueOf(record.length())+" ";
filerecord.writehead(length,headposition);
headposition += length.length();
lm = (LocationManager) getSystemService(Context.LOCATION_SERVICE);
locationListener = new MyLocationListener();
lm.requestLocationUpdates(LocationManager.GPS_PROVIDER, 0, 0,locationListener);
}
public void onDestroy() {
Toast.makeText(this, "service done", Toast.LENGTH_SHORT).show();
super.onDestroy();
lm.removeUpdates(locationListener);
// stopService(new Intent("Context.LOCATION_SERVICE"));
stopSelf();
}
protected class MyLocationListener implements LocationListener {
@Override
public void onLocationChanged(Location loc) {
Log.d(TAG, "MyLocationListener::onLocationChanged..");
/* if (loc != null) {
// //////////
if(mlcDbHelper == null){
mlcDbHelper.open();
}
mlcDbHelper.createLocate(track_id, loc.getLongitude(),loc.getLatitude(), loc.getAltitude());
}
*/
record = "GPS:"+" lon:"+String.valueOf(loc.getLongitude())+" lat:"+String.valueOf(loc.getLatitude())+" alt:"+String.valueOf(loc.getAltitude())+"\r\n";
if (loc != null)
{
// filepoint.write(edit1.getText().toString());
filerecord.write(record);
}
}
@Override
public void onProviderDisabled(String provider) {
Toast.makeText(
getBaseContext(),
"ProviderDisabled.",
Toast.LENGTH_SHORT).show(); }
@Override
public void onProviderEnabled(String provider) {
Toast.makeText(
getBaseContext(),
"ProviderEnabled,provider:"+provider,
Toast.LENGTH_SHORT).show(); }
@Override
public void onStatusChanged(String provider, int status, Bundle extras) {
// TODO Auto-generated method stub
}
}
}
要开启这个Service可以用
Intent i = new Intent("server.track.START_TRACK_SERVICE");
startService(i);
执行这句话后,Service类首先去找Create函数,再运行Onstart函数,一个Service就成功的运行了,但是退出当前程序这个操作对这个后台运行的Service并没有什么影响,那么怎样结束这个Service呢?可以这样
stopService(new Intent("server.track.START_TRACK_SERVICE"));
执行这句话,系统就会调用Service实例的onDestroy()函数,到这里本以为万事大吉了,结果发现调用stopService之后GPS发送回来的数据还是源源不断的写入到文件中,原来在onDestroy()函数中必须显示的把Service实例中调用的线程都结束之后才能停止,在我们的Service中调用的LocationLinstener是这问题的关键,必须在onDestroy()中结束它运行的线程
public void onDestroy() {
Toast.makeText(this, "service done", Toast.LENGTH_SHORT).show();
super.onDestroy();
lm.removeUpdates(locationListener);
// stopService(new Intent("Context.LOCATION_SERVICE"));
stopSelf();
}
ok了,调用这个函数之后后台运行的Service就自己终止了。