Android Service

https://github.com/YuQiao0303/ServiceTest-LearnAndroid

新建服务

com.example.servicetest→New→Service→Service
Exported 属性表示是否允许除了当前程序之外的其他程序访问这个服务
Enabled 属性表示是否启用这个服务。

服务

  • 创建,启动,销毁时分别调用onCreate(), onStartCommant(), on Destroy()
  • onBind()中return一个自定义的binder对象,里面可以写很多方法,供绑定的activity调用
package com.example.servicetest;

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

public class MyService extends Service {
    private static final String TAG = "MyService";

    private DownloadBinder mBinder = new DownloadBinder();
    class DownloadBinder extends Binder {
        public void startDownload() {
            Log.d("MyService", "startDownload executed");
        }
        public int getProgress() {
            Log.d("MyService", "getProgress executed");
            return 0;
        }
    }
    public MyService() {
    }
    @Override
    public IBinder onBind(Intent intent) {
        // TODO: Return the communication channel to the service.
        return mBinder;
    }
    @Override
    public void onCreate() {
        super.onCreate();
        Log.d(TAG, "onCreate: ");
    }
    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        Log.d(TAG, "onStartCommand: ");
        return super.onStartCommand(intent, flags, startId);
    }
    @Override
    public void onDestroy() {
        Log.d(TAG, "onDestroy: ");
        super.onDestroy();
    }
}

启动/停止服务

会执行onCreate(),onStartCommant(),onDestroy() 等方法

Intent startIntent = new Intent(this, MyService.class);
startService(startIntent); // 启动服务
//stopService(stopIntent); // 停止服务

在MyService的任何一个位置调用stopSelf()方法也能让这个服务停止下来。

绑定/解绑服务

可以调用onBind()返回的自定义binder的方法

Intent bindIntent = new Intent(this, MyService.class);
                //BIND_AUTO_CREATE 表示在活动和服务进行绑定后自动创建服务。
                //这会使得MyService中的onCreate() 方法得到执行, 但onStartCommand() 方法不会执行。
                bindService(bindIntent, connection, BIND_AUTO_CREATE); // 绑定服务,执行onBind。如果之前没创建则还要执行onCreate
                //unbindService(connection); // 解绑服务 ,调用onDestroy()

如果既启动又绑定,则要同时调用stopService() 和unbindService() 方法, onDestroy() 方法才会执行。

使用前台服务:在状态栏显示

  • 不会由于系统内存不足的原因导致被回收
  • 一直有一个正在运行的图标在系统的状态栏显示, 下拉状态栏后可以看到更加详细的信息, 非常类似于通知的效果。
    在service中:
@Override
    public void onCreate() {
        super.onCreate();
        Log.d(TAG, "onCreate: ");

        Intent intent = new Intent(this, MainActivity.class);
        PendingIntent pi;
        pi = PendingIntent.getActivity(this, 0, intent, 0);

        /**
         * 8.0及以上:前台服务
         */
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            /*create notification channel*/
            String channelId = "1";
            String channelName = "1";
            int importance = NotificationManager.IMPORTANCE_HIGH;
            createNotificationChannel(channelId, channelName, importance);

            /*show the notification*/
            Notification notification = new NotificationCompat.Builder(this, channelId)
                    .setContentTitle("This is content title")
                    .setContentText("This is content text")
                    .setWhen(System.currentTimeMillis())
                    .setSmallIcon(R.mipmap.ic_launcher)
                    .setLargeIcon(BitmapFactory.decodeResource(getResources(),
                            R.mipmap.ic_launcher))
                    .setContentIntent(pi)
                    .build();
            startForeground(1, notification);//调用startForeground() 方法后就会让MyService变成一个前台服务, 并在系统状态栏显示出来。
        }
        /**
         * 7.0及以下:前台服务
         */
        else{
            Notification notification = new NotificationCompat.Builder(this)
                    .setContentTitle("This is content title")
                    .setContentText("This is content text")
                    .setWhen(System.currentTimeMillis())
                    .setSmallIcon(R.mipmap.ic_launcher)
                    .setLargeIcon(BitmapFactory.decodeResource(getResources(),
                            R.mipmap.ic_launcher))
                    .setContentIntent(pi)
                    .build();
            startForeground(1, notification);
        }
    }
    

版本兼容问题:
郭神自己的文章

服务里干什么?——运行线程!

服务中的代码都是默认运行在主线程当中的,
如果直接在服务里去处理一些耗时的逻辑, 就很容易出现ANR(Application Not Responding) 的情况。
所以这个时候就需要用到Android多线程编程的技术了, 我们应该在服务的每个具体的方法里开启一个子线程, 然后在这里去处理那些耗时的逻辑。
因此, 一个比较标准的服务就可以写成如下形式,即在onStartCommand中运行线程:

public class MyService extends Service {
...
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
    new Thread(new Runnable() {
    @Override
    public void run() {
    // 处理具体的逻辑
    }}).start();
    return super.onStartCommand(intent, flags, startId);
    }
}

如此一来,我们也可以使用

  • AsyncTask(异步消息机制,可以改UI),
  • IntentService :自动停止的服务,其onHandleIntent(Intent intent) 方法在子线程运行

使用IntentService :自动停止的线程

总会有一些程序员忘记开启线程, 或者忘记调用stopSelf() 方法。
Android专门提供了一个IntentService 类

可以简单地创建一个异步的、 会自动停止的服务

activity:

case R.id.start_intent_service:
                // 打印主线程的id
                Log.d("MainActivity", "Thread id is " + Thread.currentThread().
                        getId());
                Intent intentService = new Intent(this, MyIntentService.class);
                startService(intentService);
                break;

MyIntentService:

package com.example.servicetest;

import android.app.IntentService;
import android.content.Intent;
import android.util.Log;

public class MyIntentService extends IntentService {
    /**
     * 首先要提供一个无参的构造函数
     */
    public MyIntentService() {
        super("MyIntentService"); // 调用父类的有参构造函数
    }

    /**
     * 在子线程中运行
     * @param intent
     */
    @Override
    protected void onHandleIntent(Intent intent) {
    // 打印当前线程的id
        Log.d("MyIntentService", "Thread id is " + Thread.currentThread(). getId());
    }
    @Override
    public void onDestroy() {
        super.onDestroy();
        Log.d("MyIntentService", "onDestroy executed");
    }
}

logcat:


logcat
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,293评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,604评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,958评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,729评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,719评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,630评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,000评论 3 397
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,665评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,909评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,646评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,726评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,400评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,986评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,959评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,996评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,481评论 2 342

推荐阅读更多精彩内容