gpt4 book ai didi

java - IntentService 中未调用 OnHandleIntent()

转载 作者:太空宇宙 更新时间:2023-11-03 10:39:08 25 4
gpt4 key购买 nike

我知道以前有人问过这个问题,但我已经遍历了所有我能找到的答案,但仍然无法解决问题。

问题是当 BroadcastReceiver 启动时,IntentService onHandleIntent() 没有被调用。奇怪的是,构造函数确实运行了(正如我从日志输出中看到的那样)。这是我的代码:

NoLiSeA.class(此类包含启动我的服务的 BroadcastReceiver)

public void toProcess(StatusBarNotification sbn) {  
LocalBroadcastManager.getInstance(this).registerReceiver(notificationForwarder, new IntentFilter("to_forward"));
Intent intent = new Intent("to_forward");
intent.putExtra("sbn", sbn);
LocalBroadcastManager.getInstance(this).sendBroadcast(intent);
Log.i("NoLiSe.TAG", "toProcess");
}

private BroadcastReceiver notificationForwarder = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
Log.i("NoLiSe.TAG", "BroadCastReceiver.onReceive");
Intent i = new Intent(context, CoreTwoA.class);
i.putExtras(intent);
startService(i);
}
}
};

CoreTwoA.class(这是 IntentService。onHandleIntent() 未被调用,正如我所见,因为控制台中没有日志文本。)

public class CoreTwoA extends IntentService {

private TextToSpeech mtts;

public CoreTwoA() {
super("TheCoreWorker");
Log.d("source", "exception", new Exception());
Log.i("CoreTwoA.TAG", "Constructor");
}

@Override
protected void onHandleIntent(Intent intent) {
Log.i("CoreTwoA.TAG", "onHandleIntent");

}
}

AndroidManifest.xml

    <service
android:name=".CoreTwoA"
android:label="@string/service_name"
android:exported="false">
</service>

更新

因此,根据下面的讨论,我能够将问题缩小到 BroadCastReceiver 中的以下代码行:

i.putExtra("sbn", sbn) 

如果我删除它,即不向 Intent 添加任何额外内容,那么我的 IntentService 中的 onHandleIntent() 方法就会运行。

如果包含它,则 onHandleIntent() 不会运行,并且以下内容由我的 IntentService 的构造函数中的 Log.d() 写入 logcat

06-10 19:40:35.355 25094-25094/com.dezainapps.myapp D/source: exception
java.lang.Exception
at com.dezainapps.myapp.CoreTwoA.<init>(CoreTwoA.java:20)
at java.lang.Class.newInstance(Native Method)
at android.app.ActivityThread.handleCreateService(ActivityThread.java:2859)
at android.app.ActivityThread.-wrap4(ActivityThread.java)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1427)
at android.os.Handler.dispatchMessage(Handler.java:102)
at android.os.Looper.loop(Looper.java:148)
at android.app.ActivityThread.main(ActivityThread.java:5417)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)
06-10 19:40:35.355 25094-25094/com.dezainapps.myapp I/CoreTwoA.TAG: Constructor

知道为什么通过 Intent 将实现 Parcelable 的 StatusBarNotification 对象传递给 IntentService 不起作用吗?奇怪的是,通过 Intent (请参阅代码)从我的 toProcess() 方法广播相同的 StatusBarNotfication sbn 对象确实有效。

最佳答案

我和OP有同样的问题。

问题原来是我通过启动 Intent 服务。诊断非常困难,因为它在客户的设备上“无声”地失败了,所以我没有收到 ACRA 错误报告。

无论如何,我使用了一个快速而肮脏的技巧来解决这个问题 - 基本上是将 ArrayList 存储在我用来设置/获取 ArrayList 的静态元素中,而不是尝试通过 Intent 。

在进一步调查(对其他设备进行一些测试)时,我发现抛出了 TransactionTooLargeExceptionMore discussion on that here .

如果有人想复制,这是我的测试代码:

测试代码

如果您想让它工作,请将 4000 值更改为更小的值。

ArrayList<ParcelableNameValuePair> testArrayList = new ArrayList<>();
for (int i = 0; i < 4000; i++) {
testArrayList.add(new ParcelableNameValuePair("name" + i, "value" + i));
}
TestIntentService.startAction(AdminHomeActivity.this, testArrayList);

