尝试在Android上启动服务

时间:2010-05-06 20:56:40

标签: android broadcastreceiver android-service

当设备在Android上启动时,我一直在尝试启动服务,但我无法让它工作。我在网上看了很多链接,但没有一个代码可以工作。我忘记了什么吗?

的AndroidManifest.xml

<receiver
    android:name=".StartServiceAtBootReceiver"
    android:enabled="true"
    android:exported="false"
    android:label="StartServiceAtBootReceiver" >
    <intent-filter>
        <action android:name="android.intent.action._BOOT_COMPLETED" />
    </intent-filter>
</receiver>

<service
    android:name="com.test.RunService"
    android:enabled="true" />

广播接收器

public void onReceive(Context context, Intent intent) {
    if ("android.intent.action.BOOT_COMPLETED".equals(intent.getAction())) {
        Intent serviceLauncher = new Intent(context, RunService.class);
        context.startService(serviceLauncher);
        Log.v("TEST", "Service loaded at start");
    }
}

16 个答案:

答案 0 :(得分:598)

其他答案看起来不错,但我想我会把所有内容都包装成一个完整的答案。

您需要AndroidManifest.xml文件中的以下内容:

  1. <manifest>元素中:

    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
    
  2. <application>元素中(请务必使用BroadcastReceiver的完全限定[或相对]类名称):

    <receiver android:name="com.example.MyBroadcastReceiver">  
        <intent-filter>  
            <action android:name="android.intent.action.BOOT_COMPLETED" />  
        </intent-filter>  
    </receiver>
    

    (您不需要android:enabledexported等属性:Android默认值是正确的)

    MyBroadcastReceiver.java

    package com.example;
    
    public class MyBroadcastReceiver extends BroadcastReceiver {
        @Override
        public void onReceive(Context context, Intent intent) {
            Intent startServiceIntent = new Intent(context, MyService.class);
            context.startService(startServiceIntent);
        }
    }
    
  3. 从最初的问题:

    • 目前尚不清楚<receiver>元素是否在<application>元素
    • 目前尚不清楚是否指定了BroadcastReceiver的正确的完全限定(或相对)类名称
    • <intent-filter>
    • 中有拼写错误

答案 1 :(得分:83)

作为附加信息:在安装外部存储之前,BOOT_COMPLETE将发送到应用程序。因此,如果应用程序安装到外部存储器,它将不会收到BOOT_COMPLETE广播消息。

广播接收器监听“已完成启动”

中的更多详细信息here

答案 2 :(得分:66)

如何在设备启动时启动服务(自动运行应用程序等)

首先:自从Android 3.1+版本开始,如果用户从未启动过您的应用至少一次或者用户强行关闭,那么您将无法收到BOOT_COMPLETE。应用。 这样做是为了防止恶意软件自动注册服务。此安全漏洞已在较新版本的Android中关闭。

解决方案:

使用活动创建应用。当用户运行它时,应用程序可以接收BOOT_COMPLETE广播消息。

第二个:在安装外部存储之前发送BOOT_COMPLETE。如果应用程序安装到外部存储器,它将无法接收BOOT_COMPLETE广播消息。

在这种情况下,有两种解决方案:

  1. 将您的应用安装到内部存储
  2. 在内部存储中安装另一个小应用。此应用程序接收BOOT_COMPLETE并在外部存储上运行第二个应用程序。
  3. 如果您的应用已安装在内部存储中,则下面的代码可以帮助您了解如何在设备启动时启动服务。


    在Manifest.xml中

    许可:

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

    注册您的BOOT_COMPLETED接收器:

    <receiver android:name="org.yourapp.OnBoot">
        <intent-filter>
            <action android:name="android.intent.action.BOOT_COMPLETED"/>
        </intent-filter>
    </receiver>
    

    注册您的服务:

    <service android:name="org.yourapp.YourCoolService" />
    

    在接收器OnBoot.java中:

    public class OnBoot extends BroadcastReceiver
    {
    
        @Override
        public void onReceive(Context context, Intent intent) 
        {
            // Create Intent
            Intent serviceIntent = new Intent(context, YourCoolService.class);
            // Start service
            context.startService(serviceIntent);
    
        }
    
     }
    

    对于HTC,如果设备没有捕获RECEIVE_BOOT_COMPLETED,您可能还需要添加Manifest代码:

    <action android:name="android.intent.action.QUICKBOOT_POWERON" />
    

    接收器现在看起来像这样:

    <receiver android:name="org.yourapp.OnBoot">
        <intent-filter>
            <action android:name="android.intent.action.BOOT_COMPLETED"/>
            <action android:name="android.intent.action.QUICKBOOT_POWERON" />
        </intent-filter>
    </receiver>
    

    如何在没有重启模拟器或真实设备的情况下测试BOOT_COMPLETED? 这很容易。试试这个:

    adb -s device-or-emulator-id shell am broadcast -a android.intent.action.BOOT_COMPLETED
    

    如何获取设备ID?获取ID为

    的已连接设备列表
    adb devices
    
    默认情况下,您可以在ADT中找到

    adb:

    adt-installation-dir/sdk/platform-tools
    

    享受! )

答案 3 :(得分:34)

一起
<action android:name="android.intent.action.BOOT_COMPLETED" />  

也用,

<action android:name="android.intent.action.QUICKBOOT_POWERON" />

HTC设备似乎没有抓住BOOT_COMPLETED

答案 4 :(得分:20)

请注意,在问题的开头,有一个拼写错误:

<action android:name="android.intent.action._BOOT_COMPLETED"/>

而不是:

<action android:name="android.intent.action.BOOT_COMPLETED"/>

一个小“_”并且所有这些麻烦:)

