c中的fprintf()在通过java本机接口调用时生成垃圾文件名

u4vypkhs  于 2021-06-30  发布在  Java
关注(0)|答案(1)|浏览(274)

我正在做一个java程序,通过jni调用一些c函数。我有一种情况,我收集了一堆c语言的计算机能量读数,想把它们写进一个文件。理想情况下,我想使用 fprintf() ,而不是将所有数据格式化为一个字符串,通过jni接口将其传递给java,然后以java方式将其写入文件,但这似乎效率要低得多。
唯一的问题是 fprintf() 在c中,我得到的输出文件有一个垃圾名称。绝对不是我提供的文件名。

alejandro@alejandro-ThinkPad-E15:~/throwawayfiles$ ls
''$'\360\210\025\032\a'

不过,文件的内容是我所期望的。我也在c中调用了这个函数,它工作得很好,唯一的问题是当我方便地从java程序调用它时。有没有办法确保输出文件的名称是我真正想要的?或者这只是jni的问题之一,我必须处理。就像我说的,计划b是把所有的数据放在一个字符串中,发送到java,然后从那里开始写文件,但是这很慢,而且代码我不想写:)
谢谢您!
用于写入文件的代码。我正在处理一个数据结构asyncenergymonitor,它收集能量读数。所有的读取和数据存储都是用c语言完成的,但都是在java程序的大环境中进行的。

private native static void writeToFileFromC(String filePath);

public void writeToFile(String filePath)
{
    writeToFileFromC(filePath);
}
JNIEXPORT void JNICALL
Java_jrapl_AsyncEnergyMonitorCSide_writeToFileFromC(JNIEnv* env,
  jclass jcls, const char* filepath)
{
    writeToFile(monitor, filepath);
}

这里是文件最初打开的地方,我在头行中写入。

void writeToFile(AsyncEnergyMonitor *monitor, const char* filepath){
    FILE * outfile = (filepath) ? fopen(filepath,"w") : stdout;

    fprintf(outfile,"samplingRate: %d milliseconds\n",monitor->samplingRate);
    fprintf(outfile,"socket,dram,gpu,core,pkg,timestamp(usec since epoch)\n");

    if (USING_DYNAMIC_ARRAY)
        writeToFile_DynamicArray(outfile, monitor->samples_dynarr);
    if (USING_LINKED_LIST)
        writeToFile_LinkedList(outfile, monitor->samples_linklist);

    if (filepath) fclose(outfile);
}

下面是我用来编写其余数据的两个函数,这取决于数据是存储在链表中还是存储在动态数组中。

void
writeToFile_DynamicArray(FILE* outfile, DynamicArray* a) {
    for (int i = 0; i < a->nItems; i++) {
        EnergyStats current = a->items[i];
        char csv_string[512];
        energy_stats_csv_string(current, csv_string);
        fprintf(outfile,"%s\n",csv_string);
    }   
}
void
writeToFile_LinkedList(FILE* outfile, LinkedList* l) {
    LinkNode* current = l->head;
    while(current != NULL) {
        int upperbound = (current == l->tail) ?
            (l->nItemsAtTail) : (NODE_CAPACITY);
        for (int i = 0; i < upperbound; i++) {
            char ener_string[512];
            energy_stats_csv_string(current->items[i], ener_string);
            fprintf(outfile,"%s\n",ener_string);
        }
        current = current->next;
    }   
}
9q78igpj

9q78igpj1#

我忘了显式地将文件路径名从java字符串转换为c字符串。与从c写文件无关。只是做了一个垃圾字符串的名字,因为我没有转换

JNIEXPORT void JNICALL
Java_jrapl_AsyncEnergyMonitorCSide_writeToFileFromC(JNIEnv* env,
  jclass jcls, const char* filepath)
{
    writeToFile(monitor, filepath);
}

固定到

JNIEXPORT void JNICALL
Java_jrapl_AsyncEnergyMonitorCSide_writeToFileFromC(JNIEnv* env, jclass jcls, 
  jstring jstringFilepath)
{
    const char* filepath = (*env)->GetStringUTFChars(env, jstringFilepath, NULL);
    writeToFile(monitor, filepath);
    (*env)->ReleaseStringUTFChars(env, jstringFilepath, filepath);
}

现在一切都好了。

相关问题