C#的??和??=运算符在空值处理中有何作用?

小老鼠
发布: 2025-09-08 08:05:01
原创
477人浏览过

?? 运算符在左侧为 null 时返回右侧默认值,否则返回左侧值;2. ??= 运算符仅在左侧为 null 时才将右侧值赋给左侧;3. 两者通过延迟计算避免不必要的性能开销且编译为高效 il 代码;4. 适用于简化 null 检查、默认值赋值、链式 null 判断、属性初始化及避免重复计算;5. 替代方案包括 if 语句、三元运算符、getvalueordefault()、扩展方法和模式匹配,但在代码简洁性和可读性上不如 ?? 和 ??=。

C#的??和??=运算符在空值处理中有何作用?

C# 中的

??
登录后复制
(空合并运算符) 和
??=
登录后复制
(空合并赋值运算符) 主要用于简化空值判断和赋值操作,让代码更简洁易读。它们能有效避免 NullReferenceException,并提供默认值。

解决方案

??
登录后复制
运算符(空合并运算符):

??
登录后复制
运算符允许你提供一个默认值,当左侧的操作数为
null
登录后复制
时,该默认值会被返回。否则,返回左侧操作数的值。

string name = null;
string displayName = name ?? "Unknown"; // displayName 将被赋值为 "Unknown"
Console.WriteLine(displayName); // 输出: Unknown

name = "Alice";
displayName = name ?? "Unknown"; // displayName 将被赋值为 "Alice"
Console.WriteLine(displayName); // 输出: Alice
登录后复制

这里,如果

name
登录后复制
null
登录后复制
displayName
登录后复制
将会被赋值为 "Unknown"。如果
name
登录后复制
有值,
displayName
登录后复制
将会直接使用
name
登录后复制
的值。

??=
登录后复制
运算符(空合并赋值运算符):

??=
登录后复制
运算符仅在左侧的操作数为
null
登录后复制
时,才将右侧的操作数赋值给左侧的操作数。

string message = null;
message ??= "Hello, World!"; // 仅当 message 为 null 时,才赋值
Console.WriteLine(message); // 输出: Hello, World!

message ??= "Goodbye!"; // 这一行不会执行任何操作,因为 message 已经有值
Console.WriteLine(message); // 输出: Hello, World!
登录后复制

在这个例子中,

message
登录后复制
最初是
null
登录后复制
,所以
??=
登录后复制
运算符将其赋值为 "Hello, World!"。第二次使用
??=
登录后复制
时,
message
登录后复制
已经不是
null
登录后复制
,所以赋值操作被跳过。

C#

??
登录后复制
??=
登录后复制
运算符的性能考量?

??
登录后复制
??=
登录后复制
运算符通常具有良好的性能,因为它们是 C# 编译器直接支持的语法糖。编译器会将这些运算符转换为高效的 IL (Intermediate Language) 代码。 但是,一些因素可能会影响性能:

  1. 操作数的复杂性: 如果左侧的操作数涉及复杂的计算或属性访问,那么计算

    null
    登录后复制
    检查之前的操作数本身可能会带来性能开销。

  2. 默认值的计算: 如果

    ??
    登录后复制
    ??=
    登录后复制
    右侧的默认值需要进行昂贵的计算(例如,创建新对象或调用耗时的方法),那么只有在左侧操作数为
    null
    登录后复制
    时才会执行这些计算,这可以避免不必要的性能开销。 这种延迟计算是
    ??
    登录后复制
    ??=
    登录后复制
    的一个优点。

  3. JIT 编译器的优化: JIT (Just-In-Time) 编译器能够对使用

    ??
    登录后复制
    ??=
    登录后复制
    运算符的代码进行优化,例如内联简单的操作。

在大多数情况下,

??
登录后复制
??=
登录后复制
运算符的性能开销可以忽略不计,并且它们带来的代码简洁性和可读性提升通常远大于潜在的性能损失。 然而,在性能敏感的代码段中,最好进行基准测试来评估其影响。

何时应该使用 C# 的

