如何在回调函数中发送广播

时间:2019-01-10 09:36:16

标签: java multithreading java-native-interface native broadcast

我创建了一个android项目,项目是服务,没有UI。 Java层(函数名称为onReceive)将接收广播到startservice。然后将调用本机c函数(函数名称为startService)。 C函数将创建一个线程,在该线程中,它将在Java中调用函数(函数名称为SendLaResult(native)和SendLaResult(java))。 Java函数会将sendBroadcast发送到另一个APP。这是一个主要流程。 以下代码是示例。

1。开始工作

public void onReceive(final Context context, final Intent intent) {
...
          new Thread() {
            public void run() {
                try {
                    String str="test";
                    startService(str);
                    Log.i(TAG, "startService end!!!");
                }
                catch(Exception e) {
                    e.printStackTrace();
                }
            }
        }.start();
...
}

2。 native-lib.cpp,使用Java调用功能。

void SendResult( LaResult         laResult)
{
    bool isAttached = false;
    if(!isAttached) {
        int getEnvStat = g_JavaVM->GetEnv((void **)&g_env, JNI_VERSION_1_6);
        if(getEnvStat < 0) {
            jint status = g_JavaVM->AttachCurrentThread(&g_env, NULL);
            if(status >= 0) {
                isAttached = true;
            }
        }
    }
    if(g_env == NULL) {
        LOGE("g_env is NULL!!!");
    }

    jshort conf = laResult.conf ;
    jshort nls = laResult.nls ;
    jshort egoln = laResult.egoln ;

    g_laresult_constructor = g_env->GetMethodID(g_callback_laresult,"<init>","(SSSI[S)V");
    if(g_laresult_constructor == NULL) {
        LOGE("GetMethodID laresult error!!!");
    }

    jobject jparams = g_env->NewObject(g_callback_laresult, g_laresult_constructor, conf, nls, egoln);
    if(jparams == NULL) {
        char * str=strerror(errno);
        LOGE("NewObject jparams error %s !!!" ,str);
    }

    g_mid_sendlaresult = g_env->GetMethodID(g_callback_laservice, "SendLaResult", "(Lcom/xxx/xxx/xxxx/LaResult;)V");
    if(g_mid_sendlaresult == NULL) {
        LOGE("GetMethodID g_mid_sendlaresult error!!!");
    }

    jobject serviceobject = g_env->AllocObject(g_callback_laservice);
    if(serviceobject == NULL) {
        LOGE("AllocObject serviceobject error!!!");
    }

    g_env->CallVoidMethod(serviceobject, g_mid_sendlaresult, jparams);

    g_env->DeleteLocalRef(jparams);
    if(isAttached ) {
        g_JavaVM->DetachCurrentThread();
    }

    return;
}

3.LaService.java,将广播发送到另一个APK

public void SendLaResult(LaResult laresult) {
    Intent intent = new Intent();
    intent.setAction( "com.xxx.xxx.xxx.LA_RESULT" );
    intent.putExtra( "conf", laresult.conf);
    Log.d( "confidence", Integer.toString(laneresult.confidence) );
    intent.putExtra( "nls", laresult.nls);
    Log.d( "nLanes", Integer.toString(laneresult.nLanes) );
    intent.putExtra( "egoln", laresult.egoln);

    sendBroadcast( intent );
}

4。调用sendBroadcast时,会发生以下崩溃。

--------- beginning of crash
01-02 18:05:11.505 13501-13537/com.xxx.xxx.xxxxx
E/AndroidRuntime: FATAL EXCEPTION: Thread-9
Process: com.xxx.xxx.xxxx, PID: 13501
java.lang.NullPointerException: Attempt to invoke virtual method 'void android.content.Context.sendBroadcast(android.content.Intent)' on a null object reference
at android.content.ContextWrapper.sendBroadcast(ContextWrapper.java:431)
at com.xxx.xxx.xxxxx.xxxxxx.SendLaResult(LaService.java:94)

我发现sendBroadcast是异步的,它最终被系统调用,并立即返回。因此,我在函数SendResult中删除了以下源代码。

g_JavaVM->DetachCurrentThread();

不会发生该崩溃,但是,会出现以下错误日志,请参考函数SendResult中的源代码。

01-02 18:15:03.696 13707-13743/com.xxx.xxx.xxxx
E/native-lib: NewObject jparams error Success !!!

如何解决第一个问题(发送广播崩溃) 或如何解决发送问题(NewObject返回null) 请提出一些建议。

1 个答案:

答案 0 :(得分:0)

我已解决此问题。 不要在功能SendLaResult中发送广播,不要将每个LaResult存储到列表中,请在Oncreate中创建另一个线程,在该线程中sendboadcast列表中的每个项目。