我正在使用需要的C API(特别是MPI_Init)int argc, char **argv.我正在尝试argc, argv使用以下代码生成等效代码:
let argc = std::env::args().len() as c_int;
let c_strs: ~[CString] = std::env:args().map(|s: & &str| s.to_c_str());
let mut argv: ~[*c_char] = c_strs.map(|c: &CString| c.with_ref(|ptr| ptr));
if null_terminate {
argv.push(std::ptr::null());
}
Run Code Online (Sandbox Code Playgroud)
它失败了:
error: expected type, found `~`
src/lib.rs:37 let c_strs: ~[CString] = std::env::args().map(|s: & &str| s.to_c_str());
^
Run Code Online (Sandbox Code Playgroud)
我摆脱了~它然后它找不到to_c_str(),并不确定要替换什么to_c_str,to_raw()(例如)失败.
有谁知道转换Args为更友好的C格式的方法?
我的答案适用于目前稳定的Rust(1.5),可能还有beta和nightly.
以下Rust代码调用foo(argc, argv)C中实现的函数.签名foo非常类似于main函数.
extern crate libc;
use libc::c_char;
use libc::c_int;
use std::ffi::CString;
#[link(name="foo")]
extern "C" {
fn foo(argc: c_int, argv: *const *const c_char);
}
fn main() {
// create a vector of zero terminated strings
let args = std::env::args().map(|arg| CString::new(arg).unwrap() ).collect::<Vec<CString>>();
// convert the strings to raw pointers
let c_args = args.iter().map(|arg| arg.as_ptr()).collect::<Vec<*const c_char>>();
unsafe {
// pass the pointer of the vector's internal buffer to a C function
foo(c_args.len() as c_int, c_args.as_ptr());
};
}
Run Code Online (Sandbox Code Playgroud)
注意,C端只借用指向字符串的指针.如果要存储它们,请使用strdup()它们.
我也用unwrap()的CString实例.如果您的字符串包含0个字节,则会返回错误,请参阅https://doc.rust-lang.org/stable/std/ffi/struct.CString.html#method.new.
证明:
我将此代码放入货物项目中,并libc作为依赖项添加.该foo()函数如下所示:
#include <stdio.h>
void foo(int argc, char* argv[]) {
int i;
for (i = 0; i < argc; i++) {
printf("argv[%d]: %s\n", i, argv[i]);
}
}
Run Code Online (Sandbox Code Playgroud)
我编译了这段代码:
gcc foo.c -o libfoo.so -shared -fPIC
Run Code Online (Sandbox Code Playgroud)
然后复制libfoo.so下target/debug/deps(只是在库搜索路径中).然后我运行我的货物项目:
$ cargo run the quick brown fox
Compiling args v0.1.0 (file:///home/tibi/Codes/Rust/argv/args)
Running `target/debug/args the quick brown fox`
argv[0]: target/debug/args
argv[1]: the
argv[2]: quick
argv[3]: brown
argv[4]: fox
Run Code Online (Sandbox Code Playgroud)