3

我有一个实现为WakefulIntentService的服务。每次BroadcastReceiver通过启动它向负责人发出适当的意图时都会启动它。该服务在两种情况下启动:在设备启动时以及当服务的运行实例即将完成时,并通过要求 Android 的传统任务调度程序 ,AlarmManager来安排新的执行,以在未来某个时间发出启动器意图。

问题是,出于安全原因,我被建议不要android:exported="true"在 Manifest 文件中声明的服务中使用。但是,省略它会导致在其中一部测试手机(运行 Android 4.1.2 的三星 S3)中拒绝服务执行:

06-13 11:34:34.181: W/ActivityManager(2270): Permission denied: checkComponentPermission() owningUid=10155
06-13 11:34:34.181: W/ActivityManager(2270): Permission Denial: Accessing service ComponentInfo{com.mypackage.myapp/com.mypackage.myapp.MyService} from pid=10320, uid=2000 that is not exported from uid 10155

添加android:exported="true"解决了这个问题。是否有替代方法可以在不影响应用程序安全性的情况下避免执行拒绝?

清单 xml 文件:

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

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="17" />

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

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

        <receiver android:name=".MyBroadcastReceiver" >
            <intent-filter>
                <action android:name="android.intent.action.BOOT_COMPLETED" />
                <category android:name="android.intent.category.DEFAULT" />
            </intent-filter>
            <intent-filter>
                <action android:name="com.mypackage.myapp" />
            </intent-filter>
        </receiver>

        <service
            android:name="com.mypackage.myapp.MyService"
            android:exported="true">
        </service>

    </application>

</manifest>

BroadcastReceiver: _

package com.mypackage.myapp;

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

import com.commonsware.cwac.wakeful.WakefulIntentService;

public class MyBroadcastReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        WakefulIntentService.sendWakefulWork(context, MyService.class);
    }
}

包含启动意图调度代码的服务onDestroy()

public class MyService extends WakefulIntentService {

(...)

    @Override
    public void doWakefulWork(Intent intent) {
        (...)
    }

    @Override
    public void onDestroy() {

        PendingIntent pi = PendingIntent.getBroadcast(this, 0, new Intent("com.mypackage.myapp"), 0);

        AlarmManager am = (AlarmManager)(this.getSystemService(Context.ALARM_SERVICE));

        am.set(AlarmManager.ELAPSED_REALTIME_WAKEUP, ONE_MINUTE, pi);

        super.onDestroy();
    }
}
4

2 回答 2

2

我在 KitKat 上遇到了相同的症状(请参阅此错误:http ://code.google.com/p/android/issues/detail?can=2&start=0&num=100&q=&colspec=ID%20Type%20Status%20Owner%20Summary %20Stars&groupby=&sort=&id=61850)。您可以尝试查看更改 Intent 的签名(例如操作)是否会产生影响。

于 2013-11-26T08:54:17.020 回答
2

实际上它不会在启动时发生,但是当我尝试在 adb shell 上手动启动它时:am startservice com.mypackage.myapp/.MyService。

然后不要那样做。您的用户不会这样做。导出服务只是为了让您可以运行adb shell命令,这并不是一个特别明智的举措。此外,您可以测试从 发送启动时广播adb shell,达到相同的目的,而不必导出服务。

删除 new Intent() 中的操作字符串后,我无法在启动时启动服务

抱歉,我的意思是您的第二个操作字符串。你<receiver>应该看起来像:

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

相应的PendingIntent将是:

PendingIntent pi = PendingIntent.getBroadcast(this, 0, new Intent(this, MyBroadcastReceiver.class), 0);
于 2013-06-13T18:03:39.893 回答