TestIntentService.java

public class TestIntentService extends IntentService {

private static final String LOG_TAG = TestIntentService.class.getSimpleName();

private static final String TEST_ACTION = "com.example.action.FOO";

private static final String EXTRA_PARAM1 = "com.example.extra.PARAM1";
private static final String EXTRA_PARAM2 = "com.example.extra.PARAM2";

public TestIntentService() {
super("TestIntentService");
}

public static void startAction(Context context, ArrayList<ParcelableNameValuePair> testArrayList) {

Log.d(LOG_TAG, "1. startAction()");
Utilities.makeToast(context, "1. startAction()");

try {
int arrayListSize = (testArrayList == null) ? -1 : testArrayList.size();
Log.d(LOG_TAG, "2. arrayListSize: " + arrayListSize);
Utilities.makeToast(context, "2. arrayListSize: " + arrayListSize);

Intent intent = new Intent(context, TestIntentService.class);
intent.setAction(TEST_ACTION);
//intent.putExtra(EXTRA_PARAM1, testArrayList);
intent.putParcelableArrayListExtra(EXTRA_PARAM2, testArrayList);

/**
* This line should result in a call to onHandleIntent() but, if we're sending a huge ArrayList, it doesn't...
*/
context.startService(intent);
}
catch(Exception e) {
Log.e(LOG_TAG, "Exception starting service", e);
Utilities.makeToast(context, "Exception starting service: " + e);
}
}

@Override
protected void onHandleIntent(Intent intent) {

Log.d(LOG_TAG, "3. onHandleIntent()");
Utilities.makeToast(getApplicationContext(), "3. onHandleIntent()");

try {
if (intent != null) {
final String action = intent.getAction();
if (TEST_ACTION.equals(action)) {

ArrayList<ParcelableNameValuePair> testArrayList = intent.getParcelableArrayListExtra(EXTRA_PARAM1);
int testArrayListSize = (testArrayList == null) ? -1 : testArrayList.size();
Log.d(LOG_TAG, "4. testArrayListSize: " + testArrayListSize);
Utilities.makeToast(getApplicationContext(), "4. testArrayListSize: " + testArrayListSize);

ArrayList<ParcelableNameValuePair> testArrayList2 = intent.getParcelableArrayListExtra(EXTRA_PARAM2);
int testArrayList2Size = (testArrayList2 == null) ? -1 : testArrayList2.size();
Log.d(LOG_TAG, "5. testArrayList2Size: " + testArrayList2Size);
Utilities.makeToast(getApplicationContext(), "5. testArrayList2Size: " + testArrayList2Size);

}
}
}
catch(Exception e) {
Log.e(LOG_TAG, "Exception handling service intent", e);
Utilities.makeToast(getApplicationContext(), "Exception handling service intent: " + e);
}
}

}

ParcelableNameValuePair.java

public class ParcelableNameValuePair implements Parcelable {

private String name, value;

public ParcelableNameValuePair(String name, String value) {
this.name = name;
this.value = value;
}

public String getName() {
return name;
}

public String getValue() {
return value;
}

@Override
public int describeContents() {
return 0;
}

@Override
public void writeToParcel(Parcel out, int flags) {
out.writeString(name);
out.writeString(value);
}

public static final Parcelable.Creator<ParcelableNameValuePair> CREATOR = new Parcelable.Creator<ParcelableNameValuePair>() {
public ParcelableNameValuePair createFromParcel(Parcel in) {
return new ParcelableNameValuePair(in);
}

public ParcelableNameValuePair[] newArray(int size) {
return new ParcelableNameValuePair[size];
}
};

private ParcelableNameValuePair(Parcel in) {
name = in.readString();
value = in.readString();
}

}

就像我说的,我使用了一个快速而肮脏的解决方案来解决这个问题。我认为更好的解决方案是应用程序将 ArrayList 写入文件系统,然后通过 Intent 将对该文件的引用(例如,文件名/路径)传递给 IntentService 然后让 IntentService 检索文件内容并将其转换回 ArrayList

IntentService 处理完文件后,它应该删除它或通过本地广播将指令传回应用以删除它创建的文件(传回提供给它的相同文件引用)。

关于java - IntentService 中未调用 OnHandleIntent(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37738262/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com