你应该在项目中使用部分类吗?

Arm*_*est 42 c# partial-classes

我有一个包含所有数据库逻辑的类库.我的DAL/BLL.

我有一些Web项目将使用相同的数据库和类,所以我认为将数据层抽象到自己的项目中是一个好主意.

但是,当为某些项目的类添加功能时,我想向某些类添加方法.

例如,我的数据层有Product和SomeItem对象:

// Data Access Layer project

namespace DAL {
  public class Product { 
     //implementation here 
  }

  public class SomeItem {
     //implementation here 
  }
}
Run Code Online (Sandbox Code Playgroud)

在一个项目中,我想添加一个由不同内容项使用的接口,所以我有一个名为的类:

// This is in Web Project
namespace DAL {
  public partial class Product : ICustomBehaviour {

    #region ICustomBehaviour Implementation
       TheSharedMethod();
    #endregion
  }
}
Run Code Online (Sandbox Code Playgroud)

使用相同的命名空间在单独的项目(创建依赖项)中编写部分类是一个好主意吗?如果这是一个坏主意,我怎样才能使这种类型的功能工作?

它似乎不想在编译时合并它们,所以我不确定我做错了什么.

Jon*_*eet 79

您不能跨项目编写部分类.部分类是一个仅编译时间的语法糖 - 整个类型最终在一个程序集中,即一个项目.

(顺便说一下,你的原始DAL文件也必须声明该类是部分的.)

  • @Atomiton:您不能从不同的项目中更改现有的类,不.您*可以*从现有类派生并实现接口.这有一些缺点 - 最值得注意的是,你必须确保你总是创建一个新类型的实例而不是基类的实例. (5认同)
  • @肯特:不是晚上八点,不是:) (2认同)
  • @Amir:整个集会并不是真的*.它只是在两个程序集中包含相同的源代码,这不是一回事. (2认同)

Jas*_*ker 5

我无法回答关于组织图层的最佳方法的问题,但我可以尝试回答有关如何最好地模拟部分类的问题.

以下是一些想法:

  • 首先想到的是继承.它不一定是最好的解决方案,但您可能没有选择,因为您可能需要能够像对象一样对待您的对象.
  • 组合也是一个不错的选择(也就是说,将类包装在另一个类中).这使您可以更好地与DAL分离,但实现起来可能很繁琐.
  • 如果您真的只需要在现有类中添加一个或两个方法,您可能还会考虑使用扩展方法,但如果您使用它们太多,这些方法可以快速创建意大利面条代码.


Kas*_*erg 5

使用Visual Studio 2015及更高版本可以在项目之间拆分部分类:使用共享项目(另请参见此MSDN博客)。

对于我的情况,我需要满足以下条件:

  • 一个类库,它定义了一些由多个客户端应用程序用作接口的类。
  • 客户端应用程序。
  • 安装程序在数据库中创建表。
    由于安装程序的限制,这个设置必须是自包含的。它不能引用.NET框架以外的任何程序集。该设置应在表中插入一些枚举常量,因此理想情况下应引用类库。
    安装程序可以导入共享项目。
  • 由于共享项目类似于复制粘贴代码,因此我想尽可能少地移入共享项目。

下面的例子演示部分类和共享项目如何允许在不同的项目分割类。

在类库中,Address.cs:

namespace SharedPartialCodeTryout.DataTypes
{
    public partial class Address
    {
        public Address(string name, int number, Direction dir)
        {
            this.Name = name;
            this.Number = number;
            this.Dir = dir;
        }

        public string Name { get; }
        public int Number { get; }
        public Direction Dir { get; }
    }
}
Run Code Online (Sandbox Code Playgroud)

类库是普通的Visual Studio类库。它导入SharedProject,除此之外,其.csproj不包含任何特殊内容:

<?xml version="1.0" encoding="utf-8"?>
<Project ToolsVersion="15.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
  <Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
  <PropertyGroup>
<!-- standard Visual Studio stuff removed -->
    <OutputType>Library</OutputType>
<!-- standard Visual Studio stuff removed -->
  </PropertyGroup>
<!-- standard Visual Studio stuff removed -->
  <ItemGroup>
    <Reference Include="System" />
  </ItemGroup>
  <ItemGroup>
    <Compile Include="Address.cs" />
    <Compile Include="Properties\AssemblyInfo.cs" />
  </ItemGroup>
  <Import Project="..\SharedProject\SharedProject.projitems" Label="Shared" />
  <Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
</Project>
Run Code Online (Sandbox Code Playgroud)

Address.Direction 在SharedProject中实现:

