Android学习感悟之Service、Broadcast以及ContentProvider的常见用法

本篇包含Service、Broadcast以及ContentProvider的常见用法

简介

四大组件,包含了Activity、Service、Broadcast以及ContentProvider,而Activity使用最多,所以在感悟的第一篇就介绍了;而当我们在项目开发过程中,就难免会用到其他的组件,这篇文章中就会讲道如何使用其他组件,并且有一些个人的理解。

Service

Service生命周期

Service,我们又叫它服务,这个东西在进程间通信的文章中,就帮了我们大忙,通过绑定服务,得到Binder对象,就能够进程间的通信了,而这只是启动服务的一种方式,还有一种就是通过context来startService。

服务和Activity一样,都是有生命周期的,下面就上一张官方Service的生命周期图:

Android学习感悟之Service、Broadcast以及ContentProvider的常见用法_第1张图片
service_lifecycle.png

可以看到Service的生命周期都是以onCreate开始,onDestory结束,但是使用不同的启动方式去启动服务,生命周期的中间是会有不同的地方。

总的来看Service中的回调方法有如下几个:

1、onCreate:只有在第一次创建(不管如何创建)的时候,才会被回调

2、onStartCommand:只有通过startService启动的服务才回回调;

3、onBind:只有第一次通过bindService启动的服务才回回调,之后再绑定也不回调了,因为该方法其实就是去返回一个已经创建好的Binder;

4、onUnbind:只有最后一个unbindService的时候调用

5、onDestory:只有在所有启动都结束后才会回调,即startService多少次就要stopService多少次,才能被结束;或者bindService多少次就要unBindService多少次才能被结束;或者两者都有,即启动和结束要匹配完才能结束。

注:具体的测试demo的地址会在最后放上

前台Service

大家都知道Service是没有界面的,给人的感觉就是后台运行的,正常情况的确如此,但是它却依旧运行在主线程,所以耗时操作都要在子线程中处理,其实Service还有一种叫做前台服务,就好比网易云音乐的播放操作栏。

其必要条件是:必须在状态栏提供通知,除非服务停止或从前台移除,否则不能清除该通知。

而具体的方法就是调用startForeground()方法,其两个参数为唯一标识通知的整型数(不能为0)和状态栏的Notification,例如:

NotificationCompat.Builder builder = new NotificationCompat.Builder(this);
builder.setSmallIcon(R.mipmap.ic_launcher);
builder.setLargeIcon(BitmapFactory.decodeResource(getResources(), R.mipmap.ic_launcher));
builder.setAutoCancel(false);
builder.setOngoing(true);
builder.setShowWhen(true);
builder.setContentTitle("这是一个前台服务");
builder.setContentText("你好啊~");
Notification notification = builder.build();
startForeground(NOTIFICATION_DOWNLOAD_PROGRESS_ID, notification);

而如果该通知要做的更好,就需要用到RemoteView的知识,可能后续的文章中会讲道。

在前台demo中既然能start就肯定会有stopForeground(true),参数表示,是否移除该通知,而该通知的开和关,这里让启动服务的组件去控制,便用到了Messenger,这是一种比较简单的进程间通信的方法,具体的代码就不上了,想了解的就看看demo去吧,代码比较简单。

Broadcast

广播,在目前的开发中用到的不多,用到的地方就如:软件的安装情况、极光推送的回调、短信监听等等;

广播的回调方法只有一个onReceive(Context context, Intent intent),且在该方法中,操作不能超过10秒,否则会ANR。参数的具体含义就不多解释了。

广播有两种注册方式,一种是静态注册,一种是动态注册,下面分辨来看看如何实现的:

1、静态注册:

需要我们在AndroidManifest.xml文件中,注册receiver,所以我们就得先实现一个:

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.widget.Toast;

/**
 * created by arvin on 17/2/24 00:02
 * email:[email protected]
 */
public class StaticReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Toast.makeText(context, "静态注册的广播", Toast.LENGTH_SHORT).show();
    }
}

然后在AndroidManifest.xml中声明:


    
        
    

这样就注册成功了,这里有个细节,exported没有设置时,如果有intent-filter则表示,该广播是可以被其他进程访问的,反之则不能被其他进程方法;如果有exported则以其值为准;

2、动态注册:

这种方式其实也是一样只是声明方式不同,首先依然时实现一个广播接收者:

