Compare commits

...

9 Commits

3 changed files with 141 additions and 45 deletions

View File

@@ -1,24 +1,7 @@
Rust build
Rust bindings to the JNI Demo.
Run
```shell
cd mylib
cargo build
```
Java build
```shell
cd java_src
javac me/ehlxr/HelloWorld.java
```
Java run
```shell
cd java_src
# Linux: export LD_LIBRARY_PATH=../mylib/target/debug/
# export JAVA_LIBRARY_PATH=../mylib/target/debug/
java -Djava.library.path=../mylib/target/debug/ me.ehlxr.HelloWorld
> make java_run
```

View File

@@ -14,7 +14,7 @@ class HelloWorld {
public Long no;
private String name;
public int age;
public List<String> ls;
public List<String> list;
public Map<String, Long> map;
private static native String hello(String input);
@@ -31,7 +31,7 @@ class HelloWorld {
private static native void asyncComputation(HelloWorld callback);
private static native String getFiled(HelloWorld param);
private static native List<Map<String, Object>> getField(HelloWorld param);
public static void main(String[] args) {
String output = HelloWorld.hello("Java");
@@ -67,9 +67,9 @@ class HelloWorld {
hw.setName("Jack");
hw.no = 123434555L;
hw.age = 30;
hw.ls = ls;
hw.list = ls;
hw.map = map;
System.out.println(HelloWorld.getFiled(hw));
System.out.println("get return: " + HelloWorld.getField(hw));
}
public String getName() {

View File

@@ -1,34 +1,53 @@
use jni::objects::{GlobalRef, JClass, JObject, JString, JValue};
use jni::sys::{jbyteArray, jint, jlong, jstring};
use jni::JNIEnv;
use jni::{
errors::Result,
objects::{GlobalRef, JClass, JList, JMap, JObject, JString, JValue},
sys::{jbyteArray, jint, jlong, jobject, jstring},
JNIEnv,
};
use std::{sync::mpsc, thread, time::Duration};
#[no_mangle]
pub extern "system" fn Java_me_ehlxr_HelloWorld_getFiled(
pub extern "system" fn Java_me_ehlxr_HelloWorld_getField(
env: JNIEnv,
_class: JClass,
input: JObject,
) -> jstring {
let map = env
.get_field(input, "map", "Ljava/util/Map;")
.unwrap()
.l()
) -> jobject {
// Map field operate
let jmap = env
.get_map(
env.get_field(input, "map", "Ljava/util/Map;")
.unwrap()
.l()
.unwrap(),
)
.unwrap();
let jmap = env.get_map(map).unwrap();
let v1 = jmap
.get(JObject::from(env.new_string("k1").unwrap()))
.unwrap()
.unwrap();
println!("get map key k1, value: {}", long_value(env, v1));
if let Ok(_) = jmap.put(
JObject::from(env.new_string("k9").unwrap()),
long_to_jobj(env, 9 as i64),
) {
println!("put key ok");
};
jmap.iter().unwrap().into_iter().for_each(|jmap_iter| {
let key: JString = jmap_iter.0.into();
let value = jmap_iter.1;
println!(
"get map key: {}, value: {}",
"iter map key: {}, value: {}",
String::from(env.get_string(key).unwrap()),
long_value(env, value)
);
});
// List field operate
let jlist = env
.get_list(
env.get_field(input, "ls", "Ljava/util/List;")
env.get_field(input, "list", "Ljava/util/List;")
.unwrap()
.l()
.unwrap(),
@@ -37,14 +56,16 @@ pub extern "system" fn Java_me_ehlxr_HelloWorld_getFiled(
jlist.iter().unwrap().into_iter().for_each(|jobj| {
let jstr: JString = jobj.into();
println!(
"get list filed: {}",
"get list field: {}",
String::from(env.get_string(jstr).unwrap())
);
});
// int field operate
let age = env.get_field(input, "age", "I").unwrap().i().unwrap();
println!("get age field: {}", age);
// String field operate
let name: JString = env
.get_field(input, "name", "Ljava/lang/String;")
.unwrap()
@@ -56,9 +77,7 @@ pub extern "system" fn Java_me_ehlxr_HelloWorld_getFiled(
String::from(env.get_string(name).unwrap())
);
// let no = env.get_field(input, "no", "J").unwrap().j().unwrap();
// println!("get no field: {}", no);
// Long field operate
let no = long_value(
env,
env.get_field(input, "no", "Ljava/lang/Long;")
@@ -68,6 +87,7 @@ pub extern "system" fn Java_me_ehlxr_HelloWorld_getFiled(
);
println!("get no field: {}", no);
// call method operate
let out_str = if let JValue::Object(result) = env
.call_method(input, "getName", "()Ljava/lang/String;", &[])
.unwrap()
@@ -78,11 +98,76 @@ pub extern "system" fn Java_me_ehlxr_HelloWorld_getFiled(
} else {
"".to_string()
};
println!("Hello {}! from Rust..", out_str);
let output = env
.new_string(format!("Hello {}! from Rust..", out_str))
.expect("Couldn't create java string!");
output.into_inner()
// new ArrayList operate
let list = unwrap(
&env,
JList::from_env(
&env,
unwrap(&env, env.new_object("java/util/ArrayList", "()V", &[])),
),
);
// new LinkedHashMap operate
let map = unwrap(
&env,
JMap::from_env(
&env,
unwrap(&env, env.new_object("java/util/LinkedHashMap", "()V", &[])),
),
);
// Map put key value operate
unwrap(
&env,
map.put(
JObject::from(env.new_string("no").unwrap()),
long_to_jobj(env, no),
),
);
unwrap(
&env,
map.put(
JObject::from(env.new_string("age").unwrap()),
long_to_jobj(env, age as i64),
),
);
unwrap(
&env,
map.put(
JObject::from(env.new_string("name").unwrap()),
JObject::from(name),
),
);
unwrap(
&env,
map.put(
JObject::from(env.new_string("list").unwrap()),
JObject::from(jlist),
),
);
unwrap(
&env,
map.put(
JObject::from(env.new_string("map").unwrap()),
JObject::from(jmap),
),
);
let jmap2 = map.clone();
// List add element operate
unwrap(&env, list.add(JObject::from(map)));
unwrap(&env, list.add(jmap2));
println!("list size {}", list.size().unwrap());
list.into_inner()
// let output = env
// .new_string(format!("Hello {}! from Rust..", out_str))
// .expect("Couldn't create java string!");
// output.into_inner()
}
#[no_mangle]
@@ -221,6 +306,34 @@ fn long_value(env: JNIEnv, jobj: JObject) -> i64 {
.unwrap()
}
fn long_to_jobj<'a>(env: JNIEnv<'a>, lv: i64) -> JObject<'a> {
env.call_static_method(
"Ljava/lang/Long;",
"valueOf",
"(J)Ljava/lang/Long;",
&[JValue::from(lv)],
)
.unwrap()
.l()
.unwrap()
}
pub fn print_exception(env: &JNIEnv) {
let exception_occurred = env.exception_check().unwrap_or_else(|e| panic!("{:?}", e));
if exception_occurred {
env.exception_describe()
.unwrap_or_else(|e| panic!("{:?}", e));
}
}
#[allow(dead_code)]
pub fn unwrap<T>(env: &JNIEnv, res: Result<T>) -> T {
res.unwrap_or_else(|e| {
print_exception(&env);
panic!("{:#?}", e);
})
}
#[cfg(test)]
mod tests {
#[test]