- 浏览: 284031 次
- 性别:
- 来自: 深圳
文章分类
最新评论
-
u011563440:
u011563440 写道用反射A.class.toGener ...
Java接口中的内部接口和接口中的内部类 -
u011563440:
用反射A.class.toGenericString()--- ...
Java接口中的内部接口和接口中的内部类 -
narutolby:
因为接口本身不能实例化,所以在new 一个 接口的内部类时默认 ...
Java接口中的内部接口和接口中的内部类 -
hekuilove:
请问楼主,第四部如何做?在eclipse哪个位置?
eclipse中使用maven插件的问题:Updating index central|http://repo1.maven.org/maven2 -
小小生:
请问下,如果修改了数据库里面的订单任务的时间,那么不是要重启? ...
利用Spring动态对Quartz定时任务管理
在android中,我们使用startService启动服务。在Service的派生类里,会调用onStartCommand(Intent,int,int)来启动服务。 onStartCommand有三种返回值: START_STICKY:sticky的意思是“粘性的”。使用这个返回值时,我们启动的服务跟应用程序"粘"在一起,如果在执行完onStartCommand后,服务被异常kill掉,系统会自动重启该服务。当再次启动服务时,传入的第一个参数将为null; START_NOT_STICKY:“非粘性的”。使用这个返回值时,如果在执行完onStartCommand后,服务被异常kill掉,系统不会自动重启该服务。 START_REDELIVER_INTENT:重传Intent。使用这个返回值时,如果在执行完onStartCommand后,服务被异常kill掉,系统会自动重启该服务,并将Intent的值传入。 转:http://www.cnblogs.com/not-code/archive/2011/06/24/2088653.html 学习android有一段时间了,在没有好的师傅带领下,入门时总是碰到这样那样的新概念、新知识,而这些知识虽说能看得明白,却没有深得它的精髓。就好比有一双好看的鞋子,我只知道它很好看,穿起来挺舒服的,但是却不了解为什么穿起来舒服,它是由什么材料组合成的,怎么穿才能更舒服有效。不过开始学习android虽然开头困难重重,但是我坚信大家只要有耐心,有决心不间断地学习下去,总能有大丰收的时候。打个比方,你每天都对着一匹马观察它,时间久了让你画出来,即使没有到庖丁解牛的境界,但至少你能将这匹马的整个外形都牢记在心中,自己慢慢地描绘,到后面有人指导一下或者自己突然的醒悟,很快可以突破这一层表面的隔膜,后面再去深入了解也自然水到渠成。 如今再去看看android 文档,发现以前很多小的知识点都没注意到,但文档上却有写着。想来想去,归根到底这就是看二手鞋(因为是中文所以学习起来很快,我就是贪那个快^-^)的悲哀,二手鞋很多都是抛去小树叶,只留一条光溜溜的枝条,告诉大家有这样的概念怎么用,但是如果想突破自身的瓶颈,那就要再去看一手鞋了(都是英文的,虽然会耗大家很多时间,但是却收获颇丰)。 这段时间总结了一下Service的概念,下面开始是小细节的深入(不过这也是二手鞋^-^,如果可以的话大家可以去看看英文文档,在Reference——android.app——service)。因为全文好像都是翻译过来的,再加上一点点自己的了解。所以难免有理解错误的时候,所以大家如果发现错误了,请猛写评论吧再猛击提交吧,尽情地向我扔砖头吧,我爱砖头^-^。 Service 作为android组件之一,但在界面上却很难看到它的身影,它负责着后台一些繁重的数据处理,比如音乐播放,单词的查询。当然也有跟activity交互的功能,比如我想跳过这首播放下一首的动作、我想查的这个单词。 什么是Service? 解惑: 1、 Service不是分离开的进程,除非其他特殊情况,它不会运行在自己的进程,而是作为启动运行它的进程的一部分。 2、 Service不是线程,这意味着它将在主线程里劳作。 启动service有两种方法: 1、 Context.startService() 调用者与服务之间没有关联,即使调用者退出,服务仍可运行 2、 Context.bindService() 调用者与服务绑定在一起,调用者一旦退出,服务也就终止 Service的生命周期 如果使用startService()启动service,系统将通过传入的Intent在底层搜索相关符合Intent里面信息的service。如果服务没有启动则先运行onCreate,然后运行onStartCommand (可在里面处理启动时传过来的Intent和其他参数),直到明显调用stopService或者stopSelf才将停止Service。无论运行startService多少次,只要调用一次stopService或者stopSelf,Service都会停止。使用stopSelf(int)方法可以保证在处理好intent后再停止。 控制service运行的主要方式有两种,主要是根据onStartCommand方法返回的数值。方法: 2、 这里主要解释这三个变量的意义: 1、 START_STICKY 在运行onStartCommand后service进程被kill后,那将保留在开始状态,但是不保留那些传入的intent。不久后service就会再次尝试重新创建,因为保留在开始状态,在创建 service后将保证调用onstartCommand。如果没有传递任何开始命令给service,那将获取到null的intent 2、 START_NOT_STICKY 在运行onStartCommand后service进程被kill后,并且没有新的intent传递给它。Service将移出开始状态,并且直到新的明显的方法(startService)调用才重新创建。因为如果没有传递任何未决定的intent那么service是不会启动,也就是期间onstartCommand不会接收到任何null的intent。 3、 START_REDELIVER_INTENT 在运行onStartCommand后service进程被kill后,系统将会再次启动service,并传入最后一个intent给onstartCommand。直到调用stopSelf(int)才停止传递intent。如果在被kill后还有未处理好的intent,那被kill后服务还是会自动启动。因此onstartCommand不会接收到任何null的intent。 客户端也可以使用bindService来保持跟service持久关联。谨记:如果使用这种方法,那么将不会调用onstartCommand(跟startService不一样,下面例子注释也有解析,大家可试试)。客户端将会在onBind回调中接收到IBinder接口返回的对象。通常IBinder作为一个复杂的接口通常是返回aidl数据。 Service也可以混合start和bind一起使用。 权限 要运行service,首先必须在 Service能够保护个人的IPC调用,所以在执行实现该调用时前先使用 进程生命周期 当service运行在低内存的环境时,将会kill掉一下存在的进程。因此进程的优先级将会很重要: 1、 如果service当前正在执行onCreate、onStartCommand、onDestroy方法,主进程将会成为前台进程来保证代码可以执行完成避免被kill 2、 如果service已经启动了,那么主进程将会比其他可见的进程的重要性低,但比其他看不见的进程高。因为只有少部分进程始终是用户可见的,因此除非在极度低内存的时候,不然 service是不会被kill的。 3、 如果有客户端关联到service,那么service永远比客户端重要。也就是说客户端可见,那么service也可见(我理解这里的可见并不是可以看到,而是重要性,因为可见往往就表示重要性高)。 4、 Service可以使用startForeground API将service放到前台状态。这样在低内存时被kill的几率更低,但是文档后面又写了,如果在极度极度低内存的压力下,该service理论上还是会被kill掉。但这个情况基本不用考虑。 当然如果service怎么保持还是被kill了,那你可以通过重写onStartCommand返回变量来设置它的启动方式。比如:START_STICKY、START_REDELIVER_INTENT等等,前面已经讨论了它们的作用,这里就不再累赘了 另外: service 的onCreate和onStartCommand 是运行在主线程的,所以如果里面有处理耗时间的任务。两种处理: 1、 请将它们都挪到新的线程里。 2、 用系统提供的IntentService,它继承了Service,它处理数据是用自身新开的线程。 好了说了这么多下面就是例子的时刻了。总共有两个例子,第一个是本地调用,第二个是远程调用. 口水快没了,所以下面就直接进入代码环节吧。代码里面已经有详细的注解了,如果真的真的还是不明白,那就是我这篇二手鞋的失败了。:( 大家可以试试startService 和bindService这两种区别。 轻松一下,Toast.makeText(this, "温馨提示:\n代码已经写好了,如果想测试一下可以去掉注释的喔", 2000).show(); ^-^ 下面是远程调用的例子,主要是用系统提供的Messenger,省去自己去写复杂的aidl文件 如果加了android:process=":remote" ,那在调试时在service断点是不会触发的。 测试远程调用,我弄多一份项目来测试,主要是查看是否连接成功和有多少个客户端连接上. 效果图:START_NOT_STICKY
or START_REDELIVER_INTENT
AndroidManifest.xml
里申明<service>标签。checkCallingPermission(String)
方法
检查是否有这个权限。
private NotificationManager mNM;
// 通知唯一标示,在通知开始和结束使用
private int NOTIFICATION = R.string.local_service_started;
// 与界面交互的类,由于service跟界面总是运行在同一程序里,所以不用处理IPC
public class LocalBinder extends Binder {
LocalService getService() {
return LocalService.this;
}
}
@Override
public void onCreate() {
mNM = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
// 在service开始时,将icon图标放到通知任务栏
showNotification();
}
//
private void showNotification() {
CharSequence text = getText(R.string.local_service_started);
Notification notification = new Notification(R.drawable.icon, text,
System.currentTimeMillis());
// 当点击通知时,启动该contentIntent关联的activity
PendingIntent contentIntent = PendingIntent.getActivity(this, 0,
new Intent(this, showActivity.class), 0);
// 在通知栏上显示标题和内容
notification.setLatestEventInfo(this,
getText(R.string.local_service_label), text, contentIntent);
mNM.notify(NOTIFICATION, notification);
}
// 兼容2.0以前版本
@Override
public void onStart(Intent intent, int startId) {
}
// 在2.0以后的版本如果重写了onStartCommand,那onStart将不会被调用,注:在2.0以前是没有onStartCommand方法
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Log.i("Service", "Received start id " + startId + ": " + intent);
// 如果服务进程在它启动后(从onStartCommand()返回后)被kill掉, 那么让他呆在启动状态但不取传给它的intent.
// 随后系统会重写创建service,因为在启动时,会在创建新的service时保证运行onStartCommand
// 如果没有任何开始指令发送给service,那将得到null的intent,因此必须检查它.
// 该方式可用在开始和在运行中任意时刻停止的情况,例如一个service执行音乐后台的重放
return START_STICKY;
}
@Override
public void onDestroy() {
mNM.cancel(NOTIFICATION);
Toast
.makeText(this, R.string.local_service_stopped,
Toast.LENGTH_SHORT).show();
}
@Override
public IBinder onBind(Intent intent) {
return mBinder;
}
private final IBinder mBinder = new LocalBinder();
}
/** Called when the activity is first created. */
private LocalService mBoundService;
private boolean mIsBound;
private ServiceConnection mConnection = new ServiceConnection() {
@Override
public void onServiceDisconnected(ComponentName name) {
// 当进程崩溃时将被调用,因为运行在同一程序,如果是崩溃将所以永远不会发生
// 当解除绑定时也被调用
mBoundService = null;
Toast.makeText(LocalActivity.this,
R.string.local_service_disconnected, Toast.LENGTH_SHORT)
.show();
}
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
// service连接建立时将调用该方法
mBoundService = ((LocalService.LocalBinder) service).getService();
Toast.makeText(LocalActivity.this,
R.string.local_service_connected, Toast.LENGTH_SHORT)
.show();
}
};
void doBindService() {
// 建立service连接。因为我们知道程序会运行在本地里,因此使用显示的类名来实现service
// (但是不支持跟其他程序交互)
// 两种传递,一种是在manifest里写好intent-filter的action,一种是显示传递
// bindService(new Intent("com.LocalService.LocalService"), mConnection,
// Context.BIND_AUTO_CREATE);
// bindService(new Intent(LocalActivity.this, LocalService.class),
// mConnection, Context.BIND_AUTO_CREATE);
//如果用这种方法将会调用onStartCommand方法
startService(new Intent(LocalActivity.this, LocalService.class));
mIsBound = true;
}
void doUnbindService() {
if (mIsBound) {
// Detach our existing connection.
stopService(new Intent(LocalActivity.this, LocalService.class));
// unbindService(mConnection);
mIsBound = false;
}
}
@Override
protected void onDestroy() {
doUnbindService();
super.onDestroy();
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
doBindService();
}
}
<intent-filter>
<action android:name="com.LocalService.MessengerService" />
</intent-filter>
</service>
private NotificationManager mNM;
// 保存所有跟服务连接的客户端
ArrayList<Messenger> mClients = new ArrayList<Messenger>();
// 保存最后一次跟服务连接的客户端的标志
int mValue = 0;
// 注册指令,Message's replyTo 字段值必须是client 的Messenger
static final int MSG_REGISTER_CLIENT = 1;
// 取消指令,Message's replyTo 字段值必须是先前給MSG_REGISTER_CLIENT的Messenger
static final int MSG_UNREGISTER_CLIENT = 2;
// 服务发送指令,可以在客户端和服务直接交流
static final int MSG_SET_VALUE = 3;
// 处理客户端传送过来的消息
class IncomingHandler extends Handler {
@Override
public void handleMessage(Message msg) {
switch (msg.what) {
case MSG_REGISTER_CLIENT:
// Optional Messenger where replies to this message can be sent.
// The semantics of exactly how this is used are up to the
// sender and receiver.
mClients.add(msg.replyTo);
break;
case MSG_UNREGISTER_CLIENT:
mClients.remove(msg.replyTo);
break;
case MSG_SET_VALUE:
mValue = msg.arg1;
for (int i = mClients.size() - 1; i >= 0; i--) {
try {
mClients.get(i).send(
Message.obtain(null, MSG_SET_VALUE, mValue, 0));
} catch (RemoteException e) {
// 远程客户端出错,从list中移除
// 遍历列表以保证内部循环安全运行
mClients.remove(i);
}
}
break;
default:
super.handleMessage(msg);
}
Log.i("Service", "有" + mClients.size() + "客户端");
}
}
// 创建一个新的Messenger跟已存在的Handler关联
// 如果有任何消息发送到Messenger,将交给Handler处理
final Messenger mMessenger = new Messenger(new IncomingHandler());
@Override
public void onCreate() {
mNM = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
// 在service开始时,将icon图标放到通知任务栏
showNotification();
}
//
private void showNotification() {
CharSequence text = getText(R.string.local_service_started);
Notification notification = new Notification(R.drawable.icon, text,
System.currentTimeMillis());
// 当点击通知时,启动该contentIntent关联的activity
PendingIntent contentIntent = PendingIntent.getActivity(this, 0,
new Intent(this, showActivity.class), 0);
// 在通知栏上显示标题和内容
notification.setLatestEventInfo(this,
getText(R.string.remote_service_label), text, contentIntent);
mNM.notify(R.string.remote_service_started, notification);
}
//
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Log.i("Service", "Received start id " + startId + ": " + intent);
return START_STICKY;
}
@Override
public void onDestroy() {
mNM.cancel(R.string.remote_service_started);
Toast.makeText(this, R.string.remote_service_stopped,
Toast.LENGTH_SHORT).show();
}
@Override
public IBinder onBind(Intent intent) {
return mMessenger.getBinder();
}
}
/** Called when the activity is first created. */
private Messenger mService = null;
private boolean mIsBound;
private TextView mCallbackText;
class IncomingHandler extends Handler {
@Override
public void handleMessage(Message msg) {
switch (msg.what) {
case MessengerService.MSG_SET_VALUE:
mCallbackText.setText("Received from service: " + msg.arg1);
break;
default:
super.handleMessage(msg);
}
}
}
final Messenger mMessenger = new Messenger(new IncomingHandler());
private ServiceConnection mConnection = new ServiceConnection() {
@Override
public void onServiceDisconnected(ComponentName name) {
// 当进程崩溃时将被调用,因为运行在同一程序,如果是崩溃将所以永远不会发生
// 当解除绑定时也被调用
mService = null;
mCallbackText.setText("Disconnected.");
Toast.makeText(MessengerActivity.this,
R.string.remote_service_disconnected, Toast.LENGTH_SHORT)
.show();
}
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
// service连接建立时将调用该方法
// 返回IBinder接口以便我们可以跟service关联。
// 我们可通过IDL接口来交流
mService = new Messenger(service);
mCallbackText.setText("Attached.");
// 只有我们连接着都监听着服务
try {
// 注册
Message msg = Message.obtain(null,
MessengerService.MSG_REGISTER_CLIENT);
msg.replyTo = mMessenger;
mService.send(msg);
// 例子
msg = Message.obtain(null, MessengerService.MSG_SET_VALUE,
11111111, 0);
mService.send(msg);
} catch (RemoteException e) {
// In this case the service has crashed before we could even
// do anything with it; we can count on soon being
// disconnected (and then reconnected if it can be restarted)
// so there is no need to do anything here.
}
Toast.makeText(MessengerActivity.this,
R.string.remote_service_connected, Toast.LENGTH_SHORT)
.show();
}
};
void doBindService() {
bindService(new Intent(MessengerActivity.this, MessengerService.class),
mConnection, Context.BIND_AUTO_CREATE);
mIsBound = true;
mCallbackText.setText("Binding.");
}
void doUnbindService() {
if (mIsBound) {
if (mService != null) {
try {
// 取消注册
Message msg = Message.obtain(null,
MessengerService.MSG_UNREGISTER_CLIENT);
msg.replyTo = mMessenger;
mService.send(msg);
} catch (RemoteException e) {
// There is nothing special we need to do if the service
// has crashed.
}
}
// Detach our existing connection.
unbindService(mConnection);
mIsBound = false;
mCallbackText.setText("Unbinding.");
}
}
@Override
protected void onDestroy() {
doUnbindService();
super.onDestroy();
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.messenger);
mCallbackText = (TextView) findViewById(R.id.text);
doBindService();
}
}
发表评论
-
android 捕获系统异常并上传日志具体实现
2014-08-22 10:45 1670在做项目时,经常会把错误利用异常抛出去,这样在开发时就可以通 ... -
Android开发之InstanceState详解
2012-05-29 15:07 2288加上转载地址:http://www.cnblogs.com ... -
Android应用开发中的风格和主题(style,themes)
2012-05-28 23:58 1486越来越多互联网企业都在Android平台上部署其客户端,为了 ... -
android 权限总结
2012-01-15 12:13 1366Android权限大全代码 android.permis ... -
转:Android 软件开发之如何使用Eclipse Debug调试程序详解
2012-01-13 16:38 1598Android 开发之使用Eclipse ... -
成为Android高手必须掌握的28大项内容和10个建议
2012-01-13 16:06 2103(一)成为Android高手必须掌握的8项基本要求 ... -
轻量级Android HTTP框架
2012-01-04 17:29 1277转自: http://www.pin5i.com/showt ... -
Java代码查询网站
2012-01-03 23:39 1252这些是经常使用的一些Java代码查询网站. 1. java ... -
收集的android开源项目,android学习必备
2011-12-11 16:16 20951、Android团队提供的示例项目 如果不是从 ... -
Android深入浅出视频中的总结
2011-10-11 23:41 964Android中5个进进程等级 下面列出进程等级,由高到低 ... -
Android权限列表permission说明
2011-05-18 16:04 2181Android权限分的很细,但命名比较人性化,Android ... -
Android手机操作系统中的常用术语
2011-02-18 10:54 1276Android手机操作系统中有 ... -
android模拟器sd卡的配置
2011-01-21 16:26 2840使用android手机模拟器进行开发时,自己定制的模拟器如果 ... -
Android开发 FAQ
2010-12-23 15:12 2259Android新手入门 FAQ 1、Q:什么是Andr ... -
Android URL
2010-12-17 17:11 1325Android API http://develope ... -
详解Android Intent
2010-09-16 11:23 7954一、 Intent 作用 ...
相关推荐
这篇博客"Android Service深入解析Demo"通过实例深入讲解了Service的相关知识,下面将详细阐述Service的核心概念、生命周期、启动与绑定方式以及如何在实践中应用。 1. **Service核心概念** Service是Android系统...
本篇文章将深入解析"android service 简单实例源代码",帮助你理解如何在Android应用中创建和使用Service。 首先,我们来了解一下Service的基本概念。Service主要分为两种类型:标准Service和IntentService。标准...
本篇文章将全面解析Android Service的相关知识点,包括Service的生命周期、启动与绑定方式、交互机制以及注意事项。 首先,Service的生命周期主要由四个方法构成:onCreate(), onStartCommand(), onBind(), 和...
本篇将详细讲解如何在Android应用中解析JSON数据,主要涉及以下几个方面: 1. JSON基本结构 JSON由两种基本结构构成:对象(Object)和数组(Array)。对象是以花括号 `{}` 包围的键值对集合,键用双引号括起,如`...
《Android Framework精编内核解析》是一本深入探讨Android系统核心框架层的专著,它主要涵盖了Android系统架构的关键组成部分,旨在帮助读者理解和掌握Android应用开发背后的机制。在这个压缩包中,包含了一份名为...
本篇文章将深入探讨“Android Service”的使用,以实例解析其功能和重要性。 首先,Service并不创建一个新的UI(用户界面),而是为其他组件(如Activity)提供服务,通常用于执行长时间运行的操作,如播放音乐、...
《字节面试Android11.0最新Framework解析》是一份专为程序员面试准备的资料,主要聚焦于Android系统的深层机制,特别是针对Android 11.0版本的Framework层进行了深入剖析。Framework是Android系统的核心部分,它定义...
Android XML 多种解析生成 Android 平台中,XML 文件解析和生成是非常重要的一部分。XML 文件可以存储和交换结构化数据,广泛应用于 Android 应用程序中。本文将介绍 Android 中使用多种解析和生成 XML 文件的方法...
在这个场景中,我们将探讨如何在Android Studio中解析JSON数组,并将获取的数据展示在ListView上。这个过程对于任何初学者来说都是一个重要的学习点,因为它涉及到网络数据获取、JSON解析以及UI展示。 首先,我们...
### Android Service 中 onStartCommand 返回值解析 在 Android 开发中,`Service` 是一个非常重要的组件,它可以在后台运行长时间的操作而不会为用户提供任何界面。`Service` 的生命周期方法之一是 `...
在"Android中Service完全解析(上)"这个主题中,我们将重点讨论Service的基础知识。 Service的生命周期主要包括onCreate()、onStartCommand()、onBind()、onUnbind()、onDestroy()等方法。当Service创建时,会调用...
3. **Service的生命周期源码解析** Android系统的Service生命周期管理主要由ActivityManagerService(AMS)负责。当你调用startService()或bindService()时,AMS会通过Binder机制将请求传递到目标进程的...
我们在使用某些Android应用的时候,可能会发现安装了某应用以后,会有一些服务也会随之运行。而且,这些服务每次都会随着手机开机而启动。有的服务做的更绝,当用户在运行的服务中手动停止该服务以后,过了一段时间...
### Android学习之Service开机启动详解 #### 一、引言 在Android开发中,有时我们需要让应用程序中的某个服务(Service)在设备启动时自动运行,例如进行后台数据同步、监控某些变化等。为了实现这一需求,可以...
在Android开发中,JSON(JavaScript Object Notation)是一种轻量级的数据交换格式,广泛用于客户端与服务器之间的数据传输,因为它易于人阅读和编写,同时也易于机器解析和生成。本篇文章将深入探讨Android平台上的...
总的来说,实现不需root的Android截图并将其解析为BMP格式是一项涉及Android系统接口、JNI编程以及图像处理技术的任务。通过理解Android系统的工作原理,熟练掌握C++和Java的混合编程,我们可以创建出自己的截图工具...
### Android各组件详解——Service深度解析 在深入探讨Service组件之前,我们首先简要回顾一下Android框架中的其他关键组件,尤其是ContentProvider。ContentProvider作为数据访问层,它提供了跨应用共享数据的能力...
Android Service完全解析,关于服务你所需知道的一切(上).html
这个资料包包含了一个关于Android解析JSON数据的PPT和源码,可以帮助开发者深入理解和实践Android中如何处理JSON数据。 首先,我们要理解JSON的基本结构。JSON由键值对组成,键用引号括起来,后跟冒号,值可以是...