trait 不能做成一个对象
trait cannot be made into an object
我正在研究光线追踪器,并希望为所有可击中的对象建模以提供一个通用接口。
我实现了一个名为 Object 的特征,所有可点击对象都实现了该特征。我创建了一个名为 Intersection 的结构,它包含一个 f32 值和对实现 Object 特征的结构的引用。
代码:
use std::sync::atomic::{AtomicUsize, Ordering};
use super::ray::Ray;
use std::ops::{Index};
static mut ID : AtomicUsize = AtomicUsize::new(0);
pub trait Object {
fn intersection<'a, T: Object>(&self, ray: &Ray) -> Intersections<'a, T>;
fn get_uid() -> usize {
unsafe {
ID.fetch_add(1, Ordering::SeqCst);
ID.load(Ordering::SeqCst)
}
}
}
pub struct Intersection<'a, T: Object>{
pub t: f32,
pub obj: &'a T,
}
impl<'a, T: Object> Intersection<'a, T> {
pub fn new(t: f32, obj: &'a Object) -> Intersection<'a, T> {
Self {t, obj}
}
}
pub struct Intersections<'a, T: Object> {
pub hits: Vec<Intersection<'a, T>>,
}
impl<'a, T: Object> Intersections<'a, T> {
pub fn new() -> Self {
Self {
hits: Vec::new(),
}
}
pub fn push(&self, hit: Intersection<'a, T>) {
self.hits.push(hit);
}
pub fn len(&self) -> usize {
self.hits.len()
}
}
报错信息如下:
error[E0038]: the trait `object::Object` cannot be made into an object
--> src/object.rs:23:5
|
23 | pub fn new(t: f32, obj: &'a Object) -> Intersection<'a, T> {
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ the trait `object::Object` cannot be made into an object
|
= note: method `intersection` has generic type parameters
= note: method `get_uid` has no receiver
因为我在 Intersection 中存储了一个引用,所以我认为它不必处理结构的实际大小。
我很确定 Intersection
不应该是通用的,而应该包含 &Object
:
pub struct Intersection<'a>{
pub t: f32,
pub obj: &'a Object,
}
如果你真的需要 Intersection
中的实际对象类型,那么 Object::intersection
不应该是通用的,而应该 return 一个 Intersection<Self>
:
pub trait Object<'a> {
fn intersection(&self, ray: &Ray) -> Intersections<'a, Self>;
}
错误的第二部分涉及 get_uid
。如果你想通过引用访问特征,它不能成为特征的一部分,因为在这种情况下只能使用带有 self
参数的函数。
另请注意,get_uid
并不像您想象的那样:如果两个线程同时调用它,则两者有可能得到相同的结果。你想要的是:
fn get_object_uid() -> usize { // <- Renamed because it needs to be outside the trait
unsafe {
ID.fetch_add (1, Ordering::SeqCst) + 1
}
}
我正在研究光线追踪器,并希望为所有可击中的对象建模以提供一个通用接口。
我实现了一个名为 Object 的特征,所有可点击对象都实现了该特征。我创建了一个名为 Intersection 的结构,它包含一个 f32 值和对实现 Object 特征的结构的引用。
代码:
use std::sync::atomic::{AtomicUsize, Ordering};
use super::ray::Ray;
use std::ops::{Index};
static mut ID : AtomicUsize = AtomicUsize::new(0);
pub trait Object {
fn intersection<'a, T: Object>(&self, ray: &Ray) -> Intersections<'a, T>;
fn get_uid() -> usize {
unsafe {
ID.fetch_add(1, Ordering::SeqCst);
ID.load(Ordering::SeqCst)
}
}
}
pub struct Intersection<'a, T: Object>{
pub t: f32,
pub obj: &'a T,
}
impl<'a, T: Object> Intersection<'a, T> {
pub fn new(t: f32, obj: &'a Object) -> Intersection<'a, T> {
Self {t, obj}
}
}
pub struct Intersections<'a, T: Object> {
pub hits: Vec<Intersection<'a, T>>,
}
impl<'a, T: Object> Intersections<'a, T> {
pub fn new() -> Self {
Self {
hits: Vec::new(),
}
}
pub fn push(&self, hit: Intersection<'a, T>) {
self.hits.push(hit);
}
pub fn len(&self) -> usize {
self.hits.len()
}
}
报错信息如下:
error[E0038]: the trait `object::Object` cannot be made into an object
--> src/object.rs:23:5
|
23 | pub fn new(t: f32, obj: &'a Object) -> Intersection<'a, T> {
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ the trait `object::Object` cannot be made into an object
|
= note: method `intersection` has generic type parameters
= note: method `get_uid` has no receiver
因为我在 Intersection 中存储了一个引用,所以我认为它不必处理结构的实际大小。
我很确定 Intersection
不应该是通用的,而应该包含 &Object
:
pub struct Intersection<'a>{
pub t: f32,
pub obj: &'a Object,
}
如果你真的需要 Intersection
中的实际对象类型,那么 Object::intersection
不应该是通用的,而应该 return 一个 Intersection<Self>
:
pub trait Object<'a> {
fn intersection(&self, ray: &Ray) -> Intersections<'a, Self>;
}
错误的第二部分涉及 get_uid
。如果你想通过引用访问特征,它不能成为特征的一部分,因为在这种情况下只能使用带有 self
参数的函数。
另请注意,get_uid
并不像您想象的那样:如果两个线程同时调用它,则两者有可能得到相同的结果。你想要的是:
fn get_object_uid() -> usize { // <- Renamed because it needs to be outside the trait
unsafe {
ID.fetch_add (1, Ordering::SeqCst) + 1
}
}