C#综合揭秘——分部类和分部方法

简介:

面向对象的“封装闭合性”开发原则中,一向提倡的是把独立的功能封装在一个类里面的!但从Visual Studio 2005开发,系统提供了一个分部类的开发方式一直受到争议,很多人认为把同一类的功能分布不同的文件中,是打破了“封装闭合原则”,一个类的功能变得难 以管理,大多数人都是在无奈的情况下才使用到分部类的方式。但在winFrom类、页面类、DataSet里面你经常可以发现分部类的身影,当你用到 Entity Framework的时候,你会发现每个映射生成的对象都是使用分部类的方式生成的,分部类似乎早已派上用场。分部类究竟有什么好处,下面为大家一一揭 露。

一、分部类

根据微软的定义,分部类就是“将类或结构、接口或方法的定义拆分到两个或多个源文件中。 每 个源文件包含类型或方法定义的一部分,编译应用程序时将把所有部分组合起来”。在使用分部类的时候,必须为类加入partial的关键字,注意每个类的可 访问性必须一致,其中一类为public,其他类也必须为public。如果其中一个类为抽象类,那合并后整个类都将被视为抽象类,其中一个类为密封类, 那合并后整个类都将视为密封类。

 1 public partial class PersonManager
2 {
3 public Person GetPersonById(int id)
4 {
5 }
6 }
7
8 public partial class PersonManager
9 {
10 public List<Person> GetList()
11 {
12 }
13 }

在合并的时候,总体类全把所有的基类和特性合并继承。要注意的是分部类必须在于同一个程序集当中,而且关键字不得相冲,如果一个类上为public ,另一个类为private,那将会出错。在第二个分部类中可以调用第一个分部类中所定义的字段与方法。

 1 [SerializableAttribute]
2 public partial class Person { }
3
4 [ObsoleteAttribute]
5 public partial class Person { }
6
7 //合并后相当于
8 [SerializableAttribute]
9 [ObsoleteAttribute]
10 class Person{ }
11
12
13 partial class PersonManager:Manager{ }
14
15 partial class PersonManager:Collection{ }
16
17 //合并后相当于
18 class PersonManager:Manager,Collection{ }

二、分部方法

分部方法与分部类十分相像,方法必须包含partial关键字,如果一个类中并不包含该方法的实 现,而另一个类中实现了该方法,将不会影响这个方法的正常运行。这个特点跟接口有异曲同工之妙,特别是在使用代码生成器的时候,开发人员可以使用工具生成 分部方法,然后手动去实现方法。分部方法有以下几个限制,第一方法必须返回void,只能默认为private;第二,分部方法不能为virtual和 extern方法;第三,分部方法可以有ref参数,但不能有out参数;

1 partial void PersonChanged(Person person);
2
3 partial void PersonChanged(Person person)
4 {
5 PersonManager personManager=new PersonManager();
6 personManager.Update(person);
7 ......
8 }

关于分部类与分部方法,在微软的官方网站上有着详细介绍,在这里不多作说明。而下面在下想介绍一下分部类与分部方法的实际用途,这才是我写这章文件的真的目的。

三、分部类与分部方法的可用性

LINQ 是微软在Framewrok3.0开发推出的新产品,其中LINQ TO SQL是实现对象与数据表相映射的神奇工具。随着Framework 4.0面世,Entity Framework成为微软项目中实现ORM的主要手段,当中*.edmx文件中使用的都是分部类的实现方式。这是因为映射过程是自动生成的,代码必须符 合定制的规则,当需要为对象添加一些额外的属性,而这些属性无需保存到数据库的时候,分部类就派上用场,我们可以使用分部类为对象提供各种的自定义属性。

特别是在使用DDD领域驱动设计的时候,分部类成为实现模型动作的一个好方法。失血模型与充血模型 是DDD长久争议的话题,在失血模型中,模型是不应该具备动作,而是把动作放在Service层中,而在充血模型中,模型类应该具有各自的方法,而“分部 类”就是实现充血模型方法的一种好手段。

  1 //Model.Designer.cs文件
