- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我接管了一些代码,并遇到了一个奇怪的数组重新分配。这是 Array 类中的函数(由 JsonValue 使用)
void reserve( uint32_t newCapacity ) {
if ( newCapacity > length + additionalCapacity ) {
newCapacity = std::min( newCapacity, length + std::numeric_limits<decltype( additionalCapacity )>::max() );
JsonValue *newPtr = new JsonValue[newCapacity];
if ( length > 0 ) {
memcpy( newPtr, values, length * sizeof( JsonValue ) );
memset( values, 0, length * sizeof( JsonValue ) );
}
delete[] values;
values = newPtr;
additionalCapacity = uint16_t( newCapacity - length );
}
}
我明白了这一点;它只是分配一个新数组,并将旧数组中的内存内容复制到新数组中,然后将旧数组的内容清零。我也知道这样做是为了防止调用析构函数和移动。
JsonValue
是一个具有函数的类,以及一些存储在 union 中的数据(字符串、数组、数字等)。
我关心的是这是否是真正定义的行为。我知道它有效,而且自从我们几个月前开始使用它以来就没有出现过问题;但如果它未定义,则并不意味着它会继续工作。
编辑:JsonValue
看起来像这样:
struct JsonValue {
// …
~JsonValue() {
switch ( details.type ) {
case Type::Array:
case Type::Object:
array.destroy();
break;
case Type::String:
delete[] string.buffer;
break;
default: break;
}
}
private:
struct Details {
Key key = Key::Unknown;
Type type = Type::Null; // (0)
};
union {
Array array;
String string;
EmbedString embedString;
Number number;
Details details;
};
};
其中 Array
是 JsonValue
数组的包装器,String
是 char*
,EmbedString
是 char[14]
,Number
是 int
的并集,unsigned int
,和 double
,Details
包含它所保存的值的类型。所有值在开头都有 16 位未使用的数据,用于详细信息
。示例:
struct EmbedString {
uint16_t : 16;
char buffer[14] = { 0 };
};
最佳答案
此代码是否具有明确定义的行为基本上取决于两件事:1) 是 JsonValue
trivially-copyable 2) 如果是这样,一堆全零字节是 JsonValue
的有效对象表示。
如果 JsonValue
是可简单复制的,那么从一个 JsonValue
数组到另一个数组的 memcpy
确实相当于复制所有元素超过 [basic.types]/3 。如果全零是 JsonValue 的有效对象表示,那么 memset 应该没问题(我相信这实际上属于当前的灰色区域)标准的措辞,但我相信至少其意图是这样就可以了)。
我不确定为什么您需要“防止调用析构函数和移动”,但用零覆盖对象并不会阻止析构函数运行。 delete[] value
将调用数组成员的析构函数。无论如何,移动可简单复制类型的数组的元素应该编译为仅复制字节。
此外,我建议摆脱这些 String
和 EmbedString
类,而只使用 std::string
。至少,在我看来,EmbedString
的唯一目的是手动执行小字符串优化。任何有值(value)的 std::string
实现都已经在幕后做到了这一点。请注意,std::string
不保证(并且通常不会)可简单复制。因此,您不能简单地用 std::string
替换 String
和 EmbedString
,同时保留当前实现的其余部分。
如果你可以使用C++17,我建议简单地使用 std::variant
而不是或至少在这个自定义 JsonValue
实现中,因为这似乎正是它想要做的。如果您需要在变体值之前存储一些通用信息,只需在保存变体值的成员前面放置一个合适的成员来保存该信息,而不是依赖于以相同的一对开始的 union 的每个成员成员(只有当所有 union 成员都是标准布局类型并将此信息保留在其公共(public)初始序列 [class.mem]/23 中时,这才是明确定义的)。
Array
的唯一目的似乎是作为一个 vector ,在出于安全原因释放内存之前将其清零。如果是这种情况,我建议只使用带有分配器的 std::vector
,该分配器在取消分配之前将内存归零。例如:
template <typename T>
struct ZeroingAllocator
{
using value_type = T;
T* allocate(std::size_t N)
{
return reinterpret_cast<T*>(new unsigned char[N * sizeof(T)]);
}
void deallocate(T* buffer, std::size_t N) noexcept
{
auto ptr = reinterpret_cast<volatile unsigned char*>(buffer);
std::fill(ptr, ptr + N, 0);
delete[] reinterpret_cast<unsigned char*>(buffer);
}
};
template <typename A, typename B>
bool operator ==(const ZeroingAllocator<A>&, const ZeroingAllocator<B>&) noexcept { return true; }
template <typename A, typename B>
bool operator !=(const ZeroingAllocator<A>&, const ZeroingAllocator<B>&) noexcept { return false; }
然后
using Array = std::vector<JsonValue, ZeroingAllocator<JsonValue>>;
注意:我通过 volatile unsigned char*
填充内存,以防止编译器优化归零。如果需要支持过度对齐类型,可以将 new[]
和 delete[]
替换为直接调用 ::operator new
和 ::operator delete
(这样做将阻止编译器优化分配)。在 C++17 之前,您必须分配足够大的缓冲区,然后手动对齐指针,例如使用 std::align
...
关于c++ - 使用 memcpy 和 memset 重新分配数组,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59155283/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!