最近在编写一个基于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) {
-
- return null;
- }
-
- public void onStart(Intent intent, int startId) {
- Log.d(TAG, "onStart.");
- Toast.makeText(getApplicationContext(), "启动服务",Toast.LENGTH_SHORT).show();
- super.onStart(intent, startId);
-
-
-
-
-
-
-
- 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);
-
- stopSelf();
- }
-
- protected class MyLocationListener implements LocationListener {
-
- @Override
- public void onLocationChanged(Location loc) {
- Log.d(TAG, "MyLocationListener::onLocationChanged..");
-
-
-
-
-
-
-
-
- record = "GPS:"+" lon:"+String.valueOf(loc.getLongitude())+" lat:"+String.valueOf(loc.getLatitude())+" alt:"+String.valueOf(loc.getAltitude())+"\r\n";
- if (loc != null)
- {
-
- 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) {
-
- }
- }
-
-
- }
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);
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"));
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);
-
- stopSelf();
- }
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就自己终止了。
Android SERVICE后台服务进程的自启动和保持
Service组件在android开发中经常遇到,其经常作为后台服务,需要始终保持运行,负责处理一些必要(见不得人)的任务。而一些安全软件,如360等,会有结束进程的功能,如果不做Service的保持,就会被其杀掉。
如何保持Service的运行状态是现在要说明的,核心就是利用ANDROID的系统广播,这一不会被其他软件影响的常驻程序触发自己的程序检查Service的运行状态,如果被杀掉,就再起来。
我利用的系统广播是Intent.ACTION_TIME_TICK,这个广播每分钟发送一次,我们可以每分钟检查一次Service的运行状态,如果已经被结束了,就重新启动Service。
下边就是具体的代码和注意事项了:
1、 Intent.ACTION_TIME_TICK的使用
我们知道广播的注册有静态注册和动态注册,但此系统广播只能通过动态注册的方式使用。即你不能通过在manifest.xml里注册的方式接收到这个广播,只能在代码里通过registerReceiver()方法注册。
在ThisApp extends Application 里注册广播:
- IntentFilter filter = newIntentFilter(Intent.ACTION_TIME_TICK);
- MyBroadcastReceiver receiver = new MyBroadcastReceiver();
- registerReceiver(receiver, filter);
在广播接收器MyBroadcastReceiver extends BroadcastReceiver的onReceive里
- if (intent.getAction().equals(Intent.ACTION_TIME_TICK)) {
-
-
-
- }
2、Service的检查与启动
- boolean isServiceRunning = false;
- ActivityManager manager = (ActivityManager)ThisApp.getContext().getSystemService(Context.ACTIVITY_SERVICE);
- for (RunningServiceInfo service :manager.getRunningServices(Integer.MAX_VALUE)) {
- if("so.xxxx.WidgetUpdateService".equals(service.service.getClassName()))
-
- {
- isServiceRunning = true;
- }
-
- }
- if (!isServiceRunning) {
- Intent i = new Intent(context, WidgetUpdateService.class);
- context.startService(i);
- }
另一个话题,Service的开机启动。
实现和上边的类似,也是通过监控开机的系统广播来启动Service。但其实你做了上边的检查也就不会做开机启动了,因为过一两分钟就会通过上边的程序启动Service了。代码如下:
- if (intent.getAction().equals(Intent.ACTION_BOOT_COMPLETED)) {
- Intent i = new Intent(context, LogService.class);
- context.startService(i);
-
posted on 2015-01-26 11:41
游子 阅读(1542)
评论(0) 编辑 收藏 引用 所属分类:
软件