- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
最近,我一直在使用很多不同的boost智能指针以及普通指针。我注意到,随着您的发展,您倾向于意识到必须切换指针类型和内存管理机制,因为您忽略了某些循环依赖性或一些烦人的事情。当发生这种情况并且您更改了指针类型时,您必须要么更改一大堆方法签名以采用新的指针类型,要么必须在每个调用站点处都必须在指针类型之间进行转换。如果您具有相同的功能,但希望它采用多种指针类型,那么您也会遇到问题。
我想知道是否已经存在一种通用的处理方式,即编写与传递给它的指针类型无关的方法?
显然,我可以看到几种方法,一种是为每种指针类型编写重载方法,但这很快就成了麻烦。另一种方法是使用带有某种类型推断的模板样式解决方案,但这将在编译后的代码中引起严重的膨胀,并可能开始引发奇怪的无法解决的模板错误。
我的想法是用所有主要指针类型的转换构造函数编写一个新类any_ptr<T>
,例如T*
,shared_ptr<T>
,auto_ptr<T>
,scoped_ptr<T>
甚至是weak_ptr<T>
,然后使其公开*
和->
运算符。这样,它可以在不返回该函数外部指针的任何函数中使用,并且可以使用常见指针类型的任意组合进行调用。
我的问题是这是否真的很愚蠢?我看到它可能会被滥用,但是假设它从未被用于返回any_ptr的函数中,我会遇到一个主要问题吗?请你的想法。
编辑1
阅读您的答案后,我想做一些笔记,以至于评论太长。
首先是关于使用原始指针或引用(@shoosh)的。我同意您可以使函数使用原始指针,但是假设我正在使用shared_ptr,这意味着我不得不在每个调用站点上使用ptr.get(),现在假设我意识到我做了一个循环引用,并且我必须将指针更改为weak_ptr,然后必须将所有这些调用站点都更改为x.lock()。get()。现在,我同意这不是灾难,而是令人烦恼的,我觉得有一个解决方案。对于传递as T&引用并转到* x,可以说是相同的,必须进行类似的 call 站点更改。
我在这里试图做的是,即使指针类型发生了很大的变化,也使代码更易于阅读且更易于重构。
关于smart_ptr语义,其次是:我同意出于不同的原因使用不同的智能指针,并且在复制和存储方面要考虑某些注意事项(这就是为什么boost::shared_ptr<T>
不能自动转换为T*
的原因) 。
但是我设想any_ptr(回想起来可能是一个坏名字)仅在指针将不被存储的情况下使用(除了栈上的临时变量外)。它应该可以从各种智能指针类型隐式构造,重载*和->运算符,并可以转换为T *(通过自定义转换函数T*()
)。这样,any_ptr
的语义与T*
完全相同。因此,它仅应在安全使用原始ptr的地方使用(这是@ Alexandre_C在评论中说的)。这也意味着@Matthieu_M所讨论的“重型机械”将不存在。
第三,关于模板,。虽然模板对于某些方面很有用,但由于上面的原因,我还是对它们保持警惕。
最终的:所以基本上我想做的是使用通常将原始ptr(T *)作为参数的函数,我想创建一个系统,这些参数可以自动接受任何各种smart_ptr类型而无需在 call 站点进行转换。我之所以这样做,是因为我认为它将通过消除转换残差(因此也稍短一些,尽管幅度不大)来使代码更具可读性,并且这将使重构和尝试不同的智能指针机制变得不那么麻烦。
也许我应该把它叫做unmanaged_ptr
而不是any_ptr
。那会更正确地描述语义。我为这个笨拙的名字表示歉意。
编辑2
好的,这是我要考虑的类(class)。我称它为dumb_ptr。
template<typename T>
class dumb_ptr {
public:
dumb_ptr(const dumb_ptr<T> & dm_ptr) : raw_ptr(dm_ptr.raw_ptr) { }
dumb_ptr(T* raw_ptr) : raw_ptr(raw_ptr) { }
dumb_ptr(const boost::shared_ptr<T> & sh_ptr) : raw_ptr(sh_ptr.get()) { }
dumb_ptr(const boost::weak_ptr<T> & wk_ptr) : raw_ptr(wk_ptr.lock().get()) { }
dumb_ptr(const boost::scoped_ptr<T> & sc_ptr) : raw_ptr(sc_ptr.get()) { }
dumb_ptr(const std::auto_ptr<T> & au_ptr) : raw_ptr(au_ptr.get()) { }
T& operator*() { return *raw_ptr; }
T * operator->() { return raw_ptr; }
operator T*() { return raw_ptr; }
private:
dumb_ptr() { }
dumb_ptr<T> operator=(const dumb_ptr<T> & x) { }
T* raw_ptr;
};
它可以自动从常见的智能指针转换,并且可以视为原始T *指针,此外,它可以自动转换为T *。隐藏了默认构造函数和赋值运算符(=),以阻止人们将其用于除函数参数以外的任何其他功能。当用作函数参数时,可以完成以下操作。
void some_fn(dumb_ptr<A> ptr) {
B = ptr->b;
A a = *ptr;
A* raw = ptr;
ptr==raw;
ptr+1;
}
您几乎要对指针执行所有操作。它具有与原始指针
T*
完全相同的语义。但是现在它可以与任何智能指针一起用作参数,而不必在每个调用站点重复转换代码(.get,.lock)。同样,如果您更改了智能指针,则无需四处修正每个 call 站点。
最佳答案
使用此类any_ptr
,除了*
和->
之外,您将几乎无法执行其他任何操作。没有分配,复制构造,重复或破坏。如果仅此而已,则只需使用原始指针T*
作为参数编写函数,然后使用.get()
或自动指针上的whatnot调用它。
关于c++ - 新的通用指针any_ptr(现为dumb_ptr)使代码在智能指针之间更可重用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5325471/
是否可以有DropShadowEffect渲染阴影时忽略某些颜色?有某种蒙版(颜色选择性)阴影? 我的问题是什么阴影可以分配给整个视觉元素(图形)。它看起来像这样: 而且我要 注意没有阴影的网格线(除
It's difficult to tell what is being asked here. This question is ambiguous, vague, incomplete, over
有时,当我尝试使用 maven mvn clean install jetty:run 运行我的项目时,它会返回一个错误消息:“地址已在使用中”。 我正在使用 Intellij IDEA 14。 你能
我见过几个数据库缓存引擎,它们都非常愚蠢(即:将此查询缓存 X 分钟)并要求您在 之后手动删除整个缓存存储库>INSERT/UPDATE/DELETE 查询已执行。 大约 2 或 3 年前,我为我正在
我正在寻找例程或寻找错误容忍字符串比较的方法。 比方说,我们有测试字符串 Čakánka - 是的,它包含 CE 字符。 现在,我想接受以下任何字符串作为 OK: 恰坎卡 cákanká ÇaKaNK
大家好 我在 windows xp 系统上使用智能 gwt 2.3 专业版和 Mozilla。 我在上面做了测试应用程序,它运行成功。 现在我想为此应用程序使用开发人员控制台。 我的gwt.xml 编
说,我有 member this.Test (x: 'a) = printfn "generic" 1 member this.Test (x:
我正在努力寻找有关如何将应用程序部署到索尼智能 watch 的任何文档或教程(我正在尝试部署由索尼编写的开源“Eight Puzzle”应用程序,我已将其导入到 Eclipse 中。我有一个三星 S3
我正在寻找一个类似于标准格式化程序的智能 Java 字符串格式化程序: StringBuffer buffer = new StringBuffer(); Formatter form
我有一个智能 JScrollPane,但与其他所有解决方案不同,我想知道如果滚动 Pane 不在底部并且文本是否可以跟随(向上滚动)查看的组件(JTextArea)缓冲区已满。包含功能齐全的代码片段。
我有一个包含 18 个类(class)的项目。它们正在工作线程保存(已检查),没有递归调用,也没有任何复杂的计算。 但是我的IDEA总是说几个小时后 There is not enough memor
我想使用 Smart XLS 库在 C# 中处理 excel 文件(特别是因为它可以生成图表,而且它的价格明显低于 Aspose Cells),但我找不到任何好的例子。我只想知道如何创建和修改图表,有
我正在寻找一个完整的 java 日期管理库,它可以让我理解像这样的字符串: 明天中午 => 2011-10-20 12:00 今天下午 4 点 => 2011-10-20 16:00 等... 但如果
我想打开手机中的文件,通过 watch 中的 Action ... 问题是: Intent intent2 = new Intent(); intent2.setAction(android.cont
我想编译 Sony 为他的 SmartWatch 提供的示例。我按照此 page 中的说明进行操作.但是当我添加 SmartExtensionUtils 项目时,我看到了很多错误,例如:com.son
我有一个 SQL 表:名称、位置、体积 名称为字符串类型 位置是两个 float 类型的字段(纬度和经度) int 类型的体积 我想运行一个 SQL 查询,该查询将对特定范围内的所有位置进行分组并对所
我需要找到当我点击某物时调用的方法。项目很大,自己找会花不少时间。有什么办法可以做到吗?我正在使用 Android Studio,是的,这是我有史以来第一次使用其他人的项目。 最佳答案 我是怎么做到的
我在我的 iMac 上安装了智能 cvs,我已经从 cvs 更新了应用程序。然后我对我的代码和核心数据做了一些更改(核心数据更改是:从核心数据实体中删除一些属性),然后我尝试将更改提交给 cvs,但我
在我的 Java 项目中,我有一个类 Person,它存在于两个具有相同包名 (com.example.beans) 的不同 jar 中。 问题是这个类在jar1中是这样定义的: class Pers
已关闭。此问题旨在寻求有关书籍、工具、软件库等的建议。不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以
我是一名优秀的程序员,十分优秀!