当前位置: 首页 > 知识库问答 >
问题:

如何将JAVA字节数组的内容转换为JNI中的C字符串?

徐洛华
2023-03-14

更新:

错误:< code>jbyte* elements = (*env)-

我刚到JNI,所以我不熟悉JNI和英语。

现在,我尝试用Java编写简单的JNI文件读取程序,并用c语言将其写入文件

文件读取

public class FileIO {
   static {
      System.loadLibrary("io");         
   }

   private native void writeToFile(byte[] msg);

   public static void main(String[] args) throws IOException { 
      FileInputStream fileInputStream=null;

      File file = new File("version1-1.png");

      byte[] bFile = new byte[(int) file.length()];

      try {
         //convert file into array of bytes
         fileInputStream = new FileInputStream(file);
         fileInputStream.read(bFile);
         fileInputStream.close();
         System.out.println("Done");

      } catch(Exception e){
         e.printStackTrace();
      }

      new FileIO().writeToFile(bFile); 
   }
}

文件编写C代码

#include <jni.h>
#include <stdio.h>
#include <string.h>
#include "FileIO.h"

JNIEXPORT void JNICALL Java_FileIO_writeToFile (JNIEnv *env, jobject job, jbyteArray array ){

    char * buf ; 

   // here what i do ???    :(

    FILE *file = fopen("test123.png", "w");

    int results = fputs(buf, file);
    if (results == EOF) {
        //Failed to write do error code here.
    }
    fclose(file);
}

我已经尝试了许多解决方案(下面的链接),但没有运气将其写入文件。请提供正确的解决方案和最佳的JNI教程网站。

已经尝试解决方案:(但不成功)

将java中的字节[]转换为C中的无符号char*的正确方法,反之亦然?

将 jbyte 数组转换为字符数组,然后打印到控制台

jni中如何将jbyteArray转换为native char*?

int len = (*env)->GetArrayLength (env , array );
printf(" Length of the bytearray %d\n", len );
unsigned char * string ;
string = (char *)malloc((len + 1) * sizeof(char)) ;

jbyte* b = (*env)->GetByteArrayElements(env, array, &isCopy);

在< code > GetByteArrayElements 之后,jbyte长度应为8,但< code>GetArrayLength返回的bytearray长度为50,335。

我尝试的是:

JNIEXPORT void JNICALL Java_HelloJNI_sayHello (JNIEnv *env, jobject job, jbyteArray array ){

    jsize num_bytes = (*env)->GetArrayLength(env, array);
    char *buffer = malloc(num_bytes + 1);

    printf("Number of jByte element    : %d\n", (int) num_bytes);

    if (!buffer) 
        printf("Buff Fail\n");

    jbyte* elements = (*env)->GetByteArrayElements(env, array, NULL);
    if (!elements)
        printf("Element Fail\n");

    printf ("Number of Byte elements   : %d\n", (int) strlen (elements));

    memcpy(buffer, elements, num_bytes);
    buffer[num_bytes] = 0;

    printf("Number of buffer elements  : %d\n", (int) strlen(elements));

    (*env)->ReleaseByteArrayElements(env, array, elements, JNI_ABORT);

    FILE *fp;
    fp = fopen( "file.txt" , "w" );
    fwrite(buffer , 1 , sizeof(buffer) , fp );
    fclose(fp);
    return;
}

输出:

Done
Number of jByte element   : 50335
Number of Byte elements   : 8
Number of buffer elements : 8

共有1个答案

孙清野
2023-03-14

正如@Olaf所观察到的,指针和数组完全是不同种类的对象。没有有意义的方法来将一个转换成另一个。因此,第一步是更好地描述你真正的意思。

因为您的最终目标似乎是通过< code>fputs()将Java数组的字节写入一个文件,所以您想要做的似乎是创建一个C字符串,其内容与Java字节数组的内容相同。这正是Olaf提出的dupe,将jbyteArray转换成一个字符数组,然后打印到控制台,请求,以及接受的答案声称提供的内容。然而,您声称您已经尝试过该解决方案,但它不起作用。你没有描述失败的性质,但我可以相信它确实失败了,因为公认答案中的解决方案有点漏洞。在下文中,我将把这个答案称为“历史性的答案”。

历史性的答案在几个问题上是相当正确的。特别是,C 字符串必须以空值终止,并且 Java 字节数组的内容不会终止,除非意外终止。此外,jbytearray 指针不是直接指向数据的指针,因此您需要使用适当的 JNI 函数来获取数据本身。您需要创建一个足够大的新缓冲区,用于字节数组的内容和终止符,将字节复制到其中,然后添加终止符。

例如:

