- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
为了更好地理解 C++ 类型系统,我努力编写了一个指针包装类,它传播类似于 std::experimental::propagate_const 的常量:
template <typename Pointee> class Ptr {
public:
Ptr() = delete;
explicit Ptr(Pointee *);
Ptr(const Ptr<Pointee> &) = delete;
Ptr(Ptr<Pointee> &&);
Ptr<Pointee> &operator=(const Ptr<Pointee> &) = delete;
Ptr<Pointee> &operator=(Ptr<Pointee> &&);
~Ptr() = default;
const Pointee *operator->() const;
Pointee *operator->();
const Pointee &operator*() const;
Pointee &operator*();
private:
Pointee *mPtr;
};
包装器旨在提供接近原始指针的行为,同时还强制执行一种“深度”const 正确性并防止无意的别名。
为此删除了拷贝构造函数和拷贝赋值运算符:
但是,上述设计有两个不幸的后果。
const Ptr<int> allocateImmutableInt(int val) { return Ptr<int>(new int(val)); }
void foo() {
Ptr<int> immutableInt = allocateImmutableInt(0); // Initializes non-const Ptr from const Ptr
*immutableInt = 100; // Oops, changed value of 'immutable' object
}
第一个问题可以通过引入一个接受 const 右值引用的 move ctor 来部分解决(尽管这感觉有些奇怪和不符合习惯):
Ptr(const Ptr<Pointee> &&);
然而,这实际上加剧了第二个问题。现在,即使没有强制移动/复制省略,也可以将 const Ptr 移动构造为非常量 Ptr。据我所知,为了解决这个问题,我们需要一个所谓的“const 构造函数”,即只能调用以生成 const 对象的构造函数:
Ptr(const Ptr<Pointee>&&) const;
即使 c++ 支持这样的构造函数,第二个问题仍然存在,因为 c++17 在决定是否可以在初始化对象时应用强制移动/复制省略时特别忽略了 cv 限定和构造函数的可行性。目前似乎没有办法要求 C++ 在将强制复制/移动省略应用于对象初始化之前检查复制/移动是否可行。
据我所知,std::experimental::propagate_const 也存在同样的问题。我想知道我是否遇到了 C++ 的基本限制,或者我是否错误地设计了 Ptr 包装器?我知道这些问题可能可以通过创建两种类型来消除,一种是用于非 const 访问的 Ptr,另一种是用于仅 const 访问的 ConstPtr。然而,这首先违背了创建常量传播包装器的目的。
也许我刚刚偶然发现了迭代器类型和 const_iterator 类型同时存在的原因。
最佳答案
您正在查看实际上并不存在的问题。
- To prevent non-const access of a constpointed-to object by copying a const Ptr into a non-const Ptr.
这不应该是一个目标,因为它违背了传播 const
的想法。 .
传播有两个方面const
.首先,当指针为const
时-合格,对象也是const
-合格的。这方面你已经涵盖了。第二,当指针不是 const
-qualified,对象使用其自然资格。也就是说,如果您可以复制 const Ptr
变成非 const
Ptr
,然后该更改传播到对象,可能使对象也非 const
.这是期望的传播,而不是要阻止的东西。
请记住 const
的一个主要用例传播:类成员。传播 const
对于成员指针,通过指向数据 const
来帮助确保常量正确性在 const
- 合格的成员函数。您想象的问题不适用于此用例。不要让情况变得比它需要的更复杂。
I am aware that these issues can likely be eliminated by creating two types, a Ptr for non-const access and ConstPtr for const-only access.
这不是必须的。如果对象应该保留 const
即使指针不是 const
, 那么类型应该是 Ptr<const T>
而不是 Ptr<T>
.例如,您的“不可变整数”应该看起来更像以下内容。
Ptr<const int> allocateImmutableInt(int val) { return Ptr<const int>(new int(val)); }
^^^^^ ^^^^^
const
已移动以符合 int
的资格, 使得 int
无论const
如何都是不可变的-资格 Ptr
.
此外,您可能会注意到 new int(val)
返回 int*
隐式转换为 const int*
为您的构造函数。您可能希望为 Ptr
复制此隐式转换.像 Ptr(Ptr<std::remove_const<Pointee>> &&)
这样的构造函数只要它仅在 Pointee
时定义,就可以解决问题是const
-限定(以避免与常规移动构造函数发生冲突)。
关于c++ - 如何编写 const 传播指针类型包装器?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/64196584/
正在尝试创建一个 python 包。似乎有效,但我收到警告。我的 setup.py 是: #! /usr/bin/env python from distutils.core import setup
我导入了一个数据类型 X ,定义为 data X a = X a 在本地,我定义了一个通用量化的数据类型,Y type Y = forall a. X a 现在我需要定义两个函数, toY 和 fro
我似乎无法让编译器让我包装 Tokio AsyncRead: use std::io::Result; use core::pin::Pin; use core::task::{Context, Po
我有两个函数“a”和“b”。当用户上传文件时,“b”被调用。 “b”重命名文件并返回新文件名。之后应该编辑该文件。像这样: def a(): edits file def b(): r
我使用 Entity Framework 作为我的 ORM,我的每个类都实现了一个接口(interface),该接口(interface)基本上表示表结构(每个字段一个只读属性)。这些接口(inter
有没有办法打开一个程序,通常会打开一个新的jframe,进入一个现有的jframe? 这里是解释,我下载了一个java游戏,其中一个是反射游戏,它在一个jframe中打开,框架内有一堆子面板,我想要做
我想要下面的布局 | AA BBBBBBB | 除非没有足够的空间,在这种情况下 | AA | | BBBBBBB | 在这种情况下,A 是复选框,B 是复选框旁边的 Text
我正在尝试以不同的方式包装我的网站,以便将背景分为 2 部分。灰色部分是主要背景,还有白色部分,它较小并包装主要内容。 基本上我想要this看起来像this . 我不太确定如何添加图像来创建阴影效果,
我正在使用 : 读取整数文件 int len = (int)(new File(file).length()); FileInputStream fis = new FileInputStream(f
我使用 maven 和 OpenJDK 1.8 打包了一个 JavaFX 应用程序我的 pom.xml 中的相关部分: maven-assembly-plugin
我正在使用两个不同的 ItemsControl 来生成一个按钮列表。
我有一个情况,有一个变量会很方便,to , 可以是 TimerOutput或 nothing .我有兴趣提供一个采用与 @timeit 相同参数的宏来自 TimerOutputs(例如 @timeit
我正在尝试包装一个名为 content 的 div与另一个具有不同背景的 div。 但是,当将“margin-top”与 content 一起使用时div,似乎包装 DIV 获得了边距顶部而不是 co
文档不清楚,它似乎允许包装 dll 和 csproj 以在 Asp.Net Core 5 应用程序中使用。它是否允许您在 .Net Core 5 网站中使用针对 .Net Framework 4.6
我被要求开发一个层,该层将充当通用总线,而不直接引用 NServiceBus。到目前为止,由于支持不引人注目的消息,这并不太难。除了现在,我被要求为 IHandleMessages 提供我们自己的定义
我正在尝试包装 getServersideProps使用身份验证处理程序函数,但不断收到此错误:TypeError: getServerSideProps is not a function我的包装看
我有一个项目,它在特定位置(不是/src/resources)包含资源(模板文件)。我希望在运行 package-bin 时将这些资源打包。 我看到了 package-options 和 packag
我正在寻找打印从一系列对象中绘制的 div。我可以通过使用下面的管道语法来实现这一点。 each i, key in faq if (key == 0) |
我在 Meteor.js“main.js - Server”中有这个方法。 Meteor.methods({ messageSent: function (message) { var a
我注意到,如果我的自定义Polymer 1.x元素的宽度比纸张输入元素上的验证错误消息的宽度窄,那么错误将超出自定义元素的右边界。参见下图: 有没有一种机制可以防止溢出,例如在到达自定义元素的边界时自
我是一名优秀的程序员,十分优秀!