告别版本地狱:C# 中央包管理

告别版本地狱:C# 中央包管理

痛点:散落各处的包版本

你是否遇到过这样的场景?

  • 解决方案有 20 个项目,Newtonsoft.Json 出现了 5 种不同版本
  • 升级一个包要改十几个 .csproj 文件
  • 代码合并时频繁因为版本号冲突

NuGet 在 .NET 6 中引入的中央包管理(CPM) 正是为了解决这些问题。

传统方式下,每个项目独立声明版本:

xml 复制代码
<!-- 传统方式:版本散落在各个项目中 -->
<PackageReference Include="Newtonsoft.Json" Version="13.0.3" />

启用中央包管理后,项目只需声明"我要用这个包":

xml 复制代码
<!-- 中央包管理:版本由统一文件管理 -->
<PackageReference Include="Newtonsoft.Json" />

核心文件:Directory.Packages.props

中央包管理的核心是一个名为 Directory.Packages.props 的文件:

xml 复制代码
<Project>
  <PropertyGroup>
    <ManagePackageVersionsCentrally>true</ManagePackageVersionsCentrally>
  </PropertyGroup>

  <ItemGroup>
    <PackageVersion Include="Newtonsoft.Json" Version="13.0.3" />
    <PackageVersion Include="Serilog" Version="3.1.1" />
    <PackageVersion Include="xunit" Version="2.6.2" />
  </ItemGroup>
</Project>

迁移三步走

Step 1:创建版本清单

在解决方案根目录创建 Directory.Packages.props,用 <PackageVersion> 列出所有包版本。

Step 2:精简项目文件

移除各 .csproj<PackageReference>Version 属性:

xml 复制代码
<!-- 修改前 -->
<PackageReference Include="Newtonsoft.Json" Version="13.0.3" />

<!-- 修改后 -->
<PackageReference Include="Newtonsoft.Json" />

Step 3:处理特殊情况

个别项目需要使用不同版本?用 VersionOverride 覆盖:

xml 复制代码
<PackageReference Include="Newtonsoft.Json" VersionOverride="12.0.3" />

幕后机制:MSBuild 何时加载版本信息

MSBuild 在项目评估阶段(Evaluation Phase) 查找并加载 Directory.Packages.props,具体时机:

  1. dotnet restore 执行时
  2. dotnet build 开始前的还原阶段
  3. Visual Studio 加载项目时

版本信息在解析 <PackageReference> 之前就已就绪。

文件查找规则:向上追溯

MSBuild 从项目文件所在目录开始,逐级向上查找 ,直到找到 Directory.Packages.props 或到达根目录。

复制代码
C:\
└── MyRepo\
    ├── Directory.Packages.props    ← 解决方案级(推荐位置)
    ├── MySolution.sln
    ├── src\
    │   ├── Directory.Packages.props  ← 子目录级(可覆盖上级)
    │   ├── ProjectA\
    │   │   └── ProjectA.csproj       ← 先找 src\,再找 MyRepo\
    │   └── ProjectB\
    │       └── ProjectB.csproj
    └── tests\
        └── TestProject\
            └── TestProject.csproj    ← 直接命中 MyRepo\ 下的文件

规则小结:

  • 起点:.csproj 所在目录
  • 方向:逐级向父目录
  • 终止:找到第一个即停止
  • 扩展:可通过 <Import> 导入多个文件

为什么值得迁移

收益 说明
版本一致性 杜绝"版本漂移",全解决方案统一
一处修改 升级包只改一个文件
告别冲突 PR 合并不再为版本号打架
依赖审计 一个文件纵览全部依赖
团队规范 强制统一管理,避免各自为战
文件瘦身 .csproj 更清爽

踩坑提醒

  1. 全员参与:启用 CPM 后,解决方案内所有项目必须遵循,不能"一半传统一半中央"
  2. 传递依赖无需声明:只需声明直接引用的包,传递依赖自动处理
  3. 全局包引用:Analyzer 等全局包可直接在 props 文件中定义:
xml 复制代码
<ItemGroup>
  <GlobalPackageReference Include="StyleCop.Analyzers" Version="1.2.0-beta.507" />
</ItemGroup>
相关推荐
追逐时光者6 小时前
一个致力于为 C# 程序员提供更佳的编码体验和效率的 Visual Studio 扩展插件
后端·c#·visual studio
SunflowerCoder8 小时前
EF Core + PostgreSQL 配置表设计踩坑记录:从 23505 到 ChangeTracker 冲突
数据库·postgresql·c#·efcore
阿蒙Amon10 小时前
C#每日面试题-常量和只读变量的区别
java·面试·c#
我是唐青枫10 小时前
C#.NET ConcurrentBag<T> 设计原理与使用场景
c#·.net
大王小生11 小时前
C# CancellationToken
开发语言·c#·token·cancellation
listhi52011 小时前
基于C#实现屏幕放大镜功能
开发语言·c#
该用户已不存在14 小时前
不止是初始化,4个C# 构造函数解析与实例
后端·c#·.net
无风听海17 小时前
深入讲解 C# 中 string 如何支持 CultureInfo
开发语言·c#
wzfj1234517 小时前
FreeRTOS xTaskCreateStatic 详解
开发语言·c#
kylezhao201918 小时前
C# DataGridView 控件使用详解
c#