Android - เริ่มบริการเมื่อบูต


108

จากทุกสิ่งที่ฉันเคยเห็นใน Stack Exchange และที่อื่น ๆ ฉันได้ตั้งค่าทุกอย่างอย่างถูกต้องเพื่อเริ่ม IntentService เมื่อระบบปฏิบัติการ Android บูท น่าเสียดายที่มันไม่ได้เริ่มต้นในการบูตและฉันไม่ได้รับข้อผิดพลาดใด ๆ บางทีผู้เชี่ยวชาญสามารถช่วย ...

ประจักษ์:

<?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");ตัวจัดการ onReceive ของ StartupIntentReceiver และไม่มีการบันทึกอะไรเลย ดังนั้นจึงไม่ได้ส่งไปยัง BroadcastReceiver

ฉันคิดว่าฉันกำลังปรับใช้ APK และทดสอบอย่างถูกต้องเพียงแค่เรียกใช้ Debug ใน Eclipse และคอนโซลแจ้งว่าติดตั้งลงในแท็บเล็ต Xoom ของฉันได้สำเร็จที่ \ BatteryLogger \ bin \ BatteryLogger.apk จากนั้นเพื่อทดสอบฉันรีบูตแท็บเล็ตจากนั้นดูบันทึกใน DDMS และตรวจสอบ Running Services ในการตั้งค่าระบบปฏิบัติการ ทั้งหมดนี้ถูกต้องหรือฉันพลาดอะไรไป? อีกครั้งความช่วยเหลือใด ๆ ที่ได้รับการชื่นชมมาก


1
ปัญหาอะไรที่คุณได้รับคุณไม่ได้รับ UI เลย .. ?
Lalit Poptani

1
บริการไม่เคยเริ่มนั่นคือปัญหา
Gady

คุณรู้ได้อย่างไรว่าบริการของคุณไม่ได้เริ่มต้นคุณได้พิมพ์บันทึกหรืออะไรทำนองนั้น .. ?
Lalit Poptani

1
คุณไม่จำเป็นต้องใช้บันทึกเพื่อดูว่ามันไม่ทำงาน ระบบปฏิบัติการ Android เปิดให้บริการที่กำลังทำงานอยู่ อย่างไรก็ตามควรใช้การบันทึกเพื่อดูว่ามีข้อผิดพลาดเกิดขึ้นหรือไม่ ฉันจะเดาว่ามันเกิดขึ้นก่อน 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 - สิ่งนี้จะปรากฏขึ้นทุกครั้งที่คุณเริ่มต้นอุปกรณ์หลังจากดำเนินการ 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 สำหรับตัวอย่างที่สมบูรณ์ บางทีฉันควรเปลี่ยน IntentService ด้วย onHandleIntent เป็นบริการเก่าธรรมดา
Gady

6
สิ่งนี้แก้ไขได้ในที่สุด ปัญหาคือการรวมกันของบริการของฉันเป็น IntentService แทนที่จะเป็นบริการเก่าธรรมดาและข้อผิดพลาดในรหัสบริการ ความละเอียดถี่ถ้วนของคำตอบนี้และการเข้าสู่ระบบ Android ช่วยฉันแก้ปัญหาของฉันได้
Gady

3
+1 สำหรับตัวอย่างโดยละเอียด สิ่งนี้จะใช้งานได้โดยไม่มีกิจกรรมหรือไม่
balas

1
การเรียกกลับ onStart () เลิกใช้งานแล้ว คุณควรใช้ onStartCommand () แทน
mmBs

1
@mmBs คำตอบนี้มีอายุเกือบ 5 ขวบจึงชัดเจน
Lalit Poptani

3

บริการของคุณอาจถูกปิดก่อนที่จะเสร็จสมบูรณ์เนื่องจากอุปกรณ์เข้าสู่โหมดสลีปหลังจากบูตเครื่อง คุณต้องได้รับการล็อกการปลุกก่อน โชคดีที่ไลบรารี Support ให้ชั้นเรียนทำสิ่งนี้:

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 ความคิดอื่น ๆ ?
Gady

1
รหัสของคุณทำงานในโปรแกรมจำลองหรือไม่ คุณสามารถรับเจตนาในโปรแกรมจำลองได้หรือไม่?
วิเวก

ฉันทดสอบในโปรแกรมจำลองและได้รับข้อผิดพลาดใน DDMS แต่ดูเหมือนว่าอย่างน้อยบริการก็พยายามเริ่มต้นแม้ว่าจะไม่มีข้อความใด ๆ ของฉันLog()อยู่ในนั้นและอุปกรณ์จำลองจะไม่แสดงบริการของฉันว่าทำงานในการตั้งค่าระบบปฏิบัติการ นี่คือข้อผิดพลาดใน DDMS: System.err - at com.phx.batterylogger$1.onReceive(BatteryLogger.java:43)หมายความว่าปัญหาอยู่ในบรรทัดที่ 43 ของบริการ BatteryLogger ของฉันหรือไม่
Gady

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);
    }
  }

กำลังขออนุญาตไม่จำเป็นต้องจัดการการประหยัดแบตเตอรี่สำหรับแอพนี้เพื่อให้คุณสามารถทำงานในพื้นหลังได้อย่างเสถียร

ประกาศรหัสนี้ใน onCreate () ของคลาส MainActivity:

   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"> 

ใช่มันเป็นเรื่องเล็กน้อยในการจัดเรียงปัญหาของคุณดังนั้นควรโพสต์ตัวอย่างสวัสดีชาวโลกดีกว่า
Lalit Poptani

1

ฉันประสบความสำเร็จหากไม่มีแพ็คเกจเต็มคุณรู้หรือไม่ว่าสายการโทรถูกขัดจังหวะที่ใด? หากคุณแก้ไขข้อบกพร่องด้วยLog()สิ่งใดที่ใช้ไม่ได้อีกต่อไป

ฉันคิดว่าอาจอยู่ใน IntentService ของคุณทั้งหมดนี้ดูดี


ฉันเพิ่มลงLog.v("BatteryLogger", "Got to onReceive, about to start service");ในตัวจัดการ onReceive แล้ว แต่มันไม่เคยปรากฏในบันทึก ดังนั้นผู้ฟังจึงล้มเหลว (?)
Gady

อะไรทำให้เครื่องรับสัญญาณออกอากาศเริ่มทำงานเมื่อโทรศัพท์ถูกบูท ขอบคุณ!
Ruchir Baronia

บอย. ฉันไม่ได้สัมผัส android มาสองสามปีแล้ว ขออภัย @Ruchir
Phix

0

เพื่อให้ค้นหาได้ง่ายขึ้นตามที่กล่าวไว้ในความคิดเห็นสิ่งนี้ไม่สามารถทำได้เนื่องจาก 3.1 https://stackoverflow.com/a/19856367/6505257


นั่นไม่เป็นความจริงตามที่ระบุไว้ในThink Twice Code Onceในส่วนความคิดเห็นของคำตอบเดียวกัน
Ricardo

คำอธิบายโดยละเอียดเพิ่มเติมเพื่อสนับสนุนข้อโต้แย้งของฉัน: stackoverflow.com/questions/20441308/…
Ricardo A.

เป็นเวลานานมากแล้วที่ฉันทำอะไรกับ Android แต่คุณพูดถูกมันเป็นไปได้ ไม่จำเป็นต้องมีข้อโต้แย้งอื่น ๆ (ฉันจะแก้ไขคำตอบ)
MrKew
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.