如何从 WakefulBroadcastReceiver 启动 IntentService

Posted

技术标签:

【中文标题】如何从 WakefulBroadcastReceiver 启动 IntentService【英文标题】:How to start an IntentService from a WakefulBroadcastReceiver 【发布时间】:2014-10-25 10:54:30 【问题描述】:

我有一个应用程序,您应该能够使用我将在此问题中发布的代码完全且非常轻松地重新创建它。这是清单文件:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.broadcasttest"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="19"
        android:targetSdkVersion="21" />

    <uses-permission android:name="android.permission.WAKE_LOCK"/>

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >

        <activity
            android:name="com.example.broadcasttest.MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

        <receiver
            android:name="com.example.broadcasttest.TestReceiver"
            android:label="@string/app_name"
            android:enabled="true" >
        </receiver>

        <intentservice 
            android:name="com.example.broadcasttest.MonitorService"
            android:enabled="true" >
            <intent-filter>
                <action android:name="com.example.broadcasttest.MonitorService" />
            </intent-filter>
        </intentservice>
    </application>

</manifest>

如您所见,它包含一个活动、一个(唤醒的)广播接收器和一个意图服务,它们都在同一个包中。活动在启动时开始,代码如下:

package com.example.broadcasttest;

import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;


public class MainActivity extends Activity 

    @Override
    protected void onCreate(Bundle savedInstanceState) 
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        sendBroadcast(new Intent(this, TestReceiver.class));
    


    @Override
    public boolean onCreateOptionsMenu(Menu menu) 
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.main, menu);
        return true;
    

    @Override
    public boolean onOptionsItemSelected(MenuItem item) 
        // Handle action bar item clicks here. The action bar will
        // automatically handle clicks on the Home/Up button, so long
        // as you specify a parent activity in AndroidManifest.xml.
        int id = item.getItemId();
        if (id == R.id.action_settings) 
            return true;
        
        return super.onOptionsItemSelected(item);
    

这样就成功触发了TestReceiveronReceive函数。

package com.example.broadcasttest;

import android.content.Context;
import android.content.Intent;
import android.support.v4.content.WakefulBroadcastReceiver;

public class TestReceiver extends WakefulBroadcastReceiver 

    @Override
    public void onReceive(Context context, Intent intent) 
        //Intent service = new Intent("com.example.broadcasttest.MonitorService");
        Intent service = new Intent(context, MonitorService.class);
        startWakefulService(context, service);
    


这就是问题所在,我在onReceive 函数中放置了一个断点,它肯定会被调用。然而,MonitorService 类永远无法到达。我在onHandleEvent 函数中放置了一个断点,但它似乎永远不会走那么远。这是这个类的代码:

package com.example.broadcasttest;

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

public class MonitorService extends IntentService 

    public MonitorService(String name) 
        super(name);
    

    public MonitorService()
    
        super("MonitorService");
    

    @Override
    protected void onHandleIntent(Intent intent) 
        try 
            Thread.sleep(1000);
         catch (InterruptedException e) 
            e.printStackTrace();
         finally 
            TestReceiver.completeWakefulIntent(intent);
        

    


TestReceiver 类中的注释行可以看出,我尝试使用隐式意图而不是显式意图。我还阅读了this question 并尝试了那里提到的所有内容。我在这里错过了什么吗?我在模拟器(Nexus7 API L)上运行它。

这里有什么我遗漏的吗?

【问题讨论】:

对不起,这些代码的具体用途是什么?它有什么作用? 【参考方案1】:

Application Manifest 中没有 &lt;intentservice&gt; 的标签。 IntentServiceService 的子类,所以需要在 manifest 中声明为服务。


改变

<intentservice 
    android:name="com.example.broadcasttest.MonitorService"
    android:enabled="true" >
        <intent-filter>
            <action android:name="com.example.broadcasttest.MonitorService" />
        </intent-filter>
</intentservice>

<service 
    android:name="com.example.broadcasttest.MonitorService"
    android:enabled="true" >
       <intent-filter>
           <action android:name="com.example.broadcasttest.MonitorService" />
       </intent-filter>
</service>

【讨论】:

谢谢,要是我知道事情就这么简单就好了。我想知道为什么我没有收到任何错误。

以上是关于如何从 WakefulBroadcastReceiver 启动 IntentService的主要内容,如果未能解决你的问题,请参考以下文章

如何将数据从回收器适配器发送到片段 |如何从 recyclerview 适配器调用片段函数

如何从 Firebase 获取所有设备令牌?

如何直接从类调用从接口继承的方法?

如何从服务器获取和设置 android 中的 API(从服务器获取 int 值)?如何绑定和实现这个

如何从Mac从android studio中的fabric注销? [复制]

如何从设备中获取 PDF 文件以便能够从我的应用程序中上传?