Android服务之Service及demo(本地service)

android中服务是运行在后台的东西,级别与activity差不多。既然说service是运行在后台的服务,那么它就是不可见的,没有界面的东西。你可以启动一个服务Service来播放音乐,或者记录你地理信息位置的改变,或者启动一个服务来运行并一直监听某种动作。

Service和其他组件一样,都是运行在主线程中,因此不能用它来做耗时的请求或者动作。你可以在服务中开一一个线程,在线程中做耗时动作。

那么究竟Service怎么使用呢?

老规矩,先来点基础知识。

一.基础知识

服务一般分为两种:

1:本地服务 Local Service 用于应用程序内部。在Service可以调用Context.startService()启动,调用Context.stopService()结束。在内部可以调用Service.stopSelf() 或 Service.stopSelfResult()来自己停止。无论调用了多少次startService(),都只需调用一次stopService()来停止。

2:远程服务, Remote Service 用于android系统内部的应用程序之间。可以定义接口并把接口暴露出来,以便其他应用进行操作。客户端建立到服务对象的连接,并通过那个连接来调用服务。调用Context.bindService()方法建立连接,并启动,以调用 Context.unbindService()关闭连接。多个客户端可以绑定至同一个服务。如果服务此时还没有加载,bindService()会先加载它。
提供给可被其他应用复用,比如定义一个天气预报服务,提供与其他应用调用即可。

那么先来看Service的生命周期吧:如图:


context.startService() ->onCreate()- >onStart()->Service running--调用context.stopService() ->onDestroy()

context.bindService()->onCreate()->onBind()->Service running--调用>onUnbind() -> onDestroy() 从上诉可以知道分别对应本地的,,以及远程的,也对应不同的方式启动这个服务。

二.实战

我们可以定义一个本地服务继承Service,然后在这个服务里播放媒体播放器或者记录地理位置变化。通常有时候我们的Service要与Activity交互,那么可以可以定义一个内部类,返回这个Service,当然我们要考虑到如果是以绑定方式启动服务,那么内部类可以定义为继承Binder,然后返回本地服务,具体代码如下。

复制代码
package com.dongzi;

import android.app.Service;
import android.content.Intent;
import android.media.MediaPlayer;
import android.os.Binder;
import android.os.IBinder;
import android.util.Log;

public class LocalService extends Service {

private static final String TAG = "LocalService";
private IBinder binder=new LocalService.LocalBinder();

@Override
public IBinder onBind(Intent intent) {

return binder;
}
MediaPlayer mediaPlayer=null;
@Override
public void onCreate() {
Log.i(TAG, "onCreate");
//这里可以启动媒体播放器
// if(mediaPlayer==null)
// mediaPlayer=MediaPlayer.create(this, uri);
super.onCreate();
}

@Override
public void onStart(Intent intent, int startId) {
Log.i(TAG, "onStart");
super.onStart(intent, startId);
}

@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Log.i(TAG, "onStartCommand");
return START_STICKY;
}



@Override
public void onDestroy() {
Log.i(TAG, "onDestroy");
super.onDestroy();
}


//定义内容类继承Binder
public class LocalBinder extends Binder{
//返回本地服务
LocalService getService(){
return LocalService.this;
}
}


}
复制代码

我们可以从上面知道

 //定义内容类继承Binder
    public class LocalBinder extends Binder{
        //返回本地服务
        LocalService getService(){
            return LocalService.this;
        }
    }

可以返回这个服务,然后activity可以通过服务调用服务的方法了。

那么如何启动服务呢?从上面基础知识中,我们知道有2种方法,如下:

  //启动服务
private void startCustomService(){
Intent intent=new Intent(this,LocalService.class);
startService(intent);
}

第2种绑定方式:

复制代码
LocalService localService=null;
//用bindService方法启动服务
private void BinderService(){
Intent intent=new Intent(this,LocalService.class);
bindService(intent, new ServiceConnection(){
@Override
public void onServiceConnected(ComponentName componentName, IBinder binder) {
//调用bindService方法启动服务时候,如果服务需要与activity交互,
//则通过onBind方法返回IBinder并返回当前本地服务
localService=((LocalService.LocalBinder)binder).getService();
//这里可以提示用户,或者调用服务的某些方法
}

@Override
public void onServiceDisconnected(ComponentName componentName) {
localService=null;
//这里可以提示用户
}
}, Context.BIND_AUTO_CREATE);
}
复制代码

在绑定服务的时候,需要一个服务连接对象,ServiceConnection,服务一旦连接,就会调用onServiceConnected方法,我们可以在这个方法里面返回我们的本地服务对象,具体看代码;而在服务断开时候会调用onServiceDisconnected方法,我们可以清理一些服务资源。