private BroadcastReceiver mReceiver = new BroadcastReceiver() {
    @Override
    public void onReceive(Context context, Intent intent) {
        showToast("动态注册的广播");
    }
};

然后通过context的registerReceiver方法注册:

registerReceiver(mReceiver, getFilter());

private IntentFilter getFilter() {
    IntentFilter filter = new IntentFilter();
    filter.addAction(DYNAMICS_ACTION);
    return filter;
}

动态注册的广播,需要在不用的时候调用unregisterReceiver,例如在onDestroy方法中调用,避免内存泄露。

这样两种注册方式简单的介绍了一下。

然后发送广播,其实这个也很简单,发送的广播有有序和无序两种,首先来看无需的。

1、发送无序广播:

只需调用context的sendBroadcast即可,传入的事intent,这里就涉及到intent-filter的匹配规则,匹配到则发送到。具体如何匹配可以查看Android感悟之Intent文章中的讲到的匹配规则。

2、发送有序广播:

只需要调用context的sendOrderedBroadcast即可,而所谓有序广播则是依次调用注册的广播,排序规则就是在广播注册的时候,设置的filter.setPriority(int priority);参数的范围事[-1000,1000]其中值越大,就越先收到广播,然后先收到的还能改变intent的值,或者是拦截掉该广播,例如拦截在黑名单中的电话的短信或来电就是这个原理。拦截的方法也很简单就是在准备拦截的广播中调用abortBroadcast()方法即可。

ContentProvider用法

ContentProvider它是一种系统提供的重要的进程间数据交互的方式,底层的实现也是Binder,具体的实现这里也就不介绍了,先看看怎么用。交互肯定分为至少两方,一方是数据提供者(实现ContentProvider),一方则是数据访问者(ContentResolver),而这两者的交互过程中又有一个最为重要的东西Uri以及UriMatcher,统一资源标识符,下面也会挨着介绍。。

Uri

这是一种用于标识某一互联网资源名称的字符串,它有三部分,在ContentProvider中大概可以这么分:content:///

  • "content://":这是第一部分,是协议;
  • :这是第二部分,是主机IP;
  • :这是第三部分,是路径,方便我们用来判断怎么操作的部分。

数据访问者

系统提供了ContentResolver类方便用户通过Uri访问ContentProvider中的提供的内容,而数据的操作无外乎增删查改,刚好这个类也提供了这几种方法,下面就先简单的介绍一下,具体如何用请参考demo:

1、query:查,其中参数有5个比较重要下面挨着说:

  • Uri uri:这个是访问数据的路径;
  • String[] projection:这个是查询需要返回的对应列的数据,null则表示所有列;
  • selection:这个是表示sql语句where后边的条件;
  • String[] selectionArgs:这个是条件中占位符对应的参数值;
  • sortOrder:这个是sql语句中order by后边的排序方式。

2、insert:增,这个是插入单条数据,参数:

  • Uri uri:这个是访问数据的路径;
  • ContentValues values:以map的方式用来存储插入的数据,key是列名,value是值。

当然也可以插入多条调用bulkInsert,只需要把第二个参数变成数组即可;

3、update:改,修改数据,参数:

  • Uri uri:这个是访问数据的路径;
  • ContentValues values:以map的方式用来存储插入的数据,key是列名,value是值。
  • String where:这个是表示sql语句where后边的条件,用于筛选;
  • String[] selectionArgs:这个是条件中占位符对应的参数值;

4、delete:删,删除数据,参数:

  • Uri uri:这个是访问数据的路径;
  • String where:这个是表示sql语句where后边的条件,用于筛选;
  • String[] selectionArgs:这个是条件中占位符对应的参数值;

到这里增删查改四个方法的参数也介绍的差不多了,具体如何使用请参考demo中的ProviderActivity和AddUserActivity。

ContentProvider

使用ContentProvider,则有两部,第一继承ContentProvider,第二在清单文件中配置provider属性,记得设置authorities属性,第二部没什么好说的。来看第一部:

继承ContentProvider,需要重写6个方法:

  • onCreate:在第一次被调用时创建,一般用于初始化工作
  • getType:这个是用于匹配Intent中的mimeType属性的,当然必须要让ContentProvider在清单文件的中的过滤器中设置该属性才有用;
  • query:查询;
  • insert:添加;
  • update:修改;
  • delete:删除。

