逗号运算符的一个令人信服的案例

逗号运算符的一个令人信服的案例

逗号运算符是类#%#$#%@%@%$#%$#%#%#$%@_9e6df79f947a44c++8a2ba49c4428632a1(例如javascript和c++)中鲜为人知的运算符之一。本质上,它界定了一系列表达式并仅返回最后一个表达式的结果。

const a = 1;const b = 2;const c = 3;const result = (a, b, c, 4, 5, 6, true);console.log(result); // true
if (false, true) console.log('hello'); // hello

很自然地会问:什么时候将多个表达式塞进一行会有用?此外,即使它有用,为什么用逗号分隔的表达式序列(在单行)比分号分隔的语句序列(跨多行)更具可读性和可维护性?我们什么时候应该选择其中一种而不是另一种?

这些是我多年来一直努力回答的问题,但现在我想我终于有了答案。在本文中,我针对逗号运算符提出了一个令人信服的案例(坦率地说,也许是唯一的一个案例)。

一个激励人心的例子

我们先来说一下条件三元运算符。如下所示,如果条件为真,则评估值。否则,它评估另一个。这里强调关键字“评估”,因为分支仅在满足条件时执行。

const result = condition ? value : another;

在大多数情况下,它都是整洁漂亮的。然而,当我们需要在返回条件值之前在分支之间执行更复杂的逻辑时,它就会崩溃。此时,我们诉诸这种不幸的变态:

let result; // uninitialized! yikes!if (condition) {    // do some complex stuff in between...    dosomething();    // ...    result = value; // actual assignment} else {    // do other complex stuff in between...    doanotherthing();    // ...    result = another; // actual assignment}// hopefully we didn't forget to initialize `result`!

现在这个配方存在很多问题。

结果一开始是未初始化的。这本质上并不是邪恶的,但避免由于未定义而导致错误的一种简单且经过验证的方法是始终初始化变量。结果的初始化实际上是在分支的底部——与其声明相分离。在条件结束时,我们最好希望结果肯定已初始化。如果不是我们,我们最好希望我们的队友同样执行这一点。如果不是现在,我们最好希望未来的开发者也能坚持这一点!

如果我们坚持使用条件三元表达式,就有办法绕过这个限制。我们只需要将代码重构为函数即可。这绝对是说起来容易做起来难。这个噱头很快就过时了!

