Android后台运行定时器,方便我们运行定位跟踪等任务需求。 以下简要说明实现Android后台定时器的要点, 文章末尾能够下载到project代码,可直接编译运行。

AndroidManifest.xml 文件内容例如以下:

<?

xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.routing.videocamera"
android:versionCode="1"
android:versionName="1.0" > <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> <uses-sdk
android:minSdkVersion="14"
android:targetSdkVersion="16" /> <application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name="com.routing.videocamera.Video"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<service android:enabled="true" android:name=".TimerService" />
<receiver android:name=".AutoReceiver" android:label="@string/app_name">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</receiver> </application> </manifest>

当中关键的代码是赋予项目 RECEIVE_BOOT_COMPLETED 权限 :

<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />

Video.java 文件的内容例如以下 :

package com.routing.videocamera;

public class Video extends Activity {

	@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_video); Intent intent=new Intent(this,AutoReceiver.class);
intent.setAction("VIDEO_TIMER");
PendingIntent sender = PendingIntent.getBroadcast(this, 0, intent, 0);
AlarmManager am = (AlarmManager)getSystemService(ALARM_SERVICE);
am.setRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP, SystemClock.elapsedRealtime(), 10*1000, sender); Button btnStart = (Button)findViewById(R.id.buttonSave);
btnStart.setOnClickListener(clickListener); Context ctx = Video.this;
SharedPreferences sp = ctx.getSharedPreferences("VIDEO", MODE_PRIVATE);
//存入数据 Editor editor = sp.edit(); String serverAddr = sp.getString("ServerAddr", "NULL");
String cameraName = sp.getString("CameraName", "NULL");
int cameraID = sp.getInt("CameraID", 0);
int cameraPort = sp.getInt("CameraPort", 0); EditText editServerAddr = (EditText)findViewById(R.id.editServerAddr);
editServerAddr.setText(serverAddr); EditText editCameraName = (EditText)findViewById(R.id.editCameraName);
editCameraName.setText(cameraName); EditText editCameraID = (EditText)findViewById(R.id.editCameraID);
editCameraID.setText(Integer.toString(cameraID)); EditText editCameraPort = (EditText)findViewById(R.id.editCameraPort);
editCameraPort.setText(Integer.toString(cameraPort)); } private void saveSetting ()
{
EditText editServerAddr = (EditText)findViewById(R.id.editServerAddr);
String serverAddr = editServerAddr.getText().toString(); EditText editCameraName = (EditText)findViewById(R.id.editCameraName);
String cameraName = editCameraName.getText().toString(); EditText editCameraID = (EditText)findViewById(R.id.editCameraID);
int cameraID = Integer.parseInt(editCameraID.getText().toString()); EditText editCameraPort = (EditText)findViewById(R.id.editCameraPort);
int cameraPort = Integer.parseInt(editCameraPort.getText().toString()); //获取SharedPreferences对象
Context ctx = Video.this;
SharedPreferences sp = ctx.getSharedPreferences("VIDEO", MODE_PRIVATE);
//存入数据
Editor editor = sp.edit();
if (serverAddr != "")
editor.putString("ServerAddr", serverAddr);
if (cameraName != "")
editor.putString("CameraName", cameraName); editor.putInt("CameraID", cameraID); editor.putInt("CameraPort", cameraPort); editor.commit(); } @Override
public boolean onCreateOptionsMenu(Menu menu)
{
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.activity_video, menu);
return true;
} private OnClickListener clickListener = new OnClickListener()
{ @Override
public void onClick(View v)
{
int ret = 0;
switch(v.getId())
{
case R.id.buttonSave:
saveSetting ();
break; } }
}; }

该文件关键的代码是OnCreate函数中的以下4行代码 :

		intent.setAction("VIDEO_TIMER");
PendingIntent sender = PendingIntent.getBroadcast(this, 0, intent, 0);
AlarmManager am = (AlarmManager)getSystemService(ALARM_SERVICE);
am.setRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP, SystemClock.elapsedRealtime(), 10*1000, sender);

intent.setAction("VIDEO_TIMER") 设置系统向应用程序发送的消息类型为 VIDEO_TIMER。

am.setRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP, SystemClock.elapsedRealtime(), 10*1000, sender);  该行代码让系统多久发送一次消息,我这里设置的是10秒发送一次VIDEO_TIMER消息。

发送出去的消息, 会在 class AutoReceiver extends BroadcastReceiver 类里进行接收,并处理。

以下是 TimerService.java 文件 :