而增删查改所操作的数据可以是,可以是数据库、文件系统或网络;而更多使用到的是数据库,所以这篇就以数据库为例;下面就以User表为例,包含三个字断:id,name,age;id自增;创建表,我们使用现在比较流行的Greendao,3.0以后使用起来更加方便,使用注解即可,它和ButterKnife的原理一样,有预编译,也不会影响效率,方法如下:

1、在项目级的build.gradle文件中加入:

dependencies {
    classpath 'org.greenrobot:greendao-gradle-plugin:3.2.1'
}

2、在module级的build.gradle文件中加入:

apply plugin: 'org.greenrobot.greendao'

greendao {//这是表示生成的包的目录以及数据库的版本
    schemaVersion 1
    daoPackage 'net.arvin.androidart.gen'
    targetGenDir 'src/main/java'
}

dependencies {
    //数据库
    compile 'org.greenrobot:greendao:3.2.0'
    compile 'org.greenrobot:greendao-generator:3.2.0'
}

等下载完就配置完成,然后就能直接写实体:

@Entity
public class User implements Parcelable {
    @Id(autoincrement = true)
    private Long id;
    private String name;
    private int age;

    @Generated(hash = 1309193360)
    public User(Long id, String name, int age) {
        this.id = id;
        this.name = name;
        this.age = age;
    }

    @Generated(hash = 586692638)
    public User() {
    }

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    @Override
    public int describeContents() {
        return 0;
    }

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeValue(this.id);
        dest.writeString(this.name);
        dest.writeInt(this.age);
    }

    protected User(Parcel in) {
        this.id = (Long) in.readValue(Long.class.getClassLoader());
        this.name = in.readString();
        this.age = in.readInt();
    }

    public static final Parcelable.Creator CREATOR = new Parcelable.Creator() {
        @Override
        public User createFromParcel(Parcel source) {
            return new User(source);
        }

        @Override
        public User[] newArray(int size) {
            return new User[size];
        }
    };
}

写好实体后,rebuild一下就生成好了,在net.arvin.androidart.gen包下会多出
DaoMaster、DaoSession、UserDao三个类;

这个实体其中构造方法是greendao生成的,然后序列化是因为,在demo中需要传递数据;

他的那些注解这里就不介绍了,有兴趣的可以自行Google。

然后我们来看ContentProvider的实现:

这时候我们需要解决一个问题,数据访问者发来的Uri,正确性我们应该如何判断呢?这里便引入了UriMatcher这个工具类,先来看使用方式:

public static final String MIME_ITEM = "user";

public static final String AUTHORITY = "net.arvin.androidart";
public static final String PATH_SINGLE = MIME_ITEM + "/#";
public static final String PATH_MULTIPLE = MIME_ITEM;

private static final int MULTIPLE_PEOPLE = 1;
private static final int SINGLE_PEOPLE = 2;
private static final UriMatcher uriMatcher;

static {
    uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
    uriMatcher.addURI(AUTHORITY, PATH_MULTIPLE, MULTIPLE_PEOPLE);
    uriMatcher.addURI(AUTHORITY, PATH_SINGLE, SINGLE_PEOPLE);
}

在访问者传入Uri是,即可通过uriMatcher.match(uri),返回code,表示对应的哪一个操作,这里有两种操作操作单条,操作全部;其中uriMatcher.addUri的参数,分别是:

  • String authority:这个是Uri中的协议;
  • String path:这个是Uri中的路径,其中#可表示任意数字,而这里这个数字就是下面会用到的id。
  • int code:这个是Uri中路径对应的code,匹配上就返回这个值。

下面就来看看全部的实现源码:

import android.content.ContentProvider;
import android.content.ContentUris;
import android.content.ContentValues;
import android.content.UriMatcher;
import android.database.Cursor;
import android.database.SQLException;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteQueryBuilder;
import android.net.Uri;
import android.support.annotation.NonNull;

import net.arvin.androidart.gen.DaoMaster;
import net.arvin.androidart.gen.UserDao;

/**
 * created by arvin on 17/2/25 15:10
 * email:[email protected]
 */
public class UserProvider extends ContentProvider {
    //mimeType
    public static final String MIME_DIR_PREFIX = "vnd.android.cursor.dir";
    public static final String MIME_ITEM_PREFIX = "vnd.android.cursor.item";
    public static final String MIME_ITEM = "user";

    public static final String MIME_TYPE_SINGLE = MIME_ITEM_PREFIX + "/" + MIME_ITEM;
    public static final String MIME_TYPE_MULTIPLE = MIME_DIR_PREFIX + "/" + MIME_ITEM;