// determine the needed length and allocate a buffer for it
jsize num_bytes = GetArrayLength(env, array);
char *buffer = malloc(num_bytes + 1);

if (!buffer) {
    // handle allocation failure ...
}

// obtain the array elements
jbyte* elements = GetByteArrayElements(env, array, NULL);

if (!elements) {
    // handle JNI error ...
}

// copy the array elements into the buffer, and append a terminator
memcpy(buffer, elements, num_bytes);
buffer[num_bytes] = 0;

// Do not forget to release the element array provided by JNI:
ReleaseByteArrayElements(env, array, elements, JNI_ABORT);

之后,在缓冲区所指向的空间中,有一个以空值终止的字节数组内容副本,例如,您可以将其传递给 fputs()

int result = fputs(buffer, file);
free(buffer);

我看到的历史答案的主要问题是,它建议使用strlen()来计算数组数据的长度,以便能够分配足够大的临时缓冲区。但是,仅当字节数组已经为 null 终止时,这才有效,在这种情况下,首先不需要这样做。

上面回答了提出的问题——如何将数据转换成C字符串。但是,请注意,问题本身是基于这样的假设:首先,将数据转换为C字符串并通过< code>fputs()输出它们是一种合适的机制。正如@Michael所观察到的,如果数据包含空字节,情况就不是这样了,因为如果它们最初来自二进制文件,可能很难排除。

如果总体目标只是将字节写入文件,那么首先将它们转换为C字符串是毫无意义的。有一些替代机制可以在不首先执行此类转换的情况下输出数据。如果可以依赖数据来不包含内部空字节,则可以使用 fprintf() 来写入它们:

fprintf(file, "%*s", (int) num_bytes, (char *) elements);

另一方面,如果数据可能包含空值,那么您应该使用适当的低级输出函数。可能是这样的:

#include <stdio.h>
#include <jni.h>
#include "FileIO.h"

JNIEXPORT void JNICALL Java_FileIO_writeToFile(JNIEnv *env, jobject job,
        jbyteArray array) {
    FILE *fp = fopen( "file.txt" , "w" );

    if (!fp) {
        // handle failure to open the file ...
    }

    // determine the needed length and allocate a buffer for it
    jsize num_bytes = GetArrayLength(env, array);

    // obtain the array elements
    jbyte* elements = GetByteArrayElements(env, array, NULL);

    if (!elements) {
        // handle JNI error ...
    }

    // output the data
    if (fwrite(elements, 1, num_bytes, fp) != num_bytes) {
        // handle I/O error ...
    }

    // Do not forget to release the element array provided by JNI:
    ReleaseByteArrayElements(env, array, elements, JNI_ABORT);

    fclose(fp);
}
 类似资料:
  • 问题内容: 我有一个一维String数组,我想将其转换为一维字节数组。我该怎么做呢?这需要ByteBuffer吗?我怎样才能做到这一点?(字符串可以是任意长度,只想知道如何执行这样的操作。将其转换为字节数组后,如何将其转换回String数组? -担 问题答案: 数组到数组 ,应该 手动 将其转换为两边,但是如果只有一个String,则可以和; 像这样 对于string []中的一个字节[],您必须

  • 问题内容: 如何将Java字符串转换为ASCII字节数组? 问题答案: 使用该方法,为其指定适当的名称(或名称)。 例: (Java 7的前:)

  • 我得到一个错误读数:

  • 问题内容: 我有一个要加密的字节数组,然后转换为字符串,以便可以传输。当我收到字符串时,我必须将字符串转换回字节数组,以便可以对其进行解密。我检查了接收到的字符串是否与发送的字符串(包括长度)匹配,但是当我使用诸如str.getBytes()之类的东西将其转换为字节数组时,它与我的原始字节数组不匹配。 示例输出: 任何想法如何将接收到的字符串转换为与发送的字节数组匹配的字节数组? 谢谢 问题答案:

  • 问题内容: 我有以下代码,我试图通过测试,但似乎无法理解Java世界中各种编码形式。 我想我的问题是:将任意字节的字节数组转换为Java字符串,然后将同一Java String转换为另一个字节数组的正确方法是什么,该字节数组将具有与原始字节相同的长度和相同的内容数组? 问题答案: 尝试特定的编码: ideone链接

  • 我得到了20字节的输入。使用的字符集是UTF-8 当我把它转换成字符串时,我看到0000000000000000000000000000000000000000 这是正确的。 但当我把它转换成字符串时,我得到一个空值。我怀疑字符集有问题,但即使尝试了不同的字符集,我也无法返回值“0”。 我试图改变字符集,但没能解决它,任何我可能缺少的线索。 US-ASCII ISO-8859-1 UTF-8 UT