2

操作系统:xp
IDE:VS 2008
在我用 Visual C++ 做的项目中,我已经声明了一个std::vector内部托管类

std::vector<pts> dataPoints;//this gives error c4368 : mixed type not allowed 

但这有效

std::vector<pts> * dataPoints;//a pointer to the vector  

然后我在免费商店中创建了这个向量,就像在托管类的构造函数中一样

dataPoints = new std::vector<pts>(noOfElements,pts());//which is not so attractive.

我需要向量的原因是因为我正在读取文件ifstream并将这些值存储在向量中。
Q1)为什么我能够声明一个指向本机类型对象的指针(我猜)但不是一个对象?此外,在尝试向量之前,我尝试了托管数组

cli::array<Point> dataPoints //and i defined it later.

但是当我这样做时

ifile >> dataPoints[i].X;   

它给出了一个错误 c2678:operator= 没有为int!! 重载。
Q2)为什么我不能在这里使用托管代码。起初我认为它可能是一个包装类 Int 但随后自动拆箱(转换运算符)应该处理它?还是 Point::X 是合格的property,因此不被认为是正常的int?我错过了什么?这就是我寻求解决方案的vector原因pts
pts如下

 struct pts
{
  int X, int Y;
  pts() : X(0),Y(0){}
  pts(int x,int y) : X(x),Y(y){}
};//this i created to store the data from the file.
4

2 回答 2

5

托管类对象的一个​​重要属性是它们被垃圾收集器移动。当它压缩堆时会发生这种情况。这对本机 C++ 对象造成严重破坏,指向其成员的指针将变得无效。因此,作为一项规则,编译器禁止将本机非 POD 对象嵌入托管对象中。指针不是问题。

您对运算符的使用存在完全相同的问题>>。int 通过引用 operator>>() 来传递。如果垃圾收集器恰好在获取 int 引用的代码和调用操作符之间介入,那么灾难就会发生。一个简单的解决方法是通过局部变量的中间步骤:

int x;
ifile >> x;
dataPoint[i].X = x;

之所以有效,是因为局部变量是稳定的并且不受垃圾收集的影响。

这在本机代码中都不是问题。请记住,您的 ref 类可以轻松调用本机函数。因此,将两者分开可能是有用的和/或必要的。

于 2013-04-19T19:20:50.407 回答
1

您不能在托管类型中直接包含本机类型:这只是对 C++/CLI 的限制。我认为这可能与本机类型中指针的可能性有关。如果本机类型直接在托管类型中,那么当托管对象在垃圾收集期间被打乱时,这些指针将指向原始的,现在不正确的内存。

因此,本机对象需要在堆上,这样它的内部结构就不会被垃圾回收所改变。因此,您需要将向量作为指针保存,并适当地删除它。请注意,后者并非完全无关紧要,您需要对 C++/CLI(与 C# 略有不同)有所了解。请参阅http://msdn.microsoft.com/en-us/library/ms177197(v=vs.100).aspx

看看我最后一次这样做,在我的文件中,我有

public:
    !NetClass();
    ~NetClass() { this->!NetClass(); } // avoid arning C4461
private:
    class NativeImpl* const m_pImpl; // can't contain NativeImpldirectly

在我的cpp文件中

NetClass::!NetClass()
{
    // implement finalizer in ref class
    delete m_pImpl;
}

如果您要包含多个本机类,您可能只想在此处使用 pimpl 习惯用法。请参阅为什么要使用“PIMPL”成语?.

最后,我上次这样做是很久以前的事了,我只是说当时对我有用的东西。如果你这样做,你真的需要知道你在做什么。我使用了一本名为C++/CLI in Action的书,我会推荐这本书。

编辑

这篇关于 STL/CLR 的文章看起来很有趣:http: //blogs.msdn.com/b/nikolad/archive/2006/06/16/stlclr-intro.aspx。去引用

STL/CLR,最初称为 STL.NET,是标准模板库 (STL) 的实现,可以对托管类型的对象进行操作。VC++ 已经实现了 STL,但是它目前仅适用于本机类型。

(我真的不能帮助你的第二季度)

于 2013-04-19T18:47:31.567 回答