Android-启动时启动服务


107

从我在Stack Exchange和其他地方看到的所有内容中,我已经正确设置了所有内容,可以在启动Android OS时启动IntentService。不幸的是,它没有在启动时启动,并且我没有收到任何错误。也许专家可以帮忙...

表现:

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

<uses-sdk android:minSdkVersion="8" />
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.BATTERY_STATS" />

<application android:icon="@drawable/icon" android:label="@string/app_name">
    <service android:name=".BatteryLogger"/>
    <receiver android:name=".StartupIntentReceiver">  
        <intent-filter>  
            <action android:name="android.intent.action.BOOT_COMPLETED" />  
        </intent-filter>  
    </receiver>
</application>

</manifest>

用于启动的BroadcastReceiver:

package com.phx.batterylogger;

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

public class StartupIntentReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Intent serviceIntent = new Intent(context, BatteryLogger.class);
        context.startService(serviceIntent);
    }
}

更新:我尝试了下面的几乎所有建议,并Log.v("BatteryLogger", "Got to onReceive, about to start service");在StartupIntentReceiver的onReceive处理程序中添加了日志记录,但从未记录过任何内容。因此,它甚至还没有到达BroadcastReceiver。

我认为我正在部署APK并正确测试,只是在Eclipse中运行Debug,控制台表示它已成功将其安装到我的Xoom平板电脑上,位于\ BatteryLogger \ bin \ BatteryLogger.apk。然后进行测试,我重新启动了平板电脑,然后查看DDMS中的日志并在OS设置中检查“正在运行的服务”。这听起来是否正确,还是我缺少了什么?再次感谢您的帮助。


1
您遇到什么问题,没有得到任何UI ..?
Lalit Poptani 2011年

1
该服务永远不会启动,这就是问题所在。
盖迪2011年

您如何知道您的服务还没有开始,是否打印了日志或类似内容。
Lalit Poptani 2011年

1
您不需要日志即可查看其未运行。Android OS公开了正在运行的服务。但是,使用日志记录来查看是否发生错误是明智的。我敢冒险猜测,如果发生错误,它会在context.startService()之前发生。
托尼

1
我添加Log.v("BatteryLogger", "Got to onReceive, about to start service");了onReceive处理程序,它从不显示在日志中。所以听众失败了(?)
盖迪

Answers:


302

好吧,这是一个自动启动应用程序的完整示例

AndroidManifest文件

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

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

    <application android:icon="@drawable/icon" android:label="@string/app_name">

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

        <activity android:name=".hello"></activity>
        <service android:enabled="true" android:name=".service" />
    </application>
</manifest>

自动启动

public class autostart extends BroadcastReceiver 
{
    public void onReceive(Context context, Intent arg1) 
    {
        Intent intent = new Intent(context,service.class);
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            context.startForegroundService(intent);
        } else {
            context.startService(intent);
        }
        Log.i("Autostart", "started");
    }
}

service.java

public class service extends Service
{
    private static final String TAG = "MyService";
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }
    public void onDestroy() {
        Toast.makeText(this, "My Service Stopped", Toast.LENGTH_LONG).show();
        Log.d(TAG, "onDestroy");
    }

    @Override
    public void onStart(Intent intent, int startid)
    {
        Intent intents = new Intent(getBaseContext(),hello.class);
        intents.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        startActivity(intents);
        Toast.makeText(this, "My Service Started", Toast.LENGTH_LONG).show();
        Log.d(TAG, "onStart");
    }
}

hello.java-执行一次Applicaton后,每次启动设备时都会弹出该窗口。

public class hello extends Activity 
{   
    public void onCreate(Bundle savedInstanceState) 
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        Toast.makeText(getBaseContext(), "Hello........", Toast.LENGTH_LONG).show();
    }
}

10
+1为完整示例。也许我应该将带有onHandleIntent的IntentService更改为简单的旧服务
Gady 2011年

6
这最终解决了它。问题是我的服务是一个IntentService而不是普通的旧Service和服务代码中的错误的结合。答案的彻底性和在Android中的登录帮助我解决了问题。
加迪2011年

3
+1为详细示例。如果没有任何活动,这也行得通吗?
balas

1
不建议使用onStart()回调。您应该改用onStartCommand()。
毫米波

1
@mmBs这个答案将近5年了,所以它很明显
Lalit Poptani

3

由于设备启动后会进入睡眠状态,因此服务可能在完成之前就已关闭。您需要先获取唤醒锁。幸运的是,支持库为我们提供了执行此操作的类

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权限:

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

1
我有同样的问题。你介意帮我吗?谢谢!stackoverflow.com/questions/35373525/starting-my-service
Ruchir Baronia

2

以下应该工作。我已经核实了。可能是您的问题在其他地方。

public class MyReceiver extends BroadcastReceiver{

    @Override
    public void onReceive(Context context, Intent intent) {
        Log.d("TAG", "MyReceiver");
        Intent serviceIntent = new Intent(context, Test1Service.class);
        context.startService(serviceIntent);
    }
}