package com.routing.videocamera;

public class TimerService extends Service
{
public void onCreate() { super.onCreate();
Log.v("TimerService", "OnCreate");
} private static String mUrl;
public void onStart(Intent intent, int startId) { Log.v("TimerService", "onStart"); Context ctx = TimerService.this;
//Context ctx = context;
SharedPreferences sp = ctx.getSharedPreferences("VIDEO", MODE_PRIVATE);
//存入数据 String serverAddr = sp.getString("ServerAddr", "");
String cameraName = sp.getString("CameraName", "");
int cameraID = sp.getInt("CameraID", 0);
int cameraPort = sp.getInt("CameraPort", 0); if (serverAddr == "" || cameraName == "" || cameraID == 0 || cameraPort == 0 || serverAddr == "NULL" || cameraName == "NULL")
return; String Url = serverAddr + "/cameramgr.php? " + "CameraName=" + cameraName + "&CameraID=" + cameraID + "&CameraPort=" + cameraPort;
Log.v("FFMPEG URL=", Url); mUrl = Url; new Thread()
{
@Override
public void run()
{
HttpParams httpParams;
// 创建 HttpParams 以用来设置 HTTP 參数(这一部分没必要的)
httpParams = new BasicHttpParams();
// 设置连接超时和 Socket 超时,以及 Socket 缓存大小
HttpConnectionParams.setConnectionTimeout(httpParams, 20 * 1000);
HttpConnectionParams.setSoTimeout(httpParams, 20 * 1000);
HttpConnectionParams.setSocketBufferSize(httpParams, 8192);
// 设置重定向,缺省为 true
HttpClientParams.setRedirecting(httpParams, true);
// 设置 user agent
String userAgent = "Mozilla/5.0 (Windows; U; Windows NT 5.1; zh-CN; rv:1.9.2) Gecko/20100115 Firefox/3.6";
HttpProtocolParams.setUserAgent(httpParams, userAgent);
// 创建一个 HttpClient 实例
// 注意 HttpClient httpClient = new HttpClient(); 是Commons HttpClient
// 中的使用方法,在 Android 1.5 中我们须要使用 Apache 的缺省实现 DefaultHttpClient
HttpClient httpClient = new DefaultHttpClient(httpParams); //String s = this.getName();
HttpGet httpRequest = new HttpGet(mUrl);
String strResult = "doGetError";
try { HttpResponse httpResponse = httpClient.execute(httpRequest); if (httpResponse.getStatusLine().getStatusCode() == 200)
{ strResult = EntityUtils.toString(httpResponse.getEntity());
}
else
{
strResult = "Error Response: " + httpResponse.getStatusLine().toString();
}
}
catch (ClientProtocolException e)
{
strResult = e.getMessage().toString();
e.printStackTrace();
}
catch (IOException e)
{
strResult = e.getMessage().toString();
e.printStackTrace();
}
catch (Exception e)
{
strResult = e.getMessage().toString();
e.printStackTrace();
}
Log.v("strResult", strResult);
}
}.start (); } public String doPost(String url, List<NameValuePair> params)
{
/* 建立HTTPPost对象 */
HttpPost httpRequest = new HttpPost(url);
String strResult = "doPostError";
try {
/* 加入请求參数到请求对象 */
httpRequest.setEntity(new UrlEncodedFormEntity(params, HTTP.UTF_8));
/* 发送请求并等待响应 */ HttpParams httpParams;
// 创建 HttpParams 以用来设置 HTTP 參数(这一部分没必要的)
httpParams = new BasicHttpParams();
// 设置连接超时和 Socket 超时。以及 Socket 缓存大小
HttpConnectionParams.setConnectionTimeout(httpParams, 20 * 1000);
HttpConnectionParams.setSoTimeout(httpParams, 20 * 1000);
HttpConnectionParams.setSocketBufferSize(httpParams, 8192);
// 设置重定向,缺省为 true
HttpClientParams.setRedirecting(httpParams, true);
// 设置 user agent
String userAgent = "Mozilla/5.0 (Windows; U; Windows NT 5.1; zh-CN; rv:1.9.2) Gecko/20100115 Firefox/3.6";
HttpProtocolParams.setUserAgent(httpParams, userAgent); HttpClient httpClient = new DefaultHttpClient(httpParams);
HttpResponse httpResponse = httpClient.execute(httpRequest);
/* 若状态码为200 ok */
if (httpResponse.getStatusLine().getStatusCode() == 200)
{
/* 读返回数据 */
strResult = EntityUtils.toString(httpResponse.getEntity());
}
else
{
strResult = "Error Response: " + httpResponse.getStatusLine().toString();
}
}
catch (ClientProtocolException e)
{
strResult = e.getMessage().toString();
e.printStackTrace();
}
catch (IOException e)
{
strResult = e.getMessage().toString();
e.printStackTrace();
}
catch (Exception e)
{
strResult = e.getMessage().toString();
e.printStackTrace();
}
Log.v("strResult", strResult);
return strResult;
} public void onDestroy() { super.onDestroy();
Log.v("TimerService", "onDestroy");
} public IBinder onBind(Intent intent) {
return null;
} public static String inStream2String(InputStream inputStream)
{
InputStreamReader inputStreamReader = null;
try
{
inputStreamReader = new InputStreamReader(inputStream, "utf8");
}
catch (UnsupportedEncodingException e1)
{
e1.printStackTrace();
}
BufferedReader reader = new BufferedReader(inputStreamReader);
StringBuffer sb = new StringBuffer("");
String line;
try
{
while ((line = reader.readLine()) != null)
{
sb.append(line);
sb.append("\n");
}
}
catch (IOException e)
{
e.printStackTrace();
}
return sb.toString();
} }

