C# 中 ?、??、??=、?: 、?. 、?[] 各种问号的用法和说明

前言

在 C# 中,问号(?)远不止是一个简单的标点符号。随着语言版本的迭代更新,C# 围绕问号(?)发展出了一套强大而优雅的空值处理和条件表达机制。熟练掌握这些操作运算符不仅能大幅提升代码的简洁性和可读性,还能有效避免恼人的空引用异常,构建更加健壮的应用程序。

可空类型修饰符(?)

在 C# 中,值类型(如int、long、bool、DateTime等)默认不能为null。使用 ? 修饰符,我们可以将值类型转换为可空类型。

ini 复制代码
            int notNullableInt;// 非空int类型默认为 0
            int? nullableInt = null;
            bool? nullableBool = null;
            long? nullableLong = null;
            DateTime? nullableDate = null;

            // 检查是否有值
            if (nullableInt.HasValue)
            {
                Console.WriteLine($"整数值: {nullableInt.Value}");
            }
            else
            {
                Console.WriteLine("变量没有值(为null)");
            }

Null 合并运算符(??)

Null 合并运算符(??)如果左边的值不为null,则返回左边的值,否则返回右边的值。

ini 复制代码
        static void Main(string[] args)
        {
            string userName1 = "小明";
            string userName2 = null;
            var getUserName = userName1 ?? userName2 ?? "默认用户";

            Console.WriteLine(getUserName); // 输出: 小明

            string config1 = null;
            string config2 = null;
            string config3 = "DefaultConfig";
            string finalConfig = config1 ?? config2 ?? config3 ?? "FallbackConfig";
            Console.WriteLine(finalConfig); // 输出: DefaultConfig
        }

Null 合并赋值运算符(??=)

C# 8.0 引入的运算符,仅当左操作数为null时,才将右操作数的值赋给左操作数。这是懒加载模式的理想选择。

csharp 复制代码
        static void Main(string[] args)
        {
            string? name = null;
            name ??= "时光者";
            Console.WriteLine(name); // 时光者

            name ??= "大姚"; // 不改变
            Console.WriteLine(name); // 时光者


            //惰性初始化
            Dictionary<string, string>? cache = null;

            cache ??= new Dictionary<string, string>();
            cache["UserName"] = name;

            Console.WriteLine(cache["UserName"]);
        }

三元条件运算符(?:)

条件运算符(?:),又称三元运算符,是一种简洁的条件表达式形式。它对布尔表达式进行求值,并根据结果为true或false,选择性地返回两个表达式中的对应结果,为简单条件判断提供了一种比传统if-else语句更紧凑、表达力更强的语法形式。

ini 复制代码
        static void Main(string[] args)
        {
            int score = 80;
            string level = score >= 60 ? "Pass" : "Fail";
            Console.WriteLine(level);
        }

Null 条件成员访问运算符 (?.)

Null 条件成员访问运算符 (?.) 在访问对象成员(属性、方法、字段等)前先检查对象是否为 null。如果对象为 null,整个表达式返回 null 而不会抛出 NullReferenceException;如果对象不为 null,则正常访问成员。

csharp 复制代码
        static void Main(string[] args)
        {
            // 基本用法
            Person person = null;
            string name = person?.Name; // 不会抛出异常,name 为 null
            Console.WriteLine(name ?? "name is null"); // 输出: name is null
        }

Null 条件索引访问运算符 (?[])

Null 条件索引访问运算符 (?[]) 在使用索引器访问集合元素前先检查集合对象是否为 null。如果集合为 null,整个表达式返回 null 而不会抛出异常;如果集合不为 null,则正常访问索引位置的元素。

csharp 复制代码
        static void Main(string[] args)
        {
            List<string> names = null;
            string firstName = names?[0]; // 不会抛出异常,firstName 为 null
            Console.WriteLine(firstName ?? "No names available"); // 输出: No names available

            // 初始化列表后访问
            names = new List<string> { "时光者", "小袁", "大姚" };
            string secondName = names?[0]; // 安全访问索引为0的元素
            Console.WriteLine(secondName); // 输出: 时光者
        }

C#/.NET/.NET Core面试宝典

本文已收录至C#/.NET/.NET Core面试宝典中,更新C#/.NET/.NET Core面试干货前往开源地址查看:

相关推荐
筱璦5 小时前
期货软件开发 - C# 调用 HQChart 指标计算 C++ 动态库
c++·microsoft·c#
武藤一雄7 小时前
C# 异常(Exception)处理避坑指南
windows·microsoft·c#·.net·.netcore·鲁棒性
武藤一雄10 小时前
WPF中ViewModel之间的5种通讯方式
开发语言·前端·microsoft·c#·wpf
雨浓YN10 小时前
OPC UA 通讯开发笔记 - 基于Opc.Ua.Client
笔记·c#
我是唐青枫11 小时前
C#.NET TPL Dataflow 深入解析:数据流管道、背压控制与实战取舍
c#·.net
SunnyDays101113 小时前
如何使用 C# 创建、修改和删除 Excel 中的 VBA 宏(无需Microsoft Excel)
c#·excel·vba宏·创建vba宏·修改vba宏·删除vba宏
唐青枫14 小时前
C#.NET gRPC 深入解析:Proto 定义、流式调用与服务间通信取舍
c#·.net
水深00安东尼14 小时前
C# 鼠标点击小游戏
c#
波波00715 小时前
每日一题:C#中using的三种用法
开发语言·c#
游乐码15 小时前
c#万物之父
开发语言·c#