更新:
错误:jbyte* elements = (*env)->GetByteArrayElements(env, array, NULL);
仅返回8个字节。提供检索字节格式jbytearray的方法的任何替代方法。
我是JNI的新人,所以我对JNI和英语都不熟悉。
现在我尝试使用Java中的文件读取简单的JNI程序,并使用C.
将其写入文件文件阅读java代码:
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教程网站。
已经尝试过的解决方案:(但不成功)
A correct way to convert byte[] in java to unsigned char* in C++, and vice versa?
Converting jbyteArray to a character array, and then printing to console
How to convert jbyteArray to native char* in jni?
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);
GetByteArrayElements
jbyte长度应为8,但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
答案 0 :(得分:3)
正如@Olaf所观察到的,指针和数组完全是不同类型的对象。没有有意义的方法将一个转换为另一个。因此,第一步是更好地描述您的真实含义。
由于你的最终目标似乎是通过fputs()
将Java数组的字节写入文件,所以你想要做的就是创建一个C字符串,其内容与Java相同字节数组。这正是奥拉夫提出的愚蠢行为Converting jbyteArray to a character array, and then printing to console所要求的,以及接受的答案意图提供的内容。但是,您声称已经尝试过该解决方案并且无效。你没有描述失败的本质,但我可以相信它确实失败了,因为接受的答案中的解决方案有点儿错误。在下文中,我将该答案称为“历史性答案”。
历史答案在几个方面是非常正确的。特别是,C字符串必须以空值终止,并且Java字节数组的内容不会终止,除非意外。此外,jbytearray
指针不是直接指向数据的指针,因此您需要使用适当的JNI函数来获取数据本身。您需要创建一个足够大的新缓冲区,用于字节数组的内容 plus 终结符,将字节复制到其中,然后添加终结符。
例如:
// 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);
之后,您在buffer
指向的空间中有一个以空字符结尾的字节数组内容副本,例如,您可以将其传递给fputs()
:
int result = fputs(buffer, file);
此外,一旦成功分配了缓冲区,您必须确保在从函数返回之前释放它,包括通过任何错误处理返回路径:
free(buffer);
我看到的历史答案的主要问题是它建议使用strlen()
来计算数组数据的长度,以便能够分配足够大的临时缓冲区。但是,只有当字节数组已经以空值终止时,这才有效,在这种情况下,首先不需要它。
以上回答了提出的问题 - 如何将数据转换为C字符串。但请注意,问题本身的前提是假设将数据转换为C字符串并通过fputs()
输出它们是首先适当的机制。正如@Michael观察到的那样,如果数据包含空字节则不是这种情况,如果它们最初来自二进制文件可能很难排除。
如果总体目标只是将字节写入文件,那么首先将它们转换为C字符串是没有意义的。在没有首先执行这种转换的情况下,存在用于输出数据的替代机制。如果可以依赖数据不包含内部空字节,那么可以使用fprintf()
来编写它们:
fprintf("%*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);
}