    //有效Uri
    public static final String AUTHORITY = "net.arvin.androidart";
    public static final String PATH_SINGLE = MIME_ITEM + "/#";
    public static final String PATH_MULTIPLE = MIME_ITEM;

    private static final int MULTIPLE_PEOPLE = 1;
    private static final int SINGLE_PEOPLE = 2;
    private static final UriMatcher uriMatcher;

    static {
        uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
        uriMatcher.addURI(AUTHORITY, PATH_MULTIPLE, MULTIPLE_PEOPLE);
        uriMatcher.addURI(AUTHORITY, PATH_SINGLE, SINGLE_PEOPLE);
    }

    public static final String CONTENT_URI_STRING = "content://" + AUTHORITY + "/" + PATH_MULTIPLE;
    public static final Uri CONTENT_URI = Uri.parse(CONTENT_URI_STRING);

    private SQLiteDatabase db;

    @Override
    public boolean onCreate() {
        DaoMaster.DevOpenHelper mHelper = new DaoMaster.DevOpenHelper(getContext(), "art-db", null);
        db = mHelper.getWritableDatabase();
        return db != null;
    }

    @Override
    public String getType(@NonNull Uri uri) {
        switch (uriMatcher.match(uri)) {
            case MULTIPLE_PEOPLE:
                return MIME_TYPE_MULTIPLE;
            case SINGLE_PEOPLE:
                return MIME_TYPE_SINGLE;
            default:
                throw new IllegalArgumentException("UnKnow uri:" + uri);
        }
    }

    @Override
    public Uri insert(@NonNull Uri uri, ContentValues values) {
        long id = db.insert(UserDao.TABLENAME, null, values);
        if (id > 0) {
            Uri newUri = ContentUris.withAppendedId(CONTENT_URI, id);
            notifyChange(newUri);
            return newUri;
        }
        throw new SQLException("failed to insert row into " + uri);
    }

    @Override
    public Cursor query(@NonNull Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {
        SQLiteQueryBuilder qb = new SQLiteQueryBuilder();

        qb.setTables(UserDao.TABLENAME);
        qb.appendWhere(getWhere(uri, selection));

        Cursor cursor = qb.query(db, projection, selection, selectionArgs, null, null, sortOrder);

        notifyChange(uri);
        return cursor;
    }

    @Override
    public int delete(@NonNull Uri uri, String selection, String[] selectionArgs) {
        int count = db.delete(UserDao.TABLENAME, getWhere(uri, selection), selectionArgs);

        notifyChange(uri);
        return count;
    }

    @Override
    public int update(@NonNull Uri uri, ContentValues values, String selection, String[] selectionArgs) {
        int count = db.update(UserDao.TABLENAME, values, getWhere(uri, selection), selectionArgs);
        notifyChange(uri);
        return count;
    }

    private String getWhere(@NonNull Uri uri, String selection) {
        String where;
        switch (uriMatcher.match(uri)) {
            case SINGLE_PEOPLE:
                where = UserDao.Properties.Id.columnName + "=" + uri.getPathSegments().get(1);
                break;
            case MULTIPLE_PEOPLE:
                where = selection;
                break;
            default:
                throw new IllegalArgumentException("UnKnow URI: " + uri);
        }
        return where;
    }

    private void notifyChange(@NonNull Uri uri) {
        if (getContext() != null) {
            getContext().getContentResolver().notifyChange(uri, null);
        }
    }
}

这里区分了操作一条还是多条数据,不同的Uri其实就是判断条件不一样。下面依次描述如何操作数据库:

  • UriMatcher初始化:静态初始化,把能匹配的Uri都加入到改类中,方便判断;
  • onCreate: 通过Grrendao拿到SQLiteDatabase,由于这里边更多接近原生代码,所以就使用SQLiteDatabase来操作,就不用Greendao的方法来操作了。
  • getType:根据Uri来判断是哪种类型;
  • 增删改查:完全是调用db的原生方法,简单的操作表方法,这里也不再介绍了。
  • notifyChange:这个的目的是在于回调告诉那些监听了这个ContentProvider的组件,哪个Uri发生了改变;这里也不再介绍。

到这里自定义ContentProvider就已经实现了,而数据访问者如何使用请查看demo。

Demo源码

Service Demo

Broadcast Demo

ContentProvider Demo

你可能感兴趣的:(Android学习感悟之Service、Broadcast以及ContentProvider的常见用法)