2 [global::System.Data.Objects.DataClasses.EdmEntityTypeAttribute(NamespaceName="BusinessModel", Name="Approve")]
3 [global::System.Runtime.Serialization.DataContractAttribute(IsReference=true)]
4 [global::System.Serializable()]
5 public partial class Approve : global::System.Data.Objects.DataClasses.EntityObject
6 {
7 /// <summary>
8 /// 创建新的 Approve 对象。
9 /// </summary>
10 /// <param name="id">ID 的初始值。</param>
11 /// <param name="functionType">FunctionType 的初始值。</param>
12 [global::System.CodeDom.Compiler.GeneratedCode("System.Data.Entity.Design.EntityClassGenerator", "4.0.0.0")]
13 public static Approve CreateApprove(int id, int functionType)
14 {
15 Approve approve = new Approve();
16 approve.ID = id;
17 approve.FunctionType = functionType;
18 return approve;
19 }
20 /// <summary>
21 /// 架构中不存在属性 ID 的任何注释。
22 /// </summary>
23 [global::System.Data.Objects.DataClasses.EdmScalarPropertyAttribute(EntityKeyProperty=true, IsNullable=false)]
24 [global::System.Runtime.Serialization.DataMemberAttribute()]
25 [global::System.CodeDom.Compiler.GeneratedCode("System.Data.Entity.Design.EntityClassGenerator", "4.0.0.0")]
26 public int ID
27 {
28 get
29 {
30 return this._ID;
31 }
32 set
33 {
34 this.OnIDChanging(value);
35 this.ReportPropertyChanging("ID");
36 this._ID = global::System.Data.Objects.DataClasses.StructuralObject.SetValidValue(value);
37 this.ReportPropertyChanged("ID");
38 this.OnIDChanged();
39 }
40 }
41 [global::System.CodeDom.Compiler.GeneratedCode("System.Data.Entity.Design.EntityClassGenerator", "4.0.0.0")]
42 private int _ID;
43 [global::System.CodeDom.Compiler.GeneratedCode("System.Data.Entity.Design.EntityClassGenerator", "4.0.0.0")]
44 partial void OnIDChanging(int value);
45 [global::System.CodeDom.Compiler.GeneratedCode("System.Data.Entity.Design.EntityClassGenerator", "4.0.0.0")]
46 partial void OnIDChanged();
47 /// <summary>
48 /// 架构中不存在属性 FunctionType 的任何注释。
49 /// </summary>
50 [global::System.Data.Objects.DataClasses.EdmScalarPropertyAttribute(IsNullable=false)]
51 [global::System.Runtime.Serialization.DataMemberAttribute()]
52 [global::System.CodeDom.Compiler.GeneratedCode("System.Data.Entity.Design.EntityClassGenerator", "4.0.0.0")]
53 public int FunctionType
54 {
55 get
56 {
57 return this._FunctionType;
58 }
59 set
60 {
61 this.OnFunctionTypeChanging(value);
62 this.ReportPropertyChanging("FunctionType");
63 this._FunctionType = global::System.Data.Objects.DataClasses.StructuralObject.SetValidValue(value);
64 this.ReportPropertyChanged("FunctionType");
65 this.OnFunctionTypeChanged();
66 }
67 }
68 [global::System.CodeDom.Compiler.GeneratedCode("System.Data.Entity.Design.EntityClassGenerator", "4.0.0.0")]
69 private int _FunctionType;
70 [global::System.CodeDom.Compiler.GeneratedCode("System.Data.Entity.Design.EntityClassGenerator", "4.0.0.0")]
71 partial void OnFunctionTypeChanging(int value);
72 [global::System.CodeDom.Compiler.GeneratedCode("System.Data.Entity.Design.EntityClassGenerator", "4.0.0.0")]
73 partial void OnFunctionTypeChanged();
74 /// <summary>
75 /// 架构中不存在属性 Title 的任何注释。
76 /// </summary>
77 [global::System.Data.Objects.DataClasses.EdmScalarPropertyAttribute()]
78 [global::System.Runtime.Serialization.DataMemberAttribute()]
79 [global::System.CodeDom.Compiler.GeneratedCode("System.Data.Entity.Design.EntityClassGenerator", "4.0.0.0")]
80 public string Title
81 {
82 get
83 {
84 return this._Title;
85 }
86 set
87 {
88 this.OnTitleChanging(value);
89 this.ReportPropertyChanging("Title");
90 this._Title = global::System.Data.Objects.DataClasses.StructuralObject.SetValidValue(value, true);
91 this.ReportPropertyChanged("Title");
92 this.OnTitleChanged();
93 }
94 }
95 ...............................
96 }
97
98 //分部类
99 public partial class Approve
100 {
101 //添加属性
102 public string Type
103 {
104 get;set;
105 }
106
107 //添加动作
108 public void AddReport(Report report)
109 {.......}
110 .................
111 }

