问题描述
Rock rock;
rock.setWeight(3);
将 VisualStudio 更新到 16.10 版后 - 将 C++ 编译器更新到 14.29 版 - 代码产生错误:
错误 C2664:'void std::vector
消息:gc 堆中的对象(托管数组的元素)无法转换为本机引用
消息:见
'std::vector
将循环体中的代码改为
std::vector<unsigned char> marshal_as(cli::array<System::Byte>^ const& from)
{
std::vector<unsigned char> result;
result.reserve(from->Length);
for (int i = 0; i < from->Length; i++)
{
result.push_back(from[i]);
}
return result;
}
解决问题。
我想了解此错误的原因。这是否与 C++ 20 标准的更改有关?
解决方法
这是否与 C++ 20 标准的更改有关?
没有。虽然 std::vector<>::push()
在 C++20 中发生了微妙的变化,但这并没有对这里发生的事情产生实质性影响,这个问题绝对是特定于 clr 的。
我想了解此错误的原因。
这几乎可以肯定(见下文)是一个始终存在于您的代码中的错误,但以前版本的 C++/CLI 编译器并未报告此错误。
考虑以下函数:
void foo(const int& v) {
int* ptr = &v;
// store ptr somewhere,long-term.
}
很明显,通过引用 gc 支持的 foo()
来调用 int
将导致灾难。然而,这正是 result.push_back(from[i]);
所做的。
您的代码“有效”,因为 push_back()
碰巧没有对其导致问题的参数执行任何操作。但是,编译器不应该知道。
注意我这么说几乎是肯定的,因为我有很多时间来追踪 cli::array<T>::operator[](std::size_t) const
的调用签名。以前返回 T
而现在返回 const T%
并非不可能。