反复出现的成本难题

2024-03-14

我经常发现自己必须定义一个函数的两个版本,以便拥有一个 const 版本和一个非常量版本(通常是 getter,但并非总是如此)。两者的区别仅在于,其中一个的输入和输出是常量,而另一个的输入和输出是非常量。该功能的核心——真正的工作,是相同的。

然而,为了常量正确性,我需要它们两者。作为一个简单的实际示例,请看以下内容:

inline const ITEMIDLIST * GetNextItem(const ITEMIDLIST * pidl)
{
    return pidl ? reinterpret_cast<const ITEMIDLIST *>(reinterpret_cast<const BYTE *>(pidl) + pidl->mkid.cb) : NULL;
}

inline ITEMIDLIST * GetNextItem(ITEMIDLIST * pidl)
{
    return pidl ? reinterpret_cast<ITEMIDLIST *>(reinterpret_cast<BYTE *>(pidl) + pidl->mkid.cb) : NULL;
}

正如你所看到的,他们做同样的事情。我可以选择使用更多的强制转换来根据另一个来定义一个,如果胆量 - 实际工作,不那么微不足道,那么这是更合适的:

inline const ITEMIDLIST * GetNextItem(const ITEMIDLIST * pidl)
{
    return pidl ? reinterpret_cast<const ITEMIDLIST *>(reinterpret_cast<const BYTE *>(pidl) + pidl->mkid.cb) : NULL;
}

inline ITEMIDLIST * GetNextItem(ITEMIDLIST * pidl)
{
    return const_cast<ITEMIDLIST *>(GetNextItem(const_cast<const ITEMIDLIST *>(pidl));
}

所以,我觉得这非常乏味和多余。但是,如果我希望编写 const 正确的代码,那么我要么必须提供上述两者,要么必须用 const 强制转换来乱扔我的“消费者代码”,以解决仅定义一个或另一个的问题。

有更好的模式吗?您认为解决此问题的“最佳”方法是什么:

  • 提供给定函数的两个副本 - const 和非 const 版本
  • 或者只是一个版本,然后要求该代码的使用者按照他们的意愿进行转换?

或者有更好的方法来解决这个问题吗? 语言本身是否正在做一些工作来减轻或完全消除这个问题?

对于奖励积分:

  • 您是否认为这是 C++ const 系统的一个不幸的副产品
  • 或者你认为这相当于触及奥林匹斯山的高度吗?

EDIT:

如果我只提供第一个 - 接受 const 返回 const,那么任何需要修改返回项或将返回项传递给另一个将修改它的函数的消费者都必须摆脱常量性。

类似地,如果我只提供第二个定义 - 接受非 const 并返回非 const,那么具有 const pidl 的使用者必须放弃 constness 才能使用上述函数,老实说,这不会修改 constness物品本身。

也许更多的抽象是可取的:

THING & Foo(THING & it);
const THING & Foo(const THING & it);

我很想有一个构造:

const_neutral THING & Foo(const_neutral THING & it);

我当然可以做这样的事情:

THING & Foo(const THING & it);

但这总是让我感到不舒服。我的意思是“我不会修改你的东西的内容,但我会在你的代码中去掉你默默委托给我的常量。”

现在,一个客户,其中有:

const THING & it = GetAConstThing();
...
ModifyAThing(Foo(it));

那是错误的。 GetAConstThing 与调用者的约定是给它一个 const 引用。调用者不应修改该事物 - 仅对其使用 const 操作。是的,调用者可能是邪恶的、错误的,并且抛弃了它的恒定性,但这只是邪恶(tm)。

对我来说,问题的关键在于 Foo 是常量中立的。它实际上并不修改其给定的内容,但其输出需要传播其参数的常量性。

注意:第二次编辑格式。


在我看来,这是 const 系统的一个不幸的副产品,但它并不经常出现:只有当函数或方法给出对某些东西的指针/引用时(无论它们是否修改某些东西,函数都不能发出)它没有的权利或常量正确性会严重破坏,因此这些重载是不可避免的)。

通常,如果这些函数只是一小行,我就会重复它们。如果实现更复杂,我使用模板来避免代码重复:

namespace
{
    //here T is intended to be either [int] or [const int]
    //basically you can also assert at compile-time 
    //whether the type is what it is supposed to be
    template <class T>
    T* do_foo(T* p)
    {
        return p; //suppose this is something more complicated than that
    }
}

int* foo(int* p)
{
    return do_foo(p);
}

const int* foo(const int* p)
{
    return do_foo(p);
}

int main()
{
    int* p = 0;
    const int* q = foo(p);  //non-const version
    foo(q);  //const version
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

反复出现的成本难题 的相关文章

  • MEX 文件中的断言导致 Matlab 崩溃

    我正在使用mxAssert 宏定义为matrix h在我的 C 代码中 mex 可以完美编译 当我调用的 mex 代码中违反断言时 该断言不会导致我的程序崩溃 而是导致 Matlab 本身崩溃 我错过了什么吗 这是有意的行为吗 当我查看 M
  • 捕获 foreach 条件中抛出的异常

    我有一个foreach在 foreach 本身的条件下循环期间中断的循环 有没有办法try catch抛出异常然后继续循环的项 这将运行几次 直到异常发生然后结束 try foreach b in bees exception is in
  • Blazor 与 Razor

    随着 Blazor 的发明 我想知道这两种语言之间是否存在显着的效率 无论是在代码创建方面还是在代码的实际编译 执行方面 https github com SteveSanderson Blazor https github com Ste
  • 获取从属性构造函数内部应用到哪个属性的成员?

    我有一个自定义属性 在自定义属性的构造函数内 我想将属性的属性值设置为属性所应用到的属性的类型 是否有某种方式可以访问该属性所应用到的成员从我的属性类内部 可以从 NET 4 5 using CallerMemberName Somethi
  • 是否有与 C++11 emplace/emplace_back 函数类似的 C# 函数?

    从 C 11 开始 可以写类似的东西 include
  • 禁用 LINQ 上下文的所有延迟加载或强制预先加载

    我有一个文档生成器 目前包含约 200 个项目的查询 但完成后可能会超过 500 个 我最近注意到一些映射表示延迟加载 这给文档生成器带来了一个问题 因为它需要根据生成的文档来访问所有这些属性 虽然我知道DataLoadOptions可以指
  • 为什么 FTPWebRequest 或 WebRequest 通常不接受 /../ 路径?

    我正在尝试从 ftp Web 服务器自动执行一些上传 下载任务 当我通过客户端甚至通过 Firefox 连接到服务器时 为了访问我的目录 我必须指定如下路径 ftp ftpserver com AB00000 incoming files
  • 两组点之间的最佳匹配

    I ve got two lists of points let s call them L1 P1 x1 y1 Pn xn yn and L2 P 1 x 1 y 1 P n x n y n 我的任务是找到它们点之间的最佳匹配 以最小化它
  • 事件日志写入错误

    很简单 我想向事件日志写入一些内容 protected override void OnStop TODO Add code here to perform any tear down necessary to stop your serv
  • 用于从字符串安全转换的辅助函数

    回到 VB6 我编写了一些函数 让我在编码时无需关心字符串的 null 和 数字的 null 和 0 等之间的区别 编码时 没有什么比添加特殊情况更能降低我的工作效率了用于处理可能导致一些不相关错误的数据的代码 9999 10000 如果我
  • 在 C 中复制两个相邻字节的最快方法是什么?

    好吧 让我们从最明显的解决方案开始 memcpy Ptr const char a b 2 调用库函数的开销相当大 编译器有时不会优化它 我不会依赖编译器优化 但即使 GCC 很聪明 如果我将程序移植到带有垃圾编译器的更奇特的平台上 我也不
  • 从匿名类型获取值

    我有一个方法如下 public void MyMethod object obj implement 我这样称呼它 MyMethod new myparam waoww 那么我该如何实施MyMethod 获取 myparam 值 Edit
  • 过期时自动重新填充缓存

    我当前缓存方法调用的结果 缓存代码遵循标准模式 如果存在 则使用缓存中的项目 否则计算结果 在返回之前将其缓存以供将来调用 我想保护客户端代码免受缓存未命中的影响 例如 当项目过期时 我正在考虑生成一个线程来等待缓存对象的生命周期 然后运行
  • Silverlight Datagrid:在对列进行排序时突出显示整个列

    我的 Silverlight 应用程序中有一个 DataGrid 我想在对该列进行排序时突出显示整个列 它在概念上与上一个问题类似 Silverlight DataGrid 突出显示整列 https stackoverflow com qu
  • 内核开发和 C++ [关闭]

    Closed 这个问题是基于意见的 help closed questions 目前不接受答案 从我know https stackoverflow com questions 580292 what languages are windo
  • Fluent NHibernate 日期时间 UTC

    我想创建一个流畅的 nhibernate 映射来通过以下方式映射 DateTime 字段 保存时 保存 UTC 值 读取时 调整为本地时区值 实现此映射的最佳方法是什么 就我个人而言 我会将日期存储在 UTC 格式的对象中 然后在读 写时在
  • 同时从多个流中捕获、最佳方法以及如何减少 CPU 使用率

    我目前正在编写一个应用程序 该应用程序将捕获大量 RTSP 流 在我的例子中为 12 个 并将其显示在 QT 小部件上 当我超过大约 6 7 个流时 问题就会出现 CPU 使用率激增并且出现明显的卡顿 我认为它不是 QT 绘制函数的原因是因
  • 以编程方式使用自定义元素创建网格

    我正在尝试以编程方式创建一个网格 并将自定义控件作为子项附加到网格中 作为 2x2 矩阵中的第 0 行第 0 列 为了让事情变得更棘手 我使用了 MVVM 设计模式 下面是一些代码可以帮助大家理解这个想法 应用程序 xaml cs base
  • Swagger 为 ASP.CORE 3 中的字典生成错误的 URL

    当从查询字符串中提取的模型将字典作为其属性之一时 Swagger 会生成不正确的 URL 如何告诉 Swagger 更改 URL 中字典的格式或手动定义输入参数模式而不自动生成 尝试使用 Swashbuckle 和 NSwag 控制器 pu
  • 如何使用 std::array 模拟 C 数组初始化“int arr[] = { e1, e2, e3, ... }”行为?

    注意 这个问题是关于不必指定元素数量并且仍然允许直接初始化嵌套类型 这个问题 https stackoverflow com questions 6111565 now that we have stdarray what uses are

随机推荐

  • 如何在 angularjs ui-router 中的状态之间共享 $scope 数据?

    如果不使用服务或在父控制器中构造观察者 如何让子状态访问主控制器的状态 scope state main controller mainController url main templateUrl main init html state
  • 如何使用WSDL2Java生成的文件?

    我使用 axis2 1 5 中的 wsdl2java 生成了 java 文件 现在它在以下文件夹结构中生成文件 src net mycompany www services services 文件夹中的文件是 SessionIntegrat
  • HTML 输入 - 以编程方式设置输入值时撤消历史记录丢失

    我有一个 HTML 输入 当用户在其中输入内容时 我设置了 输入 事件来处理将输入更新为用户输入内容的过滤版本 以及更新选择开始和选择结束以实现流畅的用户体验 为了达到适当的效果 这种情况会不断发生 然而 我注意到 每当 JS 通过设置输入
  • 在这种情况下为什么会说“位图无法解析为类型”?

    我收到这个错误 在我的情况下我该如何解决这个问题 Bitmap cannot be resolved to a type 发生错误的行 public void onPageStarted WebView view String url Bi
  • 在 Eclipse 中输入 PHP 时数组初始值设定项缩进错误

    我在 首选项 gt PHP gt 代码样式 gt 格式化程序 gt 换行 上设置了首选数组初始值设定项缩进 但在键入数组初始值设定项缩进时 这是错误的 arr array ENTER CURSOR 当我期待时 arr array ENTER
  • 从 xhr.responseText 恢复 ArrayBuffer

    我需要从向我发送 Base64 答案的 http 请求中获取数组缓冲区 对于这个请求 我无法使用XMLHttpRequest responseType arraybuffer 我从该请求中得到的响应已通读xhr responseText 因
  • Windows 通知服务:尝试在 PHP 中创建 Toast 通知时出现 401 无效令牌

    我一直在尝试向模拟器发送 Toast 通知 我创建了 Windows Phone 8 1 应用程序并将其与商店中的应用程序关联 之后 我设法获取了必须用来调用通道 URI 的访问令牌 当我尝试使用通道 URI 和访问令牌发送 Toast 通
  • XSD 验证错误:找不到元素“soapenv:Envelope”的声明

    我尝试使用以下命令根据 XSD 验证我的 XMLhttp www freeformatter com xml validator xsd html http www freeformatter com xml validator xsd h
  • rgdal 不会安装在 AWS RStudio AMI 上

    我已成功启动最新的 RStudio AWS EC2 实例 louisaslett com RStudio 1 1383 R 3 4 2 ubuntu 16 04 LTS 64 在这种情况下 R 的运行基本符合预期 我可以安装和打开许多软件包
  • Delphi 视觉组件 - 远离 TFrame 基础? [关闭]

    就目前情况而言 这个问题不太适合我们的问答形式 我们希望答案得到事实 参考资料或专业知识的支持 但这个问题可能会引发辩论 争论 民意调查或扩展讨论 如果您觉得这个问题可以改进并可能重新开放 访问帮助中心 help reopen questi
  • 我应该如何在 Scala 和 Anorm 中使用 MayErr[IntegrityConstraintViolation,Int]?

    I use Anorm http scala playframework org documentation scala 0 9 1 anorm进行数据库查询 当我做一个executeUpdate 我应该如何进行正确的错误处理 它有返回类型
  • 伪代码归纳证明

    我不太明白如何在伪代码上使用归纳证明 它的工作方式似乎与在数学方程上使用它的方式不同 我正在尝试计算数组中可被 k 整除的整数的数量 Algorithm divisibleByK a k Input array a of n size nu
  • Python按顺序执行线程

    我有以下代码 导入线程 def send to server lst Some logic to send the list to the server while 1 lst for i in range 1000 lst append
  • Ruby 和 MySQL UTF-8 字符

    我正在切换一个Sinatra http www sinatrarb com从 SQLite3 到 MySQL 的应用程序 由于某种我无法理解的原因 当我使用 Ruby 从 MySQL 中提取数据时Sequel https rubygems
  • Angular-CLI 和 Bootstrap 4

    我使用 Angular 2 迈出了第一步 特别是我使用 Angular cli 官方工具来创建新项目 我以这种方式创建了一个新项目 ng new my project name 该项目已正确创建 之后我想安装 bootstrap 4 并按照
  • 反应流中的自定义节点;创建节点后将附加数据保存到节点

    这是我第一次介绍反应流 我希望创建一个自定义节点 创建后 用户可以在节点中输入信息并保存 显示它 从反应流自定义节点的文档 https reactflow dev docs guides custom nodes 他们有一个类似的例子 他们
  • Android Studio 中外部剥离共享库的本机调试

    我已经剥离和未剥离共享库 如何在使用 LLDB 在 Android Studio 中调试剥离时加载符号 我可以成功调试位于 jniLibs 文件夹中的完整未剥离的 so 但它太大了 部署时间太长 在调试配置中指定符号目录不适用于标准和实验性
  • 在android中实现Socket.io的最佳方式

    我计划通过以下方式在 android 中实现 Socket iothis https github com socketio socket io 基于聊天的应用程序的库 据我了解 图书馆似乎相当不错 我想知道如何维护single整个应用程序
  • 新的 PendingIntent 更新当前意图

    我试图在一段时间间隔后显示不同的通知 但发生的情况是它更新了当前的通知PendingIntent结果 即使我触发 4 5 个待处理的意图请求 我也只能收到一个通知 单击按钮后我会执行以下操作 try adapter OpenDB int i
  • 反复出现的成本难题

    我经常发现自己必须定义一个函数的两个版本 以便拥有一个 const 版本和一个非常量版本 通常是 getter 但并非总是如此 两者的区别仅在于 其中一个的输入和输出是常量 而另一个的输入和输出是非常量 该功能的核心 真正的工作 是相同的