答案 5 :(得分:13)

我刚刚发现可能是因为Fast Boot中的Settings选项&gt; Power

当我关闭此选项时,我的应用程序会收到此广播,但不是。

顺便说一下,Android 2.3.3上有HTC Incredible S

希望它有所帮助。

答案 6 :(得分:12)

我认为您的清单需要添加:

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

答案 7 :(得分:6)

在尝试了所有提到的答案和技巧之后,我终于找到了为什么代码在我的手机中不起作用。一些Android手机,如&#34;华为Honor 3C Android 4.2.2 &#34;在他们的设置中有一个 Statup Manager 菜单,您的应用必须在列表中进行检查。 :)

答案 8 :(得分:5)

我有一个额外的<category> - 标签,不知道这是否有所不同。

<receiver android:name="BootIntentReceiver">  
        <intent-filter>  
            <action android:name="android.intent.action.BOOT_COMPLETED" />  
            <category android:name="android.intent.category.HOME" />  
        </intent-filter>  
</receiver>

你是否尝试过省略if-clause "android.intent.action.BOOT_COMPLETED".equals(intent.getAction(),因为接收者可能只收到那个意图?

答案 9 :(得分:4)

参考此链接http://khurramitdeveloper.blogspot.in/2013/06/start-activity-or-service-on-boot.html 一步一步的程序,以便在服务上使用启动

答案 10 :(得分:3)

在安装外部存储之前,执行BOOT_COMPLETE执行。如果您的应用安装到外部存储,它将不会收到BOOT_COMPLETE广播消息。为防止这种情况,您可以在内部存储中安装应用程序。你可以这样做,只需在menifest.xml中添加这一行

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
android:installLocation="internalOnly"
... >

某些HTC设备可以启用“快速启动”功能,更像是深度休眠,而不是真正的重启,因此不应该提供BOOT_COMPLETE意图。要恢复此功能,您可以在接收器中添加此意图过滤器:

            <intent-filter>
                <action android:name="android.intent.action.BOOT_COMPLETED" />
                <action android:name="android.intent.action.QUICKBOOT_POWERON" />
            </intent-filter>

答案 11 :(得分:2)

如果你正在使用Android Studio而且你非常喜欢自动完成,那么我必须告诉你,我使用的是Android Studio v 1.1.0并且我使用了自动完成功能以下许可

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

Android Studio自动完成RECEIVE_BOOT_COMPLETED全部为小写,如receive_boot_completed,我不停地拔头发,因为我已经勾选了我的检查清单,以便开始服务在开机。我刚刚确认了

  

Android Studio会以小写形式自动完成此权限。

答案 12 :(得分:2)

正如@Damian评论的那样,这个帖子中的所有答案都做错了。像这样手动操作可能会导致服务在设备进入休眠状态时从中间停止。您需要先获得唤醒锁定。幸运的是,Support library gives us a class要做到这一点:

public class SimpleWakefulReceiver extends WakefulBroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        // This is the Intent to deliver to our service.
        Intent service = new Intent(context, SimpleWakefulService.class);

        // Start the service, keeping the device awake while it is launching.
        Log.i("SimpleWakefulReceiver", "Starting service @ " + SystemClock.elapsedRealtime());
        startWakefulService(context, service);
    }
}

然后,在您的服务中,确保释放唤醒锁:

    @Override
    protected void onHandleIntent(Intent intent) {
        // At this point SimpleWakefulReceiver is still holding a wake lock
        // for us.  We can do whatever we need to here and then tell it that
        // it can release the wakelock.

...
        Log.i("SimpleWakefulReceiver", "Completed service @ " + SystemClock.elapsedRealtime());
        SimpleWakefulReceiver.completeWakefulIntent(intent);
    }

不要忘记将WAKE_LOCK permssion添加到你的主要节目中:

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

答案 13 :(得分:2)

这就是我做的事情

<强> 1。我做了Receiver课程

public class BootReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        //whatever you want to do on boot
       Intent serviceIntent = new Intent(context, YourService.class);
       context.startService(serviceIntent);
    }
}

2.在清单中

<manifest...>
    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
    <application...>
        <receiver android:name=".BootReceiver" android:enabled="true" android:exported="false">
            <intent-filter>
                <action android:name="android.intent.action.BOOT_COMPLETED" />
            </intent-filter>
        </receiver>
    ...

3.完成所有你需要&#34;设置&#34; MainActivity中的接收器,它可能位于onCreate

...
 final ComponentName onBootReceiver = new ComponentName(getApplication().getPackageName(), BootReceiver.class.getName());
        if(getPackageManager().getComponentEnabledSetting(onBootReceiver) != PackageManager.COMPONENT_ENABLED_STATE_ENABLED)
        getPackageManager().setComponentEnabledSetting(onBootReceiver,PackageManager.COMPONENT_ENABLED_STATE_ENABLED,PackageManager.DONT_KILL_APP);
...

我从ApiDemos学到的最后一步

答案 14 :(得分:1)

事实上,我不久前遇到了这个问题,并且它真的很容易修复,如果你设置"android.intent.action.BOOT_COMPLETED"权限和意图过滤器,你实际上没有做错任何事。

请注意,如果你在Android 4.X上,你必须在启动服务之前运行广播监听器,这意味着你必须首先添加一个活动,一旦你的广播接收器运行,你的应用程序应该起作用但是,你预计,在Android 4.X上,我没有找到一种在没有任何活动的情况下启动服务的方法,我认为google出于安全原因这样做了。

答案 15 :(得分:0)

如果我将空构造函数留在接收器类中,我就遇到了这个问题。在删除空的contsructor onRreceive方法开始工作正常。