- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我有以下代码用于存储一个小类。
#include <iostream>
template<typename T>
class storage
{
private:
struct destroy
{
T& m_t;
destroy(T& t) : m_t(t) { }
~destroy() { m_t.~T(); }
};
char m_c[sizeof(T)];
void* address() { return &m_c[0]; }
public:
void set(const T& t) { new (address()) T(t); }
T get()
{
T& t = *static_cast<T*>(address());
destroy _d(t);
return t;
}
};
template<typename T>
class choosable_storage
{
private:
union
{
T* m_p;
storage<T> m_storage;
};
bool m_direct;
public:
choosable_storage() : m_direct(false) { }
void set_direct(const T& t)
{
m_direct = true;
m_storage.set(t);
}
void set_indirect(T* const t) { m_p = t; }
T get()
{
if (m_direct) return m_storage.get();
return *m_p;
}
};
int main(void)
{
storage<int> s; // no problems
s.set(42);
std::cout << s.get() << std::endl;
int i = 10;
choosable_storage<int> c1; // strict aliasing warnings
c1.set_indirect(&i);
std::cout << c1.get() << std::endl;
choosable_storage<int> c2;
c2.set_direct(i);
std::cout << c2.get() << std::endl;
return 0;
}
gcc 4.4 警告我在返回时违反了 storage::get()
中严格的别名规则。
据我所知,我没有违反任何规则。我真的违反了严格的别名还是 gcc 在这里变得挑剔?
有没有办法在不禁用严格别名的情况下让它不发出警告?
谢谢
编辑:
另一方面,下面的实现不会给出任何警告:
template<typename T>
class storage
{
private:
struct destroy
{
T& m_t;
destroy(T& t) : m_t(t) { }
~destroy() { m_t.~T(); }
T const& operator()() const { return m_t; }
};
char m_c[sizeof(T)];
public:
void set(const T& t) { new(static_cast<void*>(m_c)) T(t); }
T get(void) { return destroy(*static_cast<T*>(static_cast<void*>(m_c)))(); }
};
编辑:
gcc 4.5 及更高版本不会发出警告 - 所以显然这只是对严格别名规则的误解或 gcc 4.4.x 中的错误
最佳答案
Do I actually violate strict aliasing or is gcc getting picky here?
严格别名规则有两种不同的解释:
char
/unsigned char
类型除外)您不能使用强制转换或 union 来执行类型双关,你需要memcpy
(或两次volatile
访问);这样的代码确实不太合理,并且在 C(除了最新的非常荒谬的 C 标准)和 C++ 中被明确禁止。malloc
替代)C,除非它每次只调用 malloc
/free
。强规则定义不明确,破坏了许多合理的代码,并且出于某种原因被 GCC 维护者选择(完全基于 self 欺骗和循环论证 - 这真的很丑陋)。为了使 C++ 代码与强大的严格别名优化一起工作,G++ 的维护者为典型的 C++ 代码添加了悲观(基于更多的 self 欺骗),以保持优化!
我不知道他们是否/何时意识到自己的错误,如有疑问,请禁用严格别名。无论如何,这是一个非常小的优化。
关于c++ - C++ 中的简单存储类和严格的别名,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4170146/
上下文:我已经在域根上设置了一个用于 GIS 测试和 dokuwiki 的虚拟机服务器。我想在子域上提供 gis Web 应用程序,这样 dokuwiki url 重命名就不会发生冲突(而且感觉更干净
这个问题在这里已经有了答案: If two variables point to the same object, why doesn't reassigning one variable affe
代码如下: Select Branch.BranchName, Sum(Inventory.OnHand) From Inventory, Branch Inventory.BranchNum = B
我的目标是创建一种“Javascript 库”,如果您可以这样调用它的话。我打算在浏览网页时使用它来操作页面,将其作为 greasemonkey 脚本动态加载。这个想法是将“win”映射到window
是否可以在 javascript 中以某种方式为本地变量分配别名/引用? 我的意思是类似 C 的东西: function foo() { var x = 1; var y = &x; y+
SQLite 别名 您可以暂时把表或列重命名为另一个名字,这被称为别名。使用表别名是指在一个特定的 SQLite 语句中重命名表。重命名是临时的改变,在数据库中实际的表的名称不会改变。 列别名用来
我有以下导入(在同一个 TS 源文件中): import {Vector as sourceVector} from "ol/source"; import {Vector} from "ol/lay
我想弄清楚 this 到底是什么行是在这个 github json 项目的 cmake 文件中, add_library(${NLOHMANN_JSON_TARGET_NAME} INTERFACE)
使用集合时如何创建别名 (AS)。 $collection->addAttributeToSelect('total_qty') 可以这样做: $collection->getSelect()->co
这个问题在这里已经有了答案: Pass an argument to a Git alias command (4 个回答) 去年关闭。 我的团队在不同的分支上工作,通常,我需要快速从分支切换来帮助他
我正在寻找一种向我的 vimrc 添加自定义命令或别名的方法,以便快速切换到 Vim 中的不同目录。我曾经设置过一些东西,但丢失了那个配置。这是用于 Linux 设置。 基本上,我想设置一个别名(如果
我使用 Delphi 和 Firebird 1.5 开发了一个应用程序,其中服务器与应用程序位于同一台计算机上。我现在正在将应用程序部署到另一个站点,其中 Firebird 服务器( super 服务
我创建了一个全新的 React 应用 create-react-app demo 我需要为某些目录/组件创建别名,例如: import { Header } from '@uicomponents'
我正在尝试打印有向图,并且不断更改各种节点属性,例如颜色和形状。有什么方法可以使用一次定义并用于多个节点的变量?理想情况下,我想要这样的东西: digraph g { building_color =
server { listen 80; server_name pwta; root html; location /test/{ alia
我想匹配多个数组,并在匹配时构建另一个数组。该键可以在任意数量的数组中匹配,也可以根本不匹配。 [ [{ 'a': 13 }, { 'b': 62 }, { 'c': 93 }, { 'd': 52
经过审查后,我需要更改一些代码并将所有更改代码重新推送到远程分支上。 所以我想使用别名(例如:git repushall)在 git 上自动化这些不同的命令: git add . git commit
我有一个这样的查询: select a1.name, b1.info from (select name, id, status from table1 a) as a1 righ
我想给一些 racket 2htdp 函数/宏起别名,这样我就可以为我的 child 将它们翻译成另一种语言。 我可以简单地用define给函数起别名。我在使用 big-bang 结构时遇到了麻烦;例
经过审查后,我需要更改一些代码并将所有更改代码重新推送到远程分支上。 所以我想使用别名(例如:git repushall)在 git 上自动化这些不同的命令: git add . git commit
我是一名优秀的程序员,十分优秀!