??
登录后复制
??=
登录后复制
运算符?

  1. 简化 Null 检查: 当你需要检查一个变量是否为

    null
    登录后复制
    ,并根据情况提供默认值时,
    ??
    登录后复制
    运算符非常有用。 这可以避免使用冗长的
    if
    登录后复制
    语句或三元运算符。

    string name = GetNameFromDatabase(); // 假设这个方法可能返回 null
    string displayName = name ?? "Guest"; // 如果 GetNameFromDatabase() 返回 null,则使用 "Guest"
    登录后复制
  2. 默认值赋值: 当你希望仅在变量为

    null
    登录后复制
    时才进行赋值时,
    ??=
    登录后复制
    运算符非常方便。

    有道小P
    有道小P

    有道小P,新一代AI全科学习助手,在学习中遇到任何问题都可以问我。

    有道小P 64
    查看详情 有道小P
    List<string> items = null;
    items ??= new List<string>(); // 仅当 items 为 null 时,才初始化列表
    items.Add("Item 1");
    登录后复制
  3. 链式 Null 检查: 你可以将

    ??
    登录后复制
    运算符链接在一起,以便在多个可能为
    null
    登录后复制
    的值中选择第一个非
    null
    登录后复制
    值。

    string result = value1 ?? value2 ?? value3 ?? "Default Value";
    // result 将被赋值为 value1, value2, value3 中第一个非 null 的值,
    // 如果所有值都是 null,则赋值为 "Default Value"
    登录后复制
  4. 简化属性初始化: 在构造函数或属性设置器中,可以使用

    ??=
    登录后复制
    运算符来确保属性具有默认值,而无需编写额外的
    if
    登录后复制
    语句。

    private string _description;
    public string Description
    {
        get => _description;
        set => _description ??= "No description provided."; // 仅在 _description 为 null 时赋值
    }
    登录后复制
  5. 避免重复计算: 当默认值的计算成本较高时,

    ??
    登录后复制
    ??=
    登录后复制
    运算符的延迟计算特性可以避免不必要的性能开销。

    string data = GetDataFromCache(); // 假设这个方法可能返回 null
    string result = data ?? LoadDataFromDatabase(); // 仅当 data 为 null 时,才从数据库加载数据
    登录后复制

??
登录后复制
??=
登录后复制
运算符提高了代码的可读性和简洁性,尤其是在处理可能为
null
登录后复制
的值时。

C# 中

??
登录后复制
??=
登录后复制
运算符的替代方案?

在 C# 中,

??
登录后复制
??=
登录后复制
运算符主要用于处理空值情况,但在没有这些运算符的情况下,仍然有其他方法可以实现类似的功能。

  1. 使用

    if
    登录后复制
    语句: 这是最基本的方法,通过显式地检查变量是否为
    null
    登录后复制
    ,然后根据结果执行不同的操作。

    string name = GetName();
    string displayName;
    if (name == null)
    {
        displayName = "Guest";
    }
    else
    {
        displayName = name;
    }
    登录后复制

    这种方法比较冗长,但易于理解,并且在所有 C# 版本中都可用。

  2. 使用三元运算符

    ?:
    登录后复制
    三元运算符提供了一种更简洁的方式来进行条件赋值。

    string name = GetName();
    string displayName = name == null ? "Guest" : name;
    登录后复制

    三元运算符比

    if
    登录后复制
    语句更简洁,但当条件逻辑复杂时,可能会降低可读性。

  3. 使用

    GetValueOrDefault()
    登录后复制
    方法: 对于可空类型(如
    int?
    登录后复制
    DateTime?
    登录后复制
    ),可以使用
    GetValueOrDefault()
    登录后复制
    方法来获取值,并在值为
    null
    登录后复制
    时返回默认值。

    int? age = GetAge();
    int actualAge = age.GetValueOrDefault(18); // 如果 age 为 null,则 actualAge 为 18
    登录后复制

    GetValueOrDefault()
    登录后复制
    方法仅适用于可空类型。

  4. 使用扩展方法: 可以创建自定义的扩展方法来模拟

    ??
    登录后复制
    ??=
    登录后复制
    运算符的行为。

    public static class NullExtensions
    {
        public static T Coalesce<T>(this T value, T defaultValue) where T : class
        {
            return value ?? defaultValue;
        }
    
        public static void CoalesceAssign<T>(this ref T value, T defaultValue) where T : class
        {
            if (value == null)
            {
                value = defaultValue;
            }
        }
    }
    
    // 使用扩展方法
    string name = GetName();
    string displayName = name.Coalesce("Guest");
    
    string message = null;
    message.CoalesceAssign("Hello, World!");
    登录后复制

    这种方法允许你自定义空值处理逻辑,但需要编写额外的代码。

  5. 使用模式匹配(C# 7.0 及更高版本): 可以使用模式匹配来简化空值检查和赋值。

    string name = GetName();
    string displayName = name switch
    {
        null => "Guest",
        _ => name
    };
    登录后复制

    模式匹配提供了一种更灵活的方式来处理多种情况,包括空值。

虽然

??
登录后复制
??=
登录后复制
运算符通常是处理空值的最佳选择,但这些替代方案在特定情况下可能仍然有用,或者在不支持这些运算符的旧版本 C# 中是必需的。

以上就是C#的??和??=运算符在空值处理中有何作用?的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号