该文件是定时器运行的代码,没什么特别的。

以下是 AutoReceiver.java 文件 :

package com.routing.videocamera;

public class AutoReceiver extends BroadcastReceiver
{
/*要接收的intent源*/
//static final String ACTION = "android.intent.action.BOOT_COMPLETED";
private static final int MODE_PRIVATE = 0;
@Override
public void onReceive(Context context, Intent intent)
{ if (intent.getAction().equals("VIDEO_TIMER"))
{
Intent Intentservice = new Intent(context, TimerService.class); // 要启动的Activity
Intentservice.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startService(Intentservice); }
} public static String inStream2String(InputStream inputStream)
{
InputStreamReader inputStreamReader = null;
try
{
inputStreamReader = new InputStreamReader(inputStream, "utf8");
}
catch (UnsupportedEncodingException e1)
{
e1.printStackTrace();
}
BufferedReader reader = new BufferedReader(inputStreamReader);
StringBuffer sb = new StringBuffer("");
String line;
try
{
while ((line = reader.readLine()) != null)
{
sb.append(line);
sb.append("\n");
}
}
catch (IOException e)
{
e.printStackTrace();
}
return sb.toString();
} public void KeepAlive()
{
new Thread()
{
public void run()
{
HttpClient client = new DefaultHttpClient(); HttpGet get = new HttpGet("http://www.baidu.com");
HttpResponse response; try {
response = client.execute(get);
if (response.getStatusLine().getStatusCode() == HttpStatus.SC_OK)
{
InputStream is = response.getEntity().getContent();
String result = inStream2String(is);
//Assert.assertEquals(result, "GET_SUCCESS");
//Toast.makeText(Video.this, "Http Get Success:", Toast.LENGTH_LONG).show();
Log.v("FFMPEG-----= ", result);
}
else
{
//Toast.makeText(Video.this, "Http Get Fail", Toast.LENGTH_LONG).show(); }
}
catch (ClientProtocolException e)
{
// TODO Auto-generated catch block
e.printStackTrace();
}
catch (IOException e)
{
// TODO Auto-generated catch block
e.printStackTrace();
}
//super.run();
}
}.start();
} }

该文件的关键函数是onReceive函数,在这里接收到系统发送出来的 VIDEO_TIMER 消息,然后我们启动一个新任务来运行 TimerService 的代码 :

    public void onReceive(Context context, Intent intent)
{ if (intent.getAction().equals("VIDEO_TIMER"))
{
Intent Intentservice = new Intent(context, TimerService.class); // 要启动的Activity
Intentservice.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startService(Intentservice); }
}

完整源码下载地址 :http://download.csdn.net/download/langeldep/7845831

Android后台执行的定时器实现的更多相关文章

  1. android: 后台执行的定时任务

    Android 中的定时任务一般有两种实现方式,一种是使用 Java API 里提供的 Timer 类, 一种是使用 Android 的 Alarm 机制.这两种方式在多数情况下都能实现类似的效果,但 ...

  2. Android应用开发按下返回键退向后台执行

    转载请注明来源:http://blog.csdn.net/kjunchen/article/details/50429694 Android应用开发按下返回键退向后台执行 我们日常使用的非常多Andr ...