function computewrappedvalue() {    // ...    return value;}function computewrappedanother() {    // ...    return another;}// how cumbersome!const result = condition ? computewrappedvalue() : computewrappedanother();

基于表达式的编程语言(例如 rust)有更优雅的解决方案。通过将 if 语句 重新分类为 if 表达式 ,可以对每个分支进行求值,从而返回稍后可以存储在变量中的值。

// a conditional ternary operator thus looks like this. each branch// returns a value, which is captured by the `result` variable.// we thus ensure that `result` is always initialized by construction.let result = if condition { value } else { another };
// if we wanted to do something more complex, we use the same syntax.let result = if condition {    do_something();    // in rust, the last expression without a semicolon is the value    // that will be "returned" by the overall `if` expression.    result} else {    do_another_thing();    another};

我们可以用类似 c 的语言来模拟这个吗?你可能早就预见到了我的发展方向,但是可以!

一个令人信服的案例

我们想要的是一种在返回三元分支内的值之前任意执行语句的方法。好吧,对我们来说幸运的是,这正是逗号运算符的用途。

// parenthesized for clarity.const result = condition    ? (dosomething(), value)       // evaluates to `value`    : (doanotherthing(), another); // evaluates to `another`

这个公式的巧妙之处在于,分支表达式仅在必要时才进行计算。我们有效地模拟基于表达式的编程语言的行为。临时包装函数的时代已经一去不复返了!

但是可惜,我们只能用这种技术走这么远。你可以想象,对于一些足够大的 n,将 n 条语句塞进一行已经需要重构到它自己的函数中。就我个人而言,到 > 3 时我已经重新考虑了。就可读性而言,任何高于此值的结构都是可疑的。

// maybe we should reconsider here?const result = condition    ? (x++, thing = hello(), dosomething(), value)    : (++y, thing = world(), doanotherthing(), another);
// Okay, stop. Definitely turn back now!const result = condition    ? (        x++,        thing = hello(),        doSomething(),        doMore(y),        doEvenMore(thing),        value,    ) : (        ++y,        thing = world(),        doAnotherThing(),        doMore(y),        doEvenMore(thing),        another,    );// Unless, of course, you're fine with this. It kinda does// look like a Rust `if` expression if you squint hard enough.

结论

总而言之,我们看到了逗号运算符的一个令人信服的案例:复杂的条件三元运算。当分支又短又甜时,逗号运算符会大放异彩,但在三个内联语句之后很快就会过时。到那时,重构代码可能会更好。

那么你应该使用逗号运算符吗?老实说……是的!可读的代码会考虑到下一个读者,因此只要逗号链不会太长,我就会接受甚至鼓励这种编码风格。如果我们考虑替代方案(即未初始化的变量和重构的微函数),逗号运算符毕竟没那么糟糕。

在实践中,我已经在自己的代码库中添加了这些看起来很有趣的逗号运算符。但公平地说,我很少需要多语句三元条件。但当我这样做时,我的腰带上就有了一个很酷的工具,可以简洁地表达我的意图。

为此,我对逗号运算符提出了令人信服的理由。

以上就是逗号运算符的一个令人信服的案例的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1458950.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月18日 11:03:40
下一篇 2025年12月18日 11:03:52

相关推荐

  • clion和vscode哪个好

    哪款 IDE 更胜一筹?入门门槛:VSCode 门槛较低,适合初学者。语言支持:VSCode 支持多种语言,CLion 专注于 C/C++。调试功能:CLion 调试工具更全面。代码补全:两者均提供优秀补全,CLion 专注于 C/C++。版本控制:两者集成良好。价格:VSCode 免费,CLion…

    2025年12月18日
    000
  • dev cpp的安装方法

    在 Dev C++ 中安装 MinGW 的步骤:从 MinGW 官方网站下载安装程序。运行安装程序,选择“自定义安装”。选择“gcc-core”、“gcc-g++”、“mingw32-headers”和“mingw32-make”组件。安装 MinGW。在 Dev C++ 中配置 MinGW 的编译…

    2025年12月18日
    000
  • clion汉化教程

    CLion可通过下载并安装汉化包来汉化,汉化步骤如下:下载汉化包;解压汉化包并将内容复制到CLion安装目录下的plugins文件夹;重启CLion并设置语言为中文。 Clion 汉化教程 如何汉化 CLion CLion 是 JetBrains 开发的一款跨平台 C/C++ IDE,默认语言为英语…

    2025年12月18日
    000
  • 如何安装clion

    安装 CLion 步骤包括:下载安装程序。安装 CLion。配置语言、主题和键盘映射。获取编译器(Windows:MinGW,macOS:Xcode,Linux:GCC/Clang)。配置编译器。创建或打开项目,即可开始使用 CLion。 如何安装 CLion 步骤 1:下载 CLion 安装程序 …

    2025年12月18日
    000
  • clion怎么配置c语言环境

    要为 CLion 配置 C 语言环境,请按照下列步骤操作:安装 C 语言编译器(如 GCC 或 Clang)。在 CLion 的“首选项”中配置编译器路径。创建项目或打开现有项目,并在“运行/调试设置”中选择编译器。将 C 文件添加到项目。编译并运行程序。 如何为 CLion 配置 C 语言环境 为…

    2025年12月18日
    000
  • C++ 函数中异常和错误处理的设计模式

    在 c++++ 中处理异常和错误的设计模式包括:try-catch 块:用于处理异常事件。nothrow 保证:指定函数不会抛出异常,否则终止程序。错误码:整数表示函数失败原因,调用者可检查以确定错误。 C++ 函数中异常和错误处理的设计模式 异常和错误处理是软件开发中不可或缺的一部分。在 C++ …

    2025年12月18日
    000
  • C++ 匿名函数与函数对象的类型推导

    匿名函数和函数对象的类型推导在 c++++11 中得到改进,允许自动类型推导,简化了使用。匿名函数的语法为:[capture_clause](parameters) -> return_type { body},其中 capture_clause 捕获外部变量,parameters 指定参数,…

    2025年12月18日
    000
  • C++ 函数中处理错误代码的技巧

    在 c++++ 中,处理错误代码至关重要。通过遵循以下技巧来有效管理错误,提高代码的稳健性:使用异常处理检测并处理意料之外的错误,携带有关错误的信息。返回错误代码表示操作状态,由调用方检查以确定成功与否。使用 errno 获取最后一个系统调用产生的错误代码,用于确定错误类型。定义自定义错误类,提供有…

    2025年12月18日
    000
  • 如何有效预防 C++ 函数中的异常抛出?

    有效预防 c++++ 函数中异常抛出的方法包括:使用 raii 释放资源;对输入参数进行验证;谨慎使用指针;指定异常规范;仔细管理线程。 如何有效预防 C++ 函数中的异常抛出 异常处理是处理运行时错误的一种机制。在 C++ 中,异常可以通过 throw 关键字抛出,并在程序中的某个位置通过 try…

    2025年12月18日
    000
  • 如何提升 C++ 函数的执行效率?

    为了提高 c++++ 函数执行效率,可以:1. 内联函数以避免函数调用开销;2. 避免动态内存分配;3. 使用引用和指针传递函数参数;4. 优化算法选择;5. 避免不必要的计算。通过采用这些技巧,优化后的代码可以显著提升执行效率,从而优化程序性能。 提升 C++ 函数执行效率的技巧 提高 C++ 函…

    2025年12月18日
    000
  • C++ 函数对象是否可以实现匿名函数的功能

    c++++函数对象是匿名函数的替代方案,用于封装函数。它们可以通过类或lambda表达式创建,在程序中像常规函数一样使用。函数对象可以传递给函数、存储在容器中,并与stl算法一起使用,以执行自定义操作。 C++ 函数对象:匿名函数的替代方案 简介 C++ 函数对象提供了一种封装函数而无需指定函数名的…

    2025年12月18日
    000
  • C++ 函数优化与调试的编程之道:代码效率提升的捷径

    在 c++++ 编程中,优化和调试函数至关重要。优化技术包括内联函数、使用常量、减少不必要的拷贝。调试技术包括使用调试器、断言和日志记录。这些技术有助于提升函数性能和可靠性。实战案例展示了使用动态规划优化斐波那契函数,降低了时间复杂度,提升了性能。持续实践这些技术将使代码更加高效和健壮。 C++ 函…

    2025年12月18日
    000
  • C++ 函数对象中状态的保持方法

    c++++ 函数对象保持状态的方法:类成员函数:函数对象作为类成员函数,状态存储在类成员变量中。静态变量:函数对象状态存储在静态变量中,整个程序范围内有效。函数:函数对象实现为函数,状态通过参数显式传递。 C++ 函数对象中状态的保持方法 函数对象是 C++ 中一种特殊类型的对象,可以作为函数进行调…

    2025年12月18日
    000
  • C++ 函数中错误处理的常见模式

    c++++ 函数中常见的错误处理模式包括:返回值:用于通过函数返回值指示错误。抛出异常:通过抛出异常对象将错误传递给调用代码。使用错误代码:通过返回整数值表示特定错误,以便捕获代码可以采取适当的操作。 C++ 函数中错误处理的常见模式 在 C++ 函数中进行错误处理对于构建健壮可靠的应用程序至关重要…

    2025年12月18日
    000
  • C++ 函数优化之天书:提升性能的秘诀

    c++++ 函数优化必备指南:内联函数: 消除函数调用的开销,使用 inline 关键字。减少函数参数: 只传递必要的参数。使用引用和指针: 避免数据复制。避免深度递归: 转而使用循环或尾递归。 C++ 函数优化指南 引言 函数优化在软件开发中至关重要,因为它可以大幅提升性能。本文将为您提供 C++…

    2025年12月18日
    000
  • C++ 匿名函数与函数对象的异步编程

    匿名函数和函数对象可用于 c++++ 中的异步编程,实现并行和并发代码。匿名函数允许在运行时定义函数,而函数对象则是一种封装函数的类。实战案例包括:使用匿名函数进行异步 i/o,使用函数对象进行多线程图像处理。 C++ 匿名函数与函数对象的异步编程 引言C++ 中的匿名函数和函数对象是两种强大的机制…

    2025年12月18日
    000
  • C++ 匿名函数与函数对象的联系与区别

    匿名函数和函数对象都是可调用的实体,区别在于:语法:匿名函数使用 lambda 表达式,函数对象使用函数指针或 std::function 包装器。使用方式:匿名函数通常用于一次性操作,而函数对象可以多次使用。存储方式:函数对象可以在标准库中直接存储或传递,而匿名函数需要 std::function…

    2025年12月18日
    000
  • C++ 函数命名中驼峰命名的优缺点

    驼峰命名是一种 c++++ 命名约定,用于复合单词,其中各个部分以小写开头,首字母大写(例如:getaveragescore())。优点包括易于阅读、一致性和工具支持,而缺点则包括键盘容错性、屏幕空间占用和缺乏灵活性。 C++ 函数命名中的驼峰命名 在 C++ 中,驼峰命名是一种命名约定,其中复合单…

    2025年12月18日
    000
  • C++ 函数优化与调试秘籍:让代码飞起来

    c++++ 函数优化与调试指南:1. 优化:内联函数,传递引用,避免虚拟调用,缓存计算结果;2. 调试:使用断点,调试器,异常处理,日志记录。 C++ 函数优化与调试秘籍:让代码飞起来 简介 优化和调试函数对于提升 C++ 代码的性能和可靠性至关重要。本文将探讨一些有助于提高函数效率和简化调试过程的…

    2025年12月18日
    000
  • C++ 函数中异常处理的常见陷阱

    c++++ 函数中异常处理的常见陷阱有:异常丢失:未处理的异常向上抛出,导致应用程序终止。多次抛出异常:覆盖异常类型,丢失异常上下文。异常泄露:在异常处理程序之外捕获异常,导致异常丢失。多余的异常处理:滥用异常处理,影响代码可读性。异常信息不足:抛出时未提供足够信息,难以调试。 C++ 函数中异常处…

    2025年12月18日
    000

发表回复

登录后才能评论
关注微信