接下来,我们会讲解一些AIDL的一些知识。这些与服务息息相关。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
钉钉的CalDAV服务器是一个标准的CalDAV服务器,因此你可以使用任何支持CalDAV协议的第三方库来访问它。下面是一个使用Android系统自带的SyncAdapter框架实现的CalDAV同步Demo。 1. 添加依赖 在build.gradle文件中添加以下依赖: ``` dependencies { implementation "com.github.aflx:sardine-android:5.7.0" } ``` 这里使用了Sardine-Android库,它是一个支持WebDAV和CalDAV协议的Android库,可以方便地与钉钉的CalDAV服务器进行交互。 2. 创建SyncAdapter 创建一个继承自AbstractThreadedSyncAdapter的SyncAdapter类,并实现其中的onPerformSync()方法,用于执行CalDAV同步任务。 ```java public class CalDAVSyncAdapter extends AbstractThreadedSyncAdapter { private static final String TAG = "CalDAVSyncAdapter"; private final Sardine mSardine; public CalDAVSyncAdapter(Context context, boolean autoInitialize) { super(context, autoInitialize); mSardine = new OkHttpSardine(); } @Override public void onPerformSync(Account account, Bundle extras, String authority, ContentProviderClient provider, SyncResult syncResult) { try { // TODO: 执行CalDAV同步任务 } catch (Exception e) { Log.e(TAG, "CalDAV sync failed", e); syncResult.stats.numIoExceptions++; } } } ``` 3. 注册SyncAdapter 在AndroidManifest.xml文件中注册SyncAdapter,并指定对应的账户类型和CalDAV服务器地址。 ```xml <application> <provider android:name="android.content.ContentProvider" android:authorities="com.android.calendar" android:exported="false" android:syncable="true" /> <service android:name=".CalDAVSyncAdapterService" android:exported="true"> <intent-filter> <action android:name="android.content.SyncAdapter" /> </intent-filter> <meta-data android:name="android.content.SyncAdapter" android:resource="@xml/caldav_sync_adapter" /> <meta-data android:name="android.provider.CONTACTS_STRUCTURE" android:resource="@xml/contacts" /> </service> </application> <uses-permission android:name="android.permission.READ_CALENDAR" /> <uses-permission android:name="android.permission.WRITE_CALENDAR" /> <uses-permission android:name="android.permission.AUTHENTICATE_ACCOUNTS" /> <account-authenticator xmlns:android="http://schemas.android.com/apk/res/android" android:accountType="com.android.exchange" android:icon="@drawable/icon_exchange" android:smallIcon="@drawable/icon_exchange" android:label="@string/app_name" /> <sync-adapter xmlns:android="http://schemas.android.com/apk/res/android" android:contentAuthority="com.android.calendar" android:accountType="com.android.exchange" android:userVisible="false" android:supportsUploading="true" android:allowParallelSyncs="false" android:isAlwaysSyncable="true" /> ``` 在res/xml目录下创建caldav_sync_adapter.xml文件,指定SyncAdapter的参数。 ```xml <sync-adapter xmlns:android="http://schemas.android.com/apk/res/android" android:contentAuthority="com.android.calendar" android:accountType="com.android.exchange" android:userVisible="false" android:supportsUploading="true" android:allowParallelSyncs="false" android:isAlwaysSyncable="true" /> ``` 4. 执行CalDAV同步任务 在SyncAdapter的onPerformSync()方法中,使用Sardine库实现CalDAV同步任务。以下是一个简单的例子,可以获取钉钉CalDAV服务器上的所有日历事件。 ```java @Override public void onPerformSync(Account account, Bundle extras, String authority, ContentProviderClient provider, SyncResult syncResult) { try { // 创建Sardine对象 mSardine.setCredentials("username", "password"); // 获取所有日历事件 List<DavResource> resources = mSardine.getResources("https://calendar.dingtalk.com/caldav/username/events/"); // 解析日历事件 for (DavResource resource : resources) { CalendarBuilder builder = new CalendarBuilder(); Calendar calendar = builder.build(resource.getInputStream()); Log.d(TAG, "Event: " + calendar.toString()); } } catch (Exception e) { Log.e(TAG, "CalDAV sync failed", e); syncResult.stats.numIoExceptions++; } } ``` 注意:在使用Sardine库访问CalDAV服务器时,需要使用完整的CalDAV资源地址,例如"https://calendar.dingtalk.com/caldav/username/events/",其中username为钉钉账号的用户名。另外,钉钉的CalDAV服务器使用的是HTTPS协议,需要添加相应的证书验证。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值