namespace SharedPartialCodeTryout.DataTypes
{
    public partial class Address
    {
        public enum Direction
        {
            NORTH,
            EAST,
            SOUTH,
            WEST
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

SharedProject.shproj是:

<?xml version="1.0" encoding="utf-8"?>
<Project ToolsVersion="15.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
  <PropertyGroup Label="Globals">
    <ProjectGuid>33b08987-4e14-48cb-ac3a-dacbb7814b0f</ProjectGuid>
    <MinimumVisualStudioVersion>14.0</MinimumVisualStudioVersion>
  </PropertyGroup>
  <Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
  <Import Project="$(MSBuildExtensionsPath32)\Microsoft\VisualStudio\v$(VisualStudioVersion)\CodeSharing\Microsoft.CodeSharing.Common.Default.props" />
  <Import Project="$(MSBuildExtensionsPath32)\Microsoft\VisualStudio\v$(VisualStudioVersion)\CodeSharing\Microsoft.CodeSharing.Common.props" />
  <PropertyGroup />
  <Import Project="SharedProject.projitems" Label="Shared" />
  <Import Project="$(MSBuildExtensionsPath32)\Microsoft\VisualStudio\v$(VisualStudioVersion)\CodeSharing\Microsoft.CodeSharing.CSharp.targets" />
</Project>
Run Code Online (Sandbox Code Playgroud)

其.projitems为:

<?xml version="1.0" encoding="utf-8"?>
<Project xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
  <PropertyGroup>
    <MSBuildAllProjects>$(MSBuildAllProjects);$(MSBuildThisFileFullPath)</MSBuildAllProjects>
    <HasSharedItems>true</HasSharedItems>
    <SharedGUID>33b08987-4e14-48cb-ac3a-dacbb7814b0f</SharedGUID>
  </PropertyGroup>
  <PropertyGroup Label="Configuration">
    <Import_RootNamespace>SharedProject</Import_RootNamespace>
  </PropertyGroup>
  <ItemGroup>
    <Compile Include="$(MSBuildThisFileDirectory)Address.Direction.cs" />
  </ItemGroup>
</Project>
Run Code Online (Sandbox Code Playgroud)

普通客户使用Address包括Address.Direction

using SharedPartialCodeTryout.DataTypes;
using System;

namespace SharedPartialCodeTryout.Client
{
    class Program
    {
        static void Main(string[] args)
        {
            // Create an Address
            Address op = new Address("Kasper", 5297879, Address.Direction.NORTH);
            // Use it
            Console.WriteLine($"Addr: ({op.Name}, {op.Number}, {op.Dir}");
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

常规客户端csproj引用类库而不是 SharedProject:

<?xml version="1.0" encoding="utf-8"?>
<Project ToolsVersion="15.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
  <Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
  <PropertyGroup>
<!-- Removed standard Visual Studio Exe project stuff -->
    <OutputType>Exe</OutputType>
<!-- Removed standard Visual Studio Exe project stuff -->
  </PropertyGroup>
<!-- Removed standard Visual Studio Exe project stuff -->
  <ItemGroup>
    <Reference Include="System" />
  </ItemGroup>
  <ItemGroup>
    <Compile Include="Program.cs" />
    <Compile Include="Properties\AssemblyInfo.cs" />
  </ItemGroup>
  <ItemGroup>
    <None Include="App.config" />
  </ItemGroup>
  <ItemGroup>
    <ProjectReference Include="..\SharedPartialCodeTryout.DataTypes\SharedPartialCodeTryout.DataTypes.csproj">
      <Project>{7383254d-bd80-4552-81f8-a723ce384198}</Project>
      <Name>SharedPartialCodeTryout.DataTypes</Name>
    </ProjectReference>
  </ItemGroup>
  <Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
</Project>
Run Code Online (Sandbox Code Playgroud)

DbSetup仅使用枚举:

DbSetup.csproj没有引用类库。它只导入SharedProject:

<?xml version="1.0" encoding="utf-8"?>
<Project ToolsVersion="15.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
  <Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
  <PropertyGroup>
<!-- Removed standard Visual Studio Exe project stuff -->
    <OutputType>Exe</OutputType>
<!-- Removed standard Visual Studio Exe project stuff -->
  <?PropertyGroup>
<!-- Removed standard Visual Studio Exe project stuff -->
  <ItemGroup>
    <Reference Include="System" />
    <Reference Include="Microsoft.CSharp" />
  </ItemGroup>
  <ItemGroup>
    <Compile Include="Program.cs" />
    <Compile Include="Properties\AssemblyInfo.cs" />
  </ItemGroup>
  <ItemGroup>
    <None Include="App.config" />
  </ItemGroup>
  <Import Project="..\SharedProject\SharedProject.projitems" Label="Shared" />
  <Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
</Project>
Run Code Online (Sandbox Code Playgroud)

结论:

您可以在项目之间划分局部类吗?

是的,使用Visual Studio的共享项目。

使用相同的名称空间在单独的项目中创建部分类(创建依赖项)是一个好主意吗?

通常不是(请参阅其他答案);在某些情况下,如果您知道自己在做什么,它会很方便。