在下也是在使用分部类对Entity Framework模型进行开发的时候才注意到分部类, 文章的目的主要是想介绍分部类在Entity Framework开发过程中的作用,敬请点评。

对JAVA与.NET开发有兴趣的朋友欢迎加入QQ群:162338858

点击这里加入此群



本文转自 leslies2  51CTO博客,原文链接:http://blog.51cto.com/79100812/663273

相关文章
|
6天前
|
开发框架 .NET 程序员
C# 去掉字符串最后一个字符的 4 种方法
在实际业务中,我们经常会遇到在循环中拼接字符串的场景,循环结束之后拼接得到的字符串的最后一个字符往往需要去掉,看看 C# 提供了哪4种方法可以高效去掉字符串的最后一个字符
|
1月前
|
存储 C# 索引
C# 一分钟浅谈:数组与集合类的基本操作
【9月更文挑战第1天】本文详细介绍了C#中数组和集合类的基本操作,包括创建、访问、遍历及常见问题的解决方法。数组适用于固定长度的数据存储,而集合类如`List<T>`则提供了动态扩展的能力。文章通过示例代码展示了如何处理索引越界、数组长度不可变及集合容量不足等问题,并提供了解决方案。掌握这些基础知识可使程序更加高效和清晰。
61 2
|
24天前
|
C#
C#一分钟浅谈:Lambda 表达式和匿名方法
本文详细介绍了C#编程中的Lambda表达式与匿名方法,两者均可用于定义无名函数,使代码更简洁易维护。文章通过基础概念讲解和示例对比,展示了各自语法特点,如Lambda表达式的`(parameters) =&gt; expression`形式及匿名方法的`delegate(parameters)`结构。并通过实例演示了两者的应用差异,强调了在使用Lambda时应注意闭包问题及其解决策略,推荐优先使用Lambda表达式以增强代码可读性。
26 8
|
1月前
|
C# 数据安全/隐私保护
C# 一分钟浅谈:类与对象的概念理解
【9月更文挑战第2天】本文从零开始详细介绍了C#中的类与对象概念。类作为一种自定义数据类型,定义了对象的属性和方法;对象则是类的实例,拥有独立的状态。通过具体代码示例,如定义 `Person` 类及其实例化过程,帮助读者更好地理解和应用这两个核心概念。此外,还总结了常见的问题及解决方法,为编写高质量的面向对象程序奠定基础。
16 2
|
2月前
|
图形学 C# 开发者
全面掌握Unity游戏开发核心技术:C#脚本编程从入门到精通——详解生命周期方法、事件处理与面向对象设计,助你打造高效稳定的互动娱乐体验
【8月更文挑战第31天】Unity 是一款强大的游戏开发平台,支持多种编程语言,其中 C# 最为常用。本文介绍 C# 在 Unity 中的应用,涵盖脚本生命周期、常用函数、事件处理及面向对象编程等核心概念。通过具体示例,展示如何编写有效的 C# 脚本,包括 Start、Update 和 LateUpdate 等生命周期方法,以及碰撞检测和类继承等高级技巧,帮助开发者掌握 Unity 脚本编程基础,提升游戏开发效率。
40 0
|
2月前
|
Java C# 索引
C# 面向对象编程(一)——类
C# 面向对象编程(一)——类
29 0
|
2月前
|
开发框架 .NET 编译器
C# 中的记录(record)类型和类(class)类型对比总结
C# 中的记录(record)类型和类(class)类型对比总结
|
2月前
|
C#
C# async await 异步执行方法
C# async await 异步执行方法
41 0
|
5月前
|
开发框架 前端开发 .NET
C#编程与Web开发
【4月更文挑战第21天】本文探讨了C#在Web开发中的应用,包括使用ASP.NET框架、MVC模式、Web API和Entity Framework。C#作为.NET框架的主要语言,结合这些工具,能创建动态、高效的Web应用。实际案例涉及企业级应用、电子商务和社交媒体平台。尽管面临竞争和挑战,但C#在Web开发领域的前景将持续拓展。
168 3
|
5月前
|
SQL 开发框架 安全
C#编程与多线程处理
【4月更文挑战第21天】探索C#多线程处理,提升程序性能与响应性。了解C#中的Thread、Task类及Async/Await关键字,掌握线程同步与安全,实践并发计算、网络服务及UI优化。跟随未来发展趋势,利用C#打造高效应用。
181 3
下一篇
无影云桌面