C# 程序的常规结构 — 命名空间、类型、入口点与表达式

34 阅读5分钟

很多人刚装好 .NET SDK,敲下 dotnet new console​ 之后会困惑:Program.cs​ 里为什么没有 Main​ 方法?namespace​ 去哪了?那些 class​、struct​、interface 又是什么关系?这篇就是 C# 程序的"骨架说明书"——从顶层到底层,把一个 .cs 文件里能看到的所有结构元素串一遍。

本文基于 Microsoft Learn 官方文档整理。

  1. 两大选择:基于文件 vs 基于项目、顶级语句 vs Main 方法
  2. 核心构建基块:命名空间、类型(类/结构/接口/枚举/委托)的层级关系
  3. 表达式与语句:程序运行时的最小执行单元

一、两大基础选择

在创建 C# 程序之前,首先要做两个决策:

1.1 基于文件还是基于项目?

维度基于文件的应用基于项目的应用
引入版本C# 14 / .NET 10所有版本
项目文件无需.csproj需要.csproj
运行方式dotnet run hello-world.csdotnet build​+dotnet run
多文件支持单文件天然支持多文件
适用场景小型命令行工具、原型、试验多文件项目、复杂生成配置

目录结构对比:

 基于文件:
 my-app/
 └── hello-world.cs
 ​
 基于项目:
 my-app/
 ├── my-app.csproj
 ├── Program.cs
 ├── Models/
 │   └── Person.cs
 └── Services/
     └── GreetingService.cs

关键点: 基于文件的应用可以随时用 dotnet project convert 转换为基于项目的应用,进可攻退可守。

1.2 顶级语句还是 Main 方法?

入口点样式写法特点
顶级语句文件顶部直接写代码dotnet new console默认样式,简洁
显式 Main 方法static void Main(string[] args)传统风格,明确入口点

限制: 一个项目中只能有一个文件使用顶级语句,且入口点从该文件的第一行开始。

 // 顶级语句 — 无需 class 和 Main 包裹
 Console.WriteLine("Hello, World!");
 ​
 // 在 Unix 下甚至可以加 shebang 直接执行
 #!/usr/bin/env dotnet
 Console.WriteLine("Hello, World!");
 // 显式 Main 方法 — 传统入口
 namespace YourNamespace;
 ​
 class Program
 {
     static void Main(string[] args)
     {
         Console.WriteLine("Hello, World!");
     }
 }

两种入口点样式支持相同的功能,选哪种纯粹是风格偏好。

二、核心构建基块:完整的程序骨架

C# 程序由以下层级构成:

 程序
  └── 命名空间(namespace)
       └── 类型(class / struct / interface / enum / delegate)
            └── 成员(字段、属性、方法、事件...)
                 └── 语句(statement)
                      └── 表达式(expression

2.1 完整示例

 // 顶级语句作为入口点
 Console.WriteLine("Hello, World!");
 ​
 // 命名空间:组织类型的容器
 namespace YourNamespace
 {
     // 类:引用类型,封装数据和行为
     class YourClass
     {
     }
 ​
     // 结构:值类型,轻量级数据载体
     struct YourStruct
     {
     }
 ​
     // 接口:定义行为契约
     interface IYourInterface
     {
     }
 ​
     // 委托:类型安全的方法指针
     delegate int YourDelegate();
 ​
     // 枚举:命名常量集合
     enum YourEnum
     {
     }
 }

各类型定位速查:

类型关键字值/引用用途
class引用类型封装数据和行为,面向对象核心
结构struct值类型轻量级数据载体,栈分配
接口interface定义行为契约,不含实现
枚举enum值类型命名常量集合
委托delegate引用类型类型安全的方法引用

划重点: class​ 和 struct​ 的核心区别是值类型 vs 引用类型。struct​ 在赋值时复制全部数据,class 只复制引用。这直接影响性能和语义,不可轻视。

三、生成和运行

C# 是编译型语言 —— 源代码先编译为 IL(中间语言),再由 .NET 运行时 JIT 编译为机器码。

命令作用适用
dotnet build编译项目基于项目的应用
dotnet run编译 + 运行(一步到位)基于项目的应用
dotnet run hello-world.cs直接编译并运行单个文件基于文件的应用

四、表达式与语句

表达式和语句是程序运行时的最小执行单元。

4.1 表达式:产生值的东西

表达式类型说明
42字面值最简单的表达式
x + y算术运算运算符连接操作数
Math.Max(a, b)方法调用调用有返回值的方法
condition ? a : b条件表达式三元运算
new Person("John")对象创建new 表达式

4.2 语句:执行操作的东西

语句通常以分号 ; 结尾。

语句类别说明
int x;声明语句声明变量
int x = 42;声明语句声明并初始化
Console.WriteLine("Hello");方法调用语句调用方法(忽略返回值)
if (condition) { ... }条件语句分支控制
return result;return 语句方法返回

4.3 语句和表达式的关系

语句常常包含表达式,表达式可以嵌套:

 var maxResult = Math.Max(a, b) + Math.Max(c, d);

拆解:

  1. Math.Max(a, b)​ 和 Math.Max(c, d) 是两个方法调用表达式
  2. + 将两个表达式的结果相加,产生一个新表达式
  3. var maxResult = ... 是一条声明语句,将表达式的结果赋给变量

一句话区分: 表达式产生值,语句执行操作。Math.Max(a, b)​ 是表达式(有返回值),Console.WriteLine(...) 是语句(重在副作用)。

五、程序结构速查图

 ┌─────────────────────────────────────────┐
 │  入口点                                   │
 │  · 顶级语句:文件第一行直接写代码           │
 │  · 或 static void Main(string[] args)    │
 ├─────────────────────────────────────────┤
 │  namespace 命名空间(可选)                │
 │  ├── class 类                            │
 │  ├── struct 结构                         │
 │  ├── interface 接口                      │
 │  ├── enum 枚举                           │
 │  └── delegate 委托                       │
 ├─────────────────────────────────────────┤
 │  类型成员                                 │
 │  └── 字段、属性、方法、事件、索引器...      │
 ├─────────────────────────────────────────┤
 │  可执行代码                               │
 │  └── 语句(包含表达式)                    │
 └─────────────────────────────────────────┘

最后

C# 程序结构看起来层级多,但理解了就那几样东西:入口点、命名空间、类型、成员、语句。新手最容易困惑的是"顶级语句为什么没有 Main"——答案是编译器帮你生成了,你看到的顶层代码最终会被包进一个隐式的 <Main>$ 方法里。语法糖是为了让你少写模板代码,底层模型并没有变。