[.NET] 《Effective C#》快速笔记(三)- 使用 C# 表达设计
《Effective C#》快速笔记(三)- 使用 C# 表达设计
目录
- 二十一、限制类型的可见性
- 二十二、通过定义并实现接口替代继承
- 二十三、理解接口方法和虚方法的区别
- 二十四、用委托实现回调
- 二十五、用事件模式实现通知
- 二十六、避免返回对内部类对象的引用
- 二十七、让类型支持序列化
- 二十八、提供组粒度的因特网服务 API
- 二十九、支持泛型协变和逆变
二十一、限制类型的可见性
1.在保证类型可以完成工作的前提下,应该尽可能地给类型分配最小的可见性。
2.我们经常下意识的创建公有类型。可见性越低,以后升级或更改时所需要的变化就越小,因为能访问你功能模块的代码越少。
3.创建内部类是一种常被忽略限制类型作用域的做法,我们经常习惯不假思索地创建公有类。你应该仔细思考这个一个类型的作用范围,即它是将被所有的客户使用,还是仅用在这个程序集的内部。
4.更少的公有类型可以减少单元测试的数量。
5.以公有形式暴露给外界的类和接口将成为你的组件的契约。接口越冗余,日后的修改就越受限。暴露的公有类型越少,以后更新扩展的时候周旋的余地就会越大。
二十二、通过定义并实现接口替代继承
1.抽象基类为类的继承体系提供了一个共用的祖先,接口描述了一组原子性的功能。接口是一种契约,抽象基类则为一组相关的类型提供了一个共用的抽象。基类描述了对象是什么,接口描述了对象如何表现它的行为。
2.我们可以将可重用的行为提取出来,定义在接口中。由于不相关的类型均可以实现一个接口,这表示代码的重用率将大大增加。
3.如果向基类中添加一个方法,所有派生类都将自动包含该方法。也就是说,随着时间的推移,仍可以有效扩展多个类型功能的途径。通过向基类添加并实现某种功能,所有的派生类都将立即拥有该功能。而向接口中添加一个成员,会破坏所有实现该接口的类。因为这些类不包含新方法,每一个实现都需要进行更新,然后重新编译。
4.在抽象基类和接口之间做选择,实际上就表示了对日后可能发生变化的不同处理态度。接口是固定的:我们将一组功能封装在一个接口,作为其他类型的契约。而基类则可以在日后扩展,这些扩展也会成为每个派生类的一部分。
5.也可以使用扩展方法进行扩展。
public static class Extendsions
{
public static void ForAll<T>(this IEnumerable<T> sequence, Action<T> action)
{
foreach (var item in sequence)
{
action(item);
}
}
}
方法调用:
IEnumerable<object> objects = new List<string>();
objects.ForAll(x => Console.WriteLine(x));
objects.ForAll(Console.WriteLine);
6.有时候,使用接口可以帮助我们避免 struct 类型的拆箱所带来的代价。
二十三、理解接口方法和虚方法的区别
1.在基类中实现一个接口时,派生类需要使用 new 来隐藏对基类方法的使用。
2.可以将基类接口的方法申明为虚方法,然后在派生类中实现。
二十四、用委托实现回调
1.类之间需要通信时,并且我们期望一种比接口所提供的更为松散的耦合机制时,委托就是最佳的选择。
2.多播委托会把所有添加到该委托中的目标函数组合成一个单一的调用。需要注意的是:
(1)如果有委托调用出现异常,那么就不能保证安全;
(2)整个调用的返回值将为最后一个函数调用的返回值。
3.在多播委托调用的过程中,每个目标会被依次调用。委托对象本身不会捕捉任何异常。因此,任何目标抛出的异常都会结束委托链的调用。
二十五、用事件模式实现通知
1..NET 中的事件就是观察者模式的一个语法上的快捷实现。
2.使用 System.ComponentModel.EventHandlerList 容器来存储各个事件处理器,在类型中包含大量事件时可以使用他来隐藏所有事件的复杂性。
3.EventHandlerList并没有提供内建的泛型实现,可以自行基于 Dictionary 构建。
二十六、避免返回对内部类对象的引用
1.4 种策略可以防止类型的内部数据结构遭受有意或无意的修改:值类型、常量类型、接口和包装器(wrapper):
(1)当客户代码通过属性来访问值类型成员时,实际返回的是值类型的副本。
(2)常量类型,如:System.String 也是安全的。
(3)通过接口向外界暴露类的功能,即可尽量地避免内部数据遭受无意的更改,定义接口将访问限制在一个子集中从而最小化对对象内部状态的破坏。
(4)仅暴露包装器,定义一个包装器对象来限制另一个对象的访问。
2.通过使用接口、包装器对象或值类型向外界提供内部的私有数据,即可限制外界对这些数据的访问能力。
二十七、让类型支持序列化
1.[Serializable] 特性支持序列化。
2.[NonSerialized] 特性阻止序列化。
二十八、提供组粒度的因特网服务 API
1.程序中最耗时的是与远程服务器之间的数据传输,每次通过网络获取一小段数据时,应用程序都需要等待网络传输的过程,API 的粒度越细,所花费在等待数据返回上的额外时间也就越多。
2.我们希望同时降低通信的频率以及每次通信时所传递的数据量。但这两个目标往往不可兼得,因此必须做出取舍。尽量不要走两个极端,可以适当选择较少的通信次数,尽量一次传输更多的数据。
二十九、支持泛型协变和逆变
1.类型变体:协变和逆变。定义了在某种情况下,某个类型可以代替另一个类型进行使用。若不能将一个类型替换成另一个,则这个类型就叫不变量,在 C# 4.0 之前,所有泛型类型都不变量。你应该尽可能地让泛型接口和泛型委托支持协变和逆变。
2.协变和逆变是两种不同形式的类型替换。若某个返回的类型可以由派生类型替换,那么这个类型就是支持协变的。若某个参数类型可以由其基类替换,那么这个类型就是支持逆变的。
3.在可能的情况下为泛型接口和委托添加上 in 和 out 参数进行修饰。
4.因为 IList<T> 没有添加 in 或 out 修饰 T,所以必须使用精确的类型匹配。
本系列
《Effective C#》快速笔记(一)- C# 语言习惯
《Effective C#》快速笔记(二)- .NET 资源托管
《Effective C#》快速笔记(三)- 使用 C# 表达设计
【博主】反骨仔
【原文】http://www.cnblogs.com/liqingwen/p/6774210.html
【参考】《Effective C#》
[.NET] 《Effective C#》快速笔记(三)- 使用 C# 表达设计的更多相关文章
- Android Studio安卓学习笔记(三)Android用户界面的设计布局与组件(一)用户界面布局设计(1)
当我们创建了一个安卓项目后,我们会发现真正建立一个完善的安卓项目并不是想象的那么容易.其实和设计GUI可视化界面一样,开发安卓也需要考虑很多方面,主要考虑的还是界面布局和需要的组件. 一:Androi ...
- 《Effective C#》快速笔记(三)- 使用 C# 表达设计
目录 二十一.限制类型的可见性 二十二.通过定义并实现接口替代继承 二十三.理解接口方法和虚方法的区别 二十四.用委托实现回调 二十五.用事件模式实现通知 二十六.避免返回对内部类对象的引用 二十七. ...
- [.NET] 《Effective C#》快速笔记(四)- 使用框架
<Effective C#>快速笔记(四)- 使用框架 .NET 是一个类库,你了解的越多,自己需要编写的代码就越少. 目录 三十.使用重写而不是事件处理函数 三十一.使用 ICompar ...
- [.NET] 《Effective C#》快速笔记 - C# 中的动态编程
<Effective C#>快速笔记 - C# 中的动态编程 静态类型和动态类型各有所长,静态类型能够让编译器帮你找出更多的错误,因为编译器能够在编译时进行大部分的检查工作.C# 是一种静 ...
- [.NET] 《Effective C#》快速笔记 - C# 高效编程要点补充
<Effective C#>快速笔记 - C# 高效编程要点补充 目录 四十五.尽量减少装箱拆箱 四十六.为应用程序创建专门的异常类 四十七.使用强异常安全保证 四十八.尽量使用安全的代码 ...
- [.NET] 《Effective C#》快速笔记(二)- .NET 资源托管
<Effective C#>快速笔记(二)- .NET 资源托管 简介 续 <Effective C#>读书笔记(一)- C# 语言习惯. .NET 中,GC 会帮助我们管理内 ...
- [.NET] 《Effective C#》快速笔记(一)- C# 语言习惯
<Effective C#>快速笔记(一)- C# 语言习惯 目录 一.使用属性而不是可访问的数据成员 二.使用运行时常量(readonly)而不是编译时常量(const) 三.推荐使用 ...
- 《Effective C#》快速笔记(一)- C# 语言习惯
目录 一.使用属性而不是可访问的数据成员 二.使用运行时常量(readonly)而不是编译时常量(const) 三.推荐使用 is 或 as 操作符而不是强制类型转换 四.使用 Conditional ...
- 《Effective C#》快速笔记(六)- - C# 高效编程要点补充
目录 四十五.尽量减少装箱拆箱 四十六.为应用程序创建专门的异常类 四十七.使用强异常安全保证 四十八.尽量使用安全的代码 四十九.实现与 CLS 兼容的程序集 五十.实现小尺寸.高内聚的程序集 这是 ...
随机推荐
- 字符编码的种类:ASCII、GB2312、GBK、GB18030、Unicode、UTF-8、UTF-16、Base64
ASCII码ASCII:https://zh.wikipedia.org/wiki/ASCIIASCII(American Standard Code for Information Intercha ...
- C#"曾经的字符串数组"string[] array=new string[]{"**","****"};
写博客是一件很伟大的事情,尤其是也牛逼的博客,因为它能帮助需要的人,更能使自己对知识有一个更为深刻的理解! 欢迎关注我的博客! 字符串操作(取当前时间) string time=convert.tos ...
- (1)写给Web初学者的教案-----学习Web的知识架构
1:学习Web的知识架构 前文中我们简单的介绍了一些关于Web的基本知识,这里任老师再次强调一下凡是用浏览器打开的网站我们就称之为Web应用程序(B/S结构).除此之外其它需要下载安装的软件或是手机 ...
- dfs 无向图两节点间的所有路径
标题:风险度量 X星系的的防卫体系包含 n 个空间站.这 n 个空间站间有 m 条通信链路,构成通信网.两个空间站间可能直接通信,也可能通过其它空间站中转. 对于两个站点x和y (x != y), 如 ...
- 日新进用户200W+,解密《龙之谷》手游背后的压测故事
2017年3月,腾讯正式于全平台上线了<龙之谷>手游,次日冲到了App Store畅销排行第二的位置,并维持到了现在.上线当日百度指数超过40万,微信游戏平台数据显示预约数780多万,而据 ...
- 走入PHP-declare、ticks、encoding、include
declare 结构用来设定一段代码的执行指令.declare 的语法和其它流程控制结构相似(该代码为语法格式,不是代码案例,无需敲打该代码): declare (directive) stateme ...
- Spring:利用PerformanceMonitorInterceptor来协助应用性能优化
前段时间对公司产品做性能优化,如果单依赖于测试,进度就会很慢.所以就通过对代码的方式来完成,并以此来加快项目进度.具体的执行方案自然就是要知道各个业务执行时间,针对业务来进行优化. 因为项目中使用了S ...
- pacejs进度条监控服务端数据加载是否完毕
记得刚刚入职新公司的时候,公司在做app里面的h5页面.跟之前公司的流程不太一样.之前都是写完静态页面后通过ajax加载接口数据,这公司省事了,写完静态页面直接扔给服务端,他们来填数据,确实给前端省事 ...
- 基于均值漂移的三维网格分割算法(Mean Shift)
mean shift算法是一种强大的无参数离散数据点的聚类方法,其在图像平滑.图像分割以及目标跟踪等方面都有着广泛的应用.[Yamauchi et al. 2005]基于mean shift算法提出了 ...
- 机器学习:Python实现单层Rosenblatt感知器
如果对Rosenblatt感知器不了解,可以先查看下相关定义,然后对照下面的代码来理解. 代码中详细解释了各步骤的含义,有些涉及到了数学公式的解释. 这篇文章是以理解Rosenblatt感知器的原理为 ...