rust中的条件编译

1、示例:对不同的操作系统调用不同的代码。

use libc;

#[cfg(target_os="linux")]
fn my_print(){
    unsafe{
        libc::printf("hello".as_ptr() as *const libc::c_char);
    }
}
#[cfg(target_os="windows")]
fn my_print(){
    unsafe{
        libc::puts("hello".as_ptr() as *const libc::c_char); 
    }
}

fn main() {
     my_print();
}

2、cfg宏

    if cfg!(target_os="windows"){
        println!("windows");
    }
    else{
        println!("not windows")
    }

if与else代码分支的都会编译。以下代码无法通过编译,因为在windows下生成时,libc中没有printf函数:

    if cfg!(target_os="windows"){
        libc::printf("hello".as_ptr() as *const libc::c_char);
    }
    else{
        libc::puts("hello".as_ptr() as *const libc::c_char); 
    }

 

更多的内容可参考:https://www.cntofu.com/book/55/content/Conditional%20Compilation%20%E6%9D%A1%E4%BB%B6%E7%BC%96%E8%AF%91.md

Rust有一个特殊的属性,#[cfg],它允许你基于一个传递给编译器的标记编译代码。它有两种形式:

#[cfg(foo)]
# fn foo() {}

#[cfg(bar = "baz")]
# fn bar() {}

它还有一些帮助选项:

#[cfg(any(unix, windows))]
# fn foo() {}

#[cfg(all(unix, target_pointer_width = "32"))]
# fn bar() {}

#[cfg(not(foo))]
# fn not_foo() {}

这些选项可以任意嵌套:

#[cfg(any(not(unix), all(target_os="macos", target_arch = "powerpc")))]
# fn foo() {}

至于如何启用和禁用这些开关,如果你使用Cargo的话,它们可以在你Cargo.toml中的[features]部分设置:

[features]
# no features by default
default = []

# The “secure-password” feature depends on the bcrypt package.
secure-password = ["bcrypt"]

当你这么做的时候,Cargo传递给rustc一个标记:

--cfg feature="${feature_name}"

这些cfg标记集合会决定哪些功能被启用,并且因此,哪些代码会被编译。让我们看看这些代码:

#[cfg(feature = "foo")]
mod foo {
}

如果你用cargo build --features "foo"编译,他会向rustc传递--cfg feature="foo"标记,并且输出中将会包含mod foo。如果我们使用常规的cargo build编译,则不会传递额外的标记,因此,(输出)不会存在foo模块。

cfg_attr

你也可以通过一个基于cfg变量的cfg_attr来设置另一个属性:

#[cfg_attr(a, b)]
# fn foo() {}

如果a通过cfg属性设置了的话这与#[b]相同,否则不起作用。

cfg!

cfg![语法扩展](Compiler Plugins 编译器插件.md)也让你可以在你的代码中使用这类标记:

if cfg!(target_os = "macos") || cfg!(target_os = "ios") {
    println!("Think Different!");
}

这会在编译时被替换为一个truefalse,依配置设定而定。

posted @ 2020-04-24 15:06  gelare  阅读(3066)  评论(0编辑  收藏  举报