IntentService的使用及原理剖析

IntentService是什么

    IntentService是一个服务类,它是Service的一个子类,同时因为它内部封装了HandlerThread,所以又是一个异步任务类。

IntentService有哪些优点

    1.IntentService是一个服务,优先级比较高,进行后台任务时,没那么容易被杀死。

    2.IntentService内部封装了HandlerThread,不用手动创建进程,使用简单。

    3.IntentService的后台任务执行完后自动销毁,不必手动销毁。

IntentService的使用

   先新建一个类继承IntentService,然后复写onHandlerIntent(Intent intent),然后在onHandlerIntent写后台执行的任务。这个方法的Intent和你启动Intentservice传入的Intent是同一个东西,所以在方法里,可以根据Intent传入的参数进行分类判断要执行的后台任务。这里给出一个例子:

 mTvText.setOnClickListener(v -> {
            Intent intent = new Intent(MultiThreadActivity.this, MyIntentService.class);
            intent.putExtra("task", "task1");
            startService(intent);
            Intent intent1 = new Intent(MultiThreadActivity.this, MyIntentService.class);
            intent.putExtra("task", "task2");
            startService(intent1);
        });
public class MyIntentService extends IntentService {
    private final static String TAG = MyIntentService.class.getSimpleName();
    private int mStartId;
    /**
     * Creates an IntentService.  Invoked by your subclass's constructor.
     */
    public MyIntentService() {
        super("MyIntentService");
    }

    @Override
    public void onCreate() {
        super.onCreate();
        Log.d(TAG, "onCreate: ");
    }

    @Override
    public int onStartCommand(@Nullable Intent intent, int flags, int startId) {
        Log.d(TAG, "onStartCommand: ");
        mStartId = startId;
        return super.onStartCommand(intent, flags, startId);
    }

    @Override
    protected void onHandleIntent(@Nullable Intent intent) {
        // 在子线程中执行(可以做耗时操作)
        String taskTips = intent.getStringExtra("task");
        switch (taskTips) {
            case "task1":
                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                break;
            case "task2":
                try {
                    Thread.sleep(3000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                break;
        }
    }

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

  IntentService原理

在了解IntentService原理之前,我们先了解一下HandlerThread。

    HandlerThread是一个线程类,它继承了Thread,里面又对Handler进行了封装,所以它是Thread和Handler的结合的一个类。HandlerThread的源码很简单,我们着重看它的run方法:

    @Override
    public void run() {
        mTid = Process.myTid();
        Looper.prepare();
        synchronized (this) {
            mLooper = Looper.myLooper();
            notifyAll();
        }
        Process.setThreadPriority(mPriority);
        onLooperPrepared();
        Looper.loop();
        mTid = -1;
    }

    当创建一个HandlerThread线程,并启动线程,线程运行的时候执行它的run方法,run方法中为该线程创建Looper,并轮询消息。

    当我们启动IntentService的时候,会调用IntentService的onCreate -> onStartCommand,当任务没有执行完,又启动Service时,它不会重新创建一个服务,而是调用onStartCommand,当任务都执行完后,销毁服务。我们来看看IntentService内部都做了什么。

 

    onCreate方法中,创建一个HandlerThread实例,并启动线程。然后拿到HandlerThread中的Looper来创建Handler。然后在onStart方法中,创建一个消息,handler发送到子线程的消息队列中,在经过Looper的轮询取出,然后调用Handler的handleMessage方法处理消息。由于这个mServicehandler与子线程中Looper绑定,所以handleMessage()方法中仍然是在子线程中。又因为,多次启动服务时,不会重新创建服务,onCreate方法只会调用一次,所以自始至终只在一个线程中执行任务,并且任务执行的顺序是串行的,因为每一个任务都会放入消息队列中轮询执行。回到handleMessage方法中,先执行我们实现的onHandlerIntent方法去处理我们要处理的后台任务,然后在执行stopSelf方法。当这个startId等于最近IntentService启动次数相同的时候,就会销毁这个服务。

 

    stopSelf()方法中,会调用Service的stopSelf(),最终会调用mActivityManager.stopServiceToken(new ComponentName(this,mClassName),mToken, startId)。底层方法中,会根据starid是否等于最近启动服务的次数来判断是否停止服务。

 

参考《Android开发艺术探索》

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
您好,如果您想在 IntentService使用 MQTT 协议进行消息传输,可以使用 Eclipse Paho Android 客户端库。以下是在 IntentService使用 Paho Android 客户端库实现 MQTT 的基本步骤: 1. 添加 Paho Android 客户端库到您的项目中。 2. 在 IntentService 的 `onCreate()` 方法中创建 MQTT 客户端并连接到 MQTT 代理服务器。您需要指定代理服务器的地址和端口号。可以使用 `MqttConnectOptions` 类设置连接选项,例如设置连接的用户名和密码、清除会话标志等。 3. 在 `onHandleIntent()` 方法中订阅主题、发布消息和处理接收到的消息。您需要实现 `MqttCallback` 接口,并在回调方法中处理接收到的消息。 以下是在 IntentService 中连接到 MQTT 代理服务器的示例代码: ```java public class MyIntentService extends IntentService { private static final String TAG = MyIntentService.class.getSimpleName(); private MqttAndroidClient client; public MyIntentService() { super(TAG); } @Override public void onCreate() { super.onCreate(); String brokerUrl = "tcp://mqtt.eclipse.org:1883"; String clientId = MqttClient.generateClientId(); client = new MqttAndroidClient(this, brokerUrl, clientId); MqttConnectOptions options = new MqttConnectOptions(); options.setUserName("username"); options.setPassword("password".toCharArray()); options.setCleanSession(true); try { IMqttToken token = client.connect(options); token.waitForCompletion(); } catch (MqttException e) { Log.e(TAG, "Failed to connect to MQTT broker", e); } } @Override protected void onHandleIntent(Intent intent) { // 在这里订阅主题、发布消息和处理接收到的消息 // ... // 订阅主题 String topic = "my/topic"; int qos = 1; try { IMqttToken token = client.subscribe(topic, qos); token.waitForCompletion(); } catch (MqttException e) { Log.e(TAG, "Failed to subscribe to topic: " + topic, e); } // 发布消息 String message = "Hello, MQTT!"; try { client.publish(topic, message.getBytes(), qos, false); } catch (MqttException e) { Log.e(TAG, "Failed to publish message: " + message, e); } } @Override public void onDestroy() { super.onDestroy(); try { IMqttToken token = client.disconnect(); token.waitForCompletion(); } catch (MqttException e) { Log.e(TAG, "Failed to disconnect from MQTT broker", e); } } } ``` 您还需要实现 `MqttCallback` 接口并在回调方法中处理接收到的消息。例如: ```java client.setCallback(new MqttCallback() { @Override public void connectionLost(Throwable cause) { // 处理连接丢失事件 } @Override public void messageArrived(String topic, MqttMessage message) throws Exception { // 处理接收到的消息 String payload = new String(message.getPayload()); Log.d(TAG, "Received message: " + payload); } @Override public void deliveryComplete(IMqttDeliveryToken token) { // 处理消息发送完成事件 } }); ``` 在 `onDestroy()` 方法中断开 MQTT 客户端与代理服务器的连接。 希望这些信息能对您有所帮助。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值