- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
更新:
错误: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
最佳答案
正如@Olaf 观察到的,指针和数组是完全不同类型的对象。没有有意义的方法可以将一个转换为另一个。因此,第一步是更好地描述您的实际意思。
由于您的最终目标似乎是通过 fputs()
将 Java 数组的字节写入文件,因此您似乎想要做的是创建一个 C 字符串,其内容是与 Java 字节数组相同。这正是 Olaf 提出的骗局,Converting jbyteArray to a character array, and then printing to console ,请求,以及接受的答案声称提供的内容。但是,您声称您已经尝试过该解决方案但没有奏效。你没有描述失败的性质,但我可以相信它确实失败了,因为接受的答案中的解决方案有点错误。在下文中,我将该答案称为“历史性答案”。
历史答案在几个点上是相当正确的。特别是,C 字符串必须以 null 终止,而 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);
之后,您在 buffer
指向的空间中拥有字节数组内容的空终止副本,例如,您可以将其传递给 fputs()
:
int result = fputs(buffer, file);
此外,一旦您成功分配了缓冲区,您必须确保在从函数返回之前释放它,包括通过任何错误处理返回路径:
free(buffer);
我在历史答案中看到的主要问题是它建议使用 strlen()
来计算数组数据的长度,以便能够分配足够大的临时缓冲区。但这只有在字节数组已经以 null 终止时才有效,在这种情况下,一开始就没有必要。
以上回答了提出的问题——如何将数据转换为 C 字符串。但是请注意,问题本身的前提是假设将数据转换为 C 字符串并通过 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);
}
关于java - 如何在 JNI 中将 Java 字节数组的内容转换为 C 字符串?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38768615/
我刚开始使用 JNI,但遇到以下问题。 我有一个包含简单类的 C++ 库。我从 Java Android 项目中调用了三个 JNI 方法,分别实例化所述类、调用实例化类的方法并销毁它。我保留了对该对象
背景 我有一个 android 项目,它使用 JNI(使用 NDK)以 Java 和 C/C++ 进行编码。 我在java端创建了一个Jni java包装器,它将自己完成所有Jni操作,而除了这个包装
我想传递一个java对象的地址,JNI方法应该填充所传递对象的字段。1. java代码传递给定的对象引用。 JNI 方法应该能够缓存这个对象。这是一次性通话。2. 然后,java 对象使用不同的参数重
我本质上是在尝试遵循描述的 SO 解决方案 here , 但我遇到了问题。 这基本上就是我在 JNI 库中所做的事情: handle = dlopen("/data/data/lib/my.packa
我试图实现某种异常处理 一开始我打电话 jni::ExceptionDescribe() 之后我使用了的实现 How to obtain a description of a Java excepti
是否可以从不同 jni 库的另一个方法调用 1 个库的 jni 方法?例如:我有 2 个库 lib_1.so 和 lib_2.so。 我想从 lib_2.so 调用 lib_1.so 的方法 get_
我想在native方法中使用动态注册,所以我需要设置JNI_onLoad功能。我只是写了一个函数来获取两个数字的总和。但是,它无法正确构建。我该如何更正错误? 这是我的 *.cpp 文件,我将此文件命
我已经为 C 头文件制作了一个 make 文件,它工作正常,但是说 JNICALL 和 JNIEnv 存在语法错误,但我已经弄明白了这是因为头文件中的类型。 Image of the failure
我需要实现一个本地方法,比方说“public native void someFunc();”。我有两个库,libabc.so 和 libdef.so。 Java 使用 System.loadLibr
背景 我正在 eclipse 中为 android 开发一个应用程序,现在我遇到了一个问题,我需要你的帮助。所以我必须从 JAVA 应用程序调用用 C 编写的函数。但是在我编写代码的过程中,我有一些问
我正在使用 Android 上的 Java native 接口(interface)将当前 Activity 传递给 native 方法。但我没有使用类似 JNI 的函数名称来执行此操作。我正在手动注
我正在使用 JNI,我想知道是否可以通过 delegate 进行通信。 例如: Kotlin typealias MessageReceived = (msg: String) -> Unit ext
看来我对 JNI 的运气并不好。我正在等待我买的书到货,但现在是试错法。 我正在使用 JNI 来实现 Lua 求值器。 evaluatorNew() 只是创建一个新的 Evaluator() 对象,创
我有下面的代码,我想调用在同一个源文件中实现的函数,在本例中使用 C 语言: JNIEXPORT jstring JNICALL MyClass_get_1Uname__C (JNIEnv *env,
硬件手机和平板电脑内存太少,但 HAXM 工作正常? 我的基本问题是我的应用程序(很可能是我的 Java 应用程序加载的 JNI 动态库)太大。如果未使用硬件电话和模板调用 JNI 指令,Java 应
我正面临崩溃 JNI WARNING : 0x44f81e80 is not a valid JNI reference, in Ldalvik/system/NativeStart;. run()v
我移植了很多数学知识。我正在使用 over to c++ from java 并看到这样做有很大的性能提升,但我无法弄清楚要使用什么 jni 函数来摆脱我不再需要的变量。例如,我知道当您的 jni 方
我正在使用 JNI 调用一个静态 java 方法,该方法又创建一个 Swing JFrame 并显示它。代码相当简单,Java 代码独立运行(即 java StartAWT 做它应该做的事),而当使用
我正在尝试创建一个新线程,因此我将 VM 从我的方法初始化(从 Java 调用)传递到我的新线程。在线程中,我调用 AttachCurrentThread 并获取 JNIEnv* env。 稍后,我尝
我想知道是否有可能从java调用C++方法。 我非常希望能够从 java 读取内存进程。 我懂c++,但我想使用像java这样的更高级别,但仍然能够侵入进程内存。 有什么线索吗? []的 最佳答案 这
我是一名优秀的程序员,十分优秀!