Skip to content

dotnet 新 SDK Style 项目格式如何使用 InternalsVisibleToAttribute 功能

Updated: at 08:22,Created: at 08:19

如果一个项目想要让其他某个指定的项目可以使用到 internal 的类或成员,可以通过标记 InternalsVisibleToAttribute 的方式实现

最简单的方法是新建一个 AssemblyInfo.cs 文件,在这个文件里面使用 System.Runtime.CompilerServices.InternalsVisibleToAttribute 指定某个程序集可见

using System;
using System.Reflection;
[assembly: System.Runtime.CompilerServices.InternalsVisibleToAttribute("Lindexi.blog.csdn.net")]

对于强签名的程序集,只能被强签名的程序集可见

如果想要在 csproj 文件上面写,也可以,在 ItemGroup 添加下面代码

<ItemGroup>
<InternalsVisibleTo Include="Lindexi.blog.csdn.net" />
</ItemGroup>

这是在 .NET 5 的 SDK 提供的。敲黑板,由 .NET 5 的 SDK 提供,不等价于你必须使用 TargetFramework 为 net5.0 版本,而是只要你在开发机器上,安装了 .NET 5 或更高版本的 SDK 即可写出如上代码,和你的项目具体使用什么版本没有任何关系。简单说就是,即使你的项目是 .NET Framework 4.5 的,但是在你的开发设备上安装了 .NET 5 或更高版本的 SDK 那么即可使用如上代码

以上功能是在 https://github.com/dotnet/sdk/pull/3439 加上的


以下是废弃的内容

在 2019 的 10 月份之前,如果你穿越回过去的话,那么也许需要使用如下方法

如果想要在 csproj 文件上面写,也可以,在 ItemGroup 添加下面代码

<ItemGroup>
<AssemblyAttribute Include="System.Runtime.CompilerServices.InternalsVisibleToAttribute">
<_Parameter1>Lindexi.blog.csdn.net</_Parameter1>
</AssemblyAttribute>
</ItemGroup>

为什么这样就可以实现和原先 AssemblyInfo.cs 相同的功能?其实在构建的之前,将会执行预编译,将 AssemblyAttribute 的内容输出到 obj\*.AssemblyInfo.cs 文件,小伙伴可以尝试打开这个文件,其实这个文件是由 WriteCodeFragment 生成,内容和刚才的 AssemblyInfo.cs 文件是差不多的

所以本质上是通过预编译创建 AssemblyInfo.cs 文件实现。只是用这个方法可以不手工创建 AssemblyInfo.cs 文件

如我创建的 WPF 项目,这个项目里面有一个 Foo 类,期望被其他两个项目使用,此时可以添加如下代码

<ItemGroup>
<AssemblyAttribute Include="System.Runtime.CompilerServices.InternalsVisibleToAttribute">
<_Parameter1>LeerijawarnikoJacallnachaykall</_Parameter1>
</AssemblyAttribute>
<AssemblyAttribute Include="System.Runtime.CompilerServices.InternalsVisibleToAttribute">
<_Parameter1>HikallyijuDallcurjemdehowai</_Parameter1>
</AssemblyAttribute>
</ItemGroup>

代码放在 github 欢迎小伙伴访问

当然这么写代码比较乱,可以通过小伙伴 Meziantou 的方法,只需要安装一个有趣的 NuGet 包,就可以使用十分清真的写法。安装 NuGet 的方法是在 csproj 里面的添加下面代码

<ItemGroup>
<PackageReference Include="Meziantou.MSBuild.InternalsVisibleTo" Version="1.0.2">
<PrivateAssets>all</PrivateAssets>
<IncludeAssets>runtime; build; native; contentfiles; analyzers</IncludeAssets>
</PackageReference>
</ItemGroup>

在安装完成了 Meziantou.MSBuild.InternalsVisibleTo 库之后,可以使用下面代码让其他项目可见 internal 的类

<ItemGroup>
<InternalsVisibleTo Include="LeerijawarnikoJacallnachaykall"/>
<InternalsVisibleTo Include="HikallyijuDallcurjemdehowai"/>
</ItemGroup>

那么这个 NuGet 库的原理是什么?其实是通过 通过重写预定义的 Target 来扩展 MSBuild / Visual Studio 的编译过程 - walterlv 的方法,类似如下代码实现

<Target Name="AddInternalsVisibleTo" BeforeTargets="BeforeCompile">
<ItemGroup Condition="'@(InternalsVisibleTo->Count())' &gt; 0">
<AssemblyAttribute Include="System.Runtime.CompilerServices.InternalsVisibleTo">
<_Parameter1>%(InternalsVisibleTo.Identity)</_Parameter1>
</AssemblyAttribute>
</ItemGroup>
</Target>

也就是添加一个 AddInternalsVisibleTo 的 Target 在开始构建之前触发

内容就是读取 InternalsVisibleTo 项的内容,加入到 AssemblyAttribute 这里,而在上文小伙伴了解到放在 AssemblyAttribute 的内容将会被输出

用这个方法就可以做到写 InternalsVisibleTo 就可以自动创建 AssemblyInfo.cs 文件

如果需要在添加强签名的程序集添加 InternalsVisibleTo 请看 dotnet 强签名下使用 InternalsVisibleToAttribute 给程序集加上友元


知识共享许可协议

原文链接: http://blog.lindexi.com/post/dotnet-%E6%96%B0-SDK-Style-%E9%A1%B9%E7%9B%AE%E6%A0%BC%E5%BC%8F%E5%A6%82%E4%BD%95%E4%BD%BF%E7%94%A8-InternalsVisibleToAttribute-%E5%8A%9F%E8%83%BD

本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。 欢迎转载、使用、重新发布,但务必保留文章署名 林德熙 (包含链接: https://blog.lindexi.com ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请与我 联系