  3. php中怎么实现后台执行?

    http://www.cnblogs.com/zdz8207/p/3765567.html php中实现后台执行的方法: ignore_user_abort(true); // 后台运行set_tim ...

  4. 在Android开发中,定时器一般有以下3种实现方法

    在Android开发中,定时器一般有以下3种实现方法: 原文地址http://www.360doc.com/content/12/0619/13/87000_219180978.shtml 一.采用H ...

  5. android 后台服务定时通知

    最近有个项目的要求是在程序退出之后,任然可以每天定时发通知,我们可以想下,其实就是后台开一个服务,然后时间到了就发下通知. 1.首先我们需要用到Service类. 先上代码在慢慢解释 package ...

  6. Android初级教程启动定时器详解

    本案例知识是:后台执行定时任务. Alarm机制: 一.创建LongRunningService类 package com.example.servicebestpractice; import ja ...

  7. 全面盘点当前Android后台保活方案的真实运行效果(截止2019年前)

    本文原作者“minminaya”,作者网站:minminaya.cn,为了提升文章品质,即时通讯网对内容作了幅修订和改动,感谢原作者. 1.引言 对于IM应用和消息推送服务的开发者来说,在Androi ...

  8. Android后台处理最佳实践(Best Practices for Background Jobs)

    本课将告诉你如何通过后台加载来加速应用启动和降低应用耗电. 后台跑服务 除非你做了特殊指定,否则在应用中的大部分前台操作都是在一个特殊的UI线程里面进行的.这有可能会导致一些问题,因为长时间运行的操作 ...

  9. iOS和Android后台机制对比

    转自:http://blog.csdn.net/zsch591488385/article/details/27232881 一.iOS的“伪后台”程序 首先,先了解一下ios 中所谓的「后台进程」到 ...

随机推荐

  1. JAVA 后台SSM框架接收安卓端的json数据

    最近项目上与安卓端做JSON数据交互,使用的SSM框架,刚开始的时候感觉很简单,想着不就是把安卓端的JSON数据封装为Bean类对象吗? 于是就这样写了 可是这样一直报400,百度原因是因为请求url ...

  2. C#递归查询

    一.sql --构造测试数据: 只作演示用 CREATE TABLE [dbo].[Tim_LinqTable]( [Id] int PRIMARY KEY IDENTITY(1,1) NOT NUL ...

  3. Python把给定的列表转化成二叉树

    在LeetCode上做题时,有很多二叉树相关题目的测试数据是用列表给出的,提交的时候有时会出现一些数据通不过,这就需要在本地调试,因此需要使用列表来构建二叉树,方便自己调试.LeetCode上二叉树结 ...

  4. Python 解LeetCode:680. Valid Palindrome II

    题目:给定一个字符串,在最多删除一个字符的情况下,判断这个字符串是不是回文字符串. 思路:回文字符串,第一想到的就是使用两个指针,前后各一个,当遇到前后字符不一致的时候,有两种情况,删除前面字符或者删 ...

  5. C++雾中风景2:struct还是class?

    之前因为都在忙着毕业的开题答辩与投稿论文的事宜,一直没有时间更新这个系列的文章.师弟看了上一篇雾中风景的文章,希望我继续把这个系列的文章写下去.坦白说,C++的特性很多,这也不是教学指南的文章,我会选 ...

  6. Lucene全文检索引擎

    <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/20 ...

  7. 二、VueJs 填坑日记之基础项目构建

    在上一篇文章中,大致介绍了一下本系列博文以及学习vuejs我们需要了解的一些概念,希望大家认真阅读,所谓知己知彼,百战百胜,学习也一样,工欲善其事,必先利其器,要想学好vuejs,那前提的概念一定要熟 ...

  8. 极简版ASP.NET Core学习路径及教程

    绝承认这是一个七天速成教程,即使有这个效果,我也不愿意接受这个名字.嗯. 这个路径分为两块: 实践入门 理论延伸 有了ASP.NET以及C#的知识以及项目经验,我们几乎可以不再需要了解任何新的知识就开 ...

  9. 自己动手修改Robotium代码(下)

    public void takeScreenshot(){   View decorView = viewFetcher.getRecentDecorView(viewFetcher.getWindo ...

  10. HCTF

    题目:魂斗罗 介绍:这个是HCTF里面的杂项,很好玩的, 1,这个看链接可以下载(http://139.224.54.27/gogogo/hundouluo.nes),然后在网上下载一个虚拟器(htt ...