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
サービスが起動しないだけです。それが問題です。
Gady、

サービスが開始されていないことをどのようにして知りましたか、ログなどを印刷しましたか?
Lalit Poptani、2011年

1
実行されていないことを確認するためにログは必要ありません。Android OSは実行中のサービスを公開します。ただし、ログを使用してエラーが発生しているかどうかを確認することをお勧めします。エラーが発生している場合は、context.startService()の前に発生すると思います。
Tony

1
Log.v("BatteryLogger", "Got to onReceive, about to start service");onReceiveハンドラーに追加しましたが、ログに表示されません。したがって、リスナーは失敗しています(?)
Gady

回答:


302

これがAutoStartアプリケーションの完全な例です

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>

autostart.java

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-これは、アプリケーションを1回実行した後でデバイスを起動するたびにポップアップします。

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

6
これは最終的にそれを解決しました。問題は、サービスがプレーンな古いサービスではなくIntentServiceであることと、サービスコードのエラーの組み合わせでした。この回答の完全性とAndroidへのログインが問題の解決に役立ちました。
Gady、

3
詳細な例については+1。これは活動なしでも機能しますか?
バラス2013年

1
onStart()コールバックは非推奨です。代わりにonStartCommand()を使用してください。
mmBs 2016年

1
@mmBsこの答えはほぼ5年前のものなので、明らかです
Lalit Poptani、2016年

3

起動後にデバイスがスリープ状態になるため、サービスが完了する前にシャットダウンされる場合があります。最初にwake lockを取得する必要があります。さいわい、サポートライブラリは、これを行うクラスを提供します。

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 2016

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に到達したことが示されていません。他のアイデアはありますか?
Gady、2011年

1
あなたのコードはエミュレータで動作していますか?エミュレータで意図を受け取ることができますか?
Vivek、2011年

エミュレータでテストしたところ、DDMSでエラーが発生しましたが、少なくともサービスを開始しようとしているようですが、Log()ステートメントが1つもなく、エミュレータデバイスで、サービスがOS設定で実行されていると表示されません。DDMSのエラーは次のとおりですSystem.err - at com.phx.batterylogger$1.onReceive(BatteryLogger.java:43)。それは、問題がBatteryLoggerサービスの43行目にあることを意味しますか?
Gady、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ハンドラーに追加しましたが、ログに表示されません。したがって、リスナーは失敗しています(?)
Gady

電話が起動したときに放送受信機が起動するのはなぜですか?ありがとう!
Ruchir Baronia 2016

男の子。私は数年でアンドロイドに触れていません。申し訳ありませんが、@ Ruchir
Phix

0

コメントで述べたように、検索を簡単にするために、3.1以降はこれは不可能です https://stackoverflow.com/a/19856367/6505257


Think Twice Code Onceで述べられているように、それは真実ではありません同じ答えのコメントセクションでが。
Ricardo A.

私の主張を裏付けるためのより詳細な説明:stackoverflow.com/questions/20441308/…
リカルドA.

私はアンドロイドで何もしていないので本当に長い時間ですが、あなたが正しい、それは可能です。他の引数は必要ありません。(私は答えを編集します)
MrKew
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.