Rust 常量表达式取决于泛型参数

rmc*_*r99 4 generics rust

我试图概括 Rust 中的一些代数运算(例如组、环、字段等),并且在尝试实现两个“集合”(Vec)的叉积时遇到了这个问题。

(请注意,这是const_generics在夜间工具链中使用的。)

fn CrossProduct<T, const N: usize, const M: usize>(lhs: Vec<[T;N]>, rhs: Vec<[T;M]>) -> Vec<[T;N+M]> {
    let mut out: Vec<[T;N+M]> = Vec::with_capacity(lhs.len() * rhs.len());

    for i in 0..lhs.len() {
        for j in 0..rhs.len() {
            let mut inner: [T; N+M] = [lhs[i][0]; N+M];
            for idx in 1..N {
                inner[idx] = lhs[i][idx];
            }
            for idx in 0..M {
                inner[idx + N] = rhs[j][idx];
            }
            out.push(inner);
        }
    }
    out
}
Run Code Online (Sandbox Code Playgroud)

无论我在哪里有表达式[T;N+M],我都会收到一个错误,指出常量表达式取决于通用参数,这是没有意义的。

Iva*_*n C 5

您的代码实际上确实在最新的 nightly 上编译,您所需要的只是几个功能标志。

#![feature(const_generics)]
#![feature(const_evaluatable_checked)]
fn CrossProduct<T: Copy, const N: usize, const M: usize>(lhs: Vec<[T;N]>, rhs: Vec<[T;M]>) -> Vec<[T;N+M]> {
    let mut out: Vec<[T;N+M]> = Vec::with_capacity(lhs.len() * rhs.len());

    for i in 0..lhs.len() {
        for j in 0..rhs.len() {
            let mut inner: [T; N+M] = [lhs[i][0]; N+M];
            for idx in 1..N {
                inner[idx] = lhs[i][idx];
            }
            for idx in 0..M {
                inner[idx + N] = rhs[j][idx];
            }
            out.push(inner);
        }
    }
    out
}

fn main() {
    let a = vec![[1i32,2,3],[1,2,3],[1,2,3]];
    let b = vec![[1,2,3],[1,2,3],[1,2,3]];
    println!("{:?}", CrossProduct(a, b));
}
Run Code Online (Sandbox Code Playgroud)

当尝试仅使用 进行编译时#![feature(min_const_generics)],编译器实际上推荐此解决方案:

error: generic parameters may not be used in const operations
 --> src/main.rs:2:102
  |
2 | fn CrossProduct<T: Copy, const N: usize, const M: usize>(lhs: Vec<[T;N]>, rhs: Vec<[T;M]>) -> Vec<[T;N+M]> {
  |                                                                                                      ^ cannot perform const operation using `N`
  |
  = help: const parameters may only be used as standalone arguments, i.e. `N`
  = help: use `#![feature(const_generics)]` and `#![feature(const_evaluatable_checked)]` to allow generic const expressions
Run Code Online (Sandbox Code Playgroud)

  • 请注意,这两个功能仍然不完整,因此即使您可以使用它们,它们也可能带来编译器错误,导致您的程序行为异常(正如编译器警告的那样)。您绝对不应该在任何生产代码中使用这些功能。 (2认同)