public class Test1Service extends Service {
    /** Called when the activity is first created. */
    @Override
    public void onCreate() {
        super.onCreate();
        Log.d("TAG", "Service created.");
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        Log.d("TAG", "Service started.");
        return super.onStartCommand(intent, flags, startId);
    }

    @Override
    public void onStart(Intent intent, int startId) {
        super.onStart(intent, startId);
        Log.d("TAG", "Service started.");
    }
    @Override
    public IBinder onBind(Intent arg0) {
        return null;
    }
}




<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
      package="com.test"
      android:versionCode="1"
      android:versionName="1.0"
      android:installLocation="internalOnly">
    <uses-sdk android:minSdkVersion="8" />

    <application android:icon="@drawable/icon" android:label="@string/app_name">

    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.BATTERY_STATS" 
    />
<!--        <activity android:name=".MyActivity">
            <intent-filter>  
                <action android:name="android.intent.action.MAIN" /> 
                <category android:name="android.intent.category.LAUNCHER"></category> 
            </intent-filter>
       </activity> -->
        <service android:name=".Test1Service" 
                  android:label="@string/app_name"
                  >
        </service>
        <receiver android:name=".MyReceiver">  
            <intent-filter>  
                <action android:name="android.intent.action.BOOT_COMPLETED" /> 
            </intent-filter>  
        </receiver> 
    </application>
</manifest>

谢谢你,但是还是没用。日志甚至没有显示它进入了StartupIntentReceiver。还有其他想法吗?
盖迪2011年

1
您的代码在模拟器中工作吗?您可以在模拟器中接收意图吗?
Vivek

我在模拟器中进行了测试,但在DDMS中收到错误消息,但看起来至少该服务正在尝试启动,尽管我的所有Log()语句都没有,并且模拟器设备未将我的服务显示为在OS设置中运行。这是DDMS中的错误:这System.err - at com.phx.batterylogger$1.onReceive(BatteryLogger.java:43)是否意味着问题出在我的BatteryLogger服务的第43行?
盖迪2011年

2

我找到了一种方法,可以在设备重新启动时使您的应用程序正常运行,请按照以下步骤操作,以确保成功。

AndroidManifest文件

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="pack.saltriver" android:versionCode="1" android:versionName="1.0">
<uses-permission 
android:name="android.permission.REQUEST_IGNORE_BATTERY_OPTIMIZATIONS"/>
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />

<application android:icon="@drawable/icon" android:label="@string/app_name">

<activity android:name=".MainActivity">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
    <receiver android:name=".UIBootReceiver" android:enabled="true" 
    android:exported="true">
        <intent-filter>
            <action android:name="android.intent.action.BOOT_COMPLETED" />
            <category android:name="android.intent.category.DEFAULT" />
        </intent-filter>
    </receiver>
     <service android:name=".class_Service" />
</application>

UIBootReceiver

public class UIBootReceiver extends BroadcastReceiver {

private static final String TAG = "UIBootReceiver";
@Override

    public void onReceive(Context context, Intent arg1)
    {
        Toast.makeText(context, "started", Toast.LENGTH_SHORT).show();
        Intent intent = new Intent(context,class_Service.class);
        context.startService(intent);
    }
  }

这是请求权限,无需管理此应用程序的电池节省,因此您可以稳定地在后台运行。

在MainActivity类的onCreate()中声明以下代码:

   Intent myIntent = new Intent();
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
        myIntent.setAction(Settings.ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS);
        myIntent.setData(Uri.parse("package:" + 
   DeviceMovingSpeed.this.getPackageName()));
    }
    startActivity(myIntent);

1

看起来很像我的,但我用的是全包名的接收器:

<receiver android:name=".StartupIntentReceiver">

我有:

<receiver android:name="com.your.package.AutoStart"> 

是的,虽然有点可以解决您的问题,所以更好地发布一个hello world示例。
Lalit Poptani 2011年

1

没有完整包装,我就取得了成功,您知道呼叫链在哪里中断吗?如果您使用进行调试Log(),则在什么时候不再起作用?

我认为可能在您的IntentService中,一切看起来都很好。


我添加Log.v("BatteryLogger", "Got to onReceive, about to start service");了onReceive处理程序,它从不显示在日志中。所以听众失败了(?)
盖迪

引导电话时,广播接收机如何启动?谢谢!
Ruchir Baronia'2

男孩。几年来我没有碰过android。抱歉@Ruchir
Phix 2016年

0

如注释中所述,只是为了使搜索更容易,所以自3.1 https://stackoverflow.com/a/19856367/6505257 起这是不可能的


正如完全相同的答案的“注释两次”中的“ 思考两次”所述,那不是真的。
里卡多(Ricardo A.)

更详细的解释来支持我的观点:stackoverflow.com/questions/20441308/...
里卡多A.

自从我用android做任何事情以来已经很长时间了,但是你是对的,这是可能的。无需其他参数。(我将编辑答案)
MrKew
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.