Fork me on GitHub

代码的印象派:写点好代码吧

最近有一位猎头顾问打电话询问是否有换工作的意向,对推荐的公司和职位的描述为:"我们这里有一家非常关注软件质量的公司,在寻找一位不仅能完成有挑战的软件开发任务,并且还对代码质量有非常高追求的软件工程师。"。

很难得猎头顾问会以这样的切入点来推荐职位,而不是诸如 "我们是互联网公司","我们是著名互联网公司","我们可以提供业内领先的薪 资","我们创业的方向是行业的趋势","我们提供创业公司期权","我们提供人体工程座椅","我们公司漂亮妹子多" 等等。

谁会为了把椅子或者每天能看公司漂亮的前台妹子而跳槽呢?将这些描述可以概括成一点,就是 "我们给的钱多"。诚然,好的薪水是可以招募到杰出的软件工程师的。然而,优秀的软件工程师通常已经得到了较好的薪水,所以如果不是给出足够的量,不一定会为五斗米而折腰。大多数的软件工程师更看重的是技术兴趣,所以诸如 "来我们这做 Openstack 吧","我们需要用 Go 实现 Docker 相关组件的人才" 看起来更有吸引力。

而软件质量,代码风格,则是另一个吸引优秀工程师的方向。追求卓越软件质量的工程师,通常有着自己的软件设计与实现思路,并在不断的实践中锤炼出自己的编程风格和代码品味。这一类工程师,其实无关使用什么语言、做什么产品,他们会始终保持自己的品味,追求软件实现的卓越,进而产出高质量的软件。更重要的是,优秀的工程师希望与更多优秀的工程师合作,并更愿意工作在崇尚代码质量的氛围中。

一般条件下,对软件质量的要求通常与软件生命周期的长短相关。按照软件生命周期的长短,我们可以将软件公司分为两类:

  • 快公司:创业公司,互联网公司。推崇快速开发,快速试错。软件生命周期较短,代码质量相对要求不高。
  • 慢公司:传统行业软件公司,产品公司。推崇稳定可靠的软件设计。软件生命周期较长,代码质量相对要求较高。

软件生命周期的长短,通常也决定了实现软件所使用的编程语言。比如,快公司通常会使用 PHP/Python/Ruby 等动态类型语言,而慢公司通常会选择 C/C++/C#/Java 等静态类型语言。

当猎头顾问还没有说出公司的名字之前,我们也可以大体猜测出该公司所属的行业方向或软件产品类型。比如,该公司可能来自传统的金融、电信、医疗、石 油、ERP 行业等,这些行业中有 IBM、Huawei、GE、Schlumberger、SAP 等等世界 500 强巨头更重视软件质量。当然,通常推崇软件质量的公司中,大概率条件下碰到的会是外企,即使是中小外企,也会对软件质量有相对较高的要求。对于产品类型, 越是接近 Mission Critical 的产品形态则对软件质量的要求越高。各种软件中间件的软件质量要求也相对较高,比如内存数据库、Message Queue 等。或者如果区分 Framework 和 Application,则 Framework 的软件质量显然要求更高。

每一家较成熟的软件公司,都会设计自己的软件编码规范,增强软件工程师的协同效应,相互之间可以读懂对方的代码。但实际操作中,又有多少人会执着遵 守呢?编码规范并不能保证产生出好代码,那代码编写的好坏又如何具体衡量呢?笔者经历过的公司中,多半都是以软件发布后的 Bug 数量来衡量软件的质量的,这种统计形式简单粗暴,优点就是可以量化,缺点就是很难评判出软件代码编写的优雅程度。我听过一则笑话,说软件质量也不能做的太 好,软件一定要有 Bug,这样客户才会买我们的服务,而我们就是靠后期卖服务赚钱的。好吧,情何以堪~~

好了,说了这么多,好像文不切题,这篇文章不是叫《代码的印象派》吗?和上面说的这些有什么关系呀?

关系就在于,软件质量与代码编写的优雅程度息息相关。而是否能编写出优雅的代码与软件工程师的个人风格和品味息息相关。

在软件工程领域,通常生命周期长的软件会有着更高的软件质量需求,描述软件质量的内容可以参考下面两篇文章。

在各种软件质量模型的描述中,都包含着软件可维护性(Maintainability)这一属性。而越是生命周期长的软件,对其软件可维护性的要求 越高。而提高软件可维护性的根本方式就是编写可阅读的代码,让其他人理解代码的时间最小化。代码生来就是为人阅读的,只是顺便在机器上执行以完成功能。

在漫长的软件生命周期中,我们有很多机会去修改软件代码,比如发现了新的 Bug、增加新的功能、改进已有功能等。修改代码的第一步当然就是阅读代码,以了解当前的设计和思路。如果代码都读不懂的话,何谈修改呢?还有,大概率条 件下,修复自己实现模块的 Bug 的人通常就是你自己,如果时隔几个月后自己都读不懂自己编写的代码,会是什么感受呢?

所以,如何编写出易读的代码就成了问题的关键。而能否编写出易读代码,则直接取决于软件工程师自己的的编程风格和代码品味。

在《孙子兵法》中有云:"上兵伐谋,其次伐交,其次伐兵,其下攻城。攻城之法,为不得已。" 对应到软件领域,软件架构师可以通过出色的系统分析来构建可演进的软件架构,讲究谋略;而软件工程师则通过良好的设计和编程风格来完成攻城任务,讲究方法。

Paul Graham 的《黑客与画家》中描述了黑客与画家的共同点,就是他们都是创作者,并且都在努力创作优秀的作品。画家创作的作品就是画,内嵌着自己的风格和品味。软件工程师的作品就是软件和代码,如果可以的话,你可以将代码打印成卷,出版成书,只是,阅读的人会向你那样幸福吗?

画家的作品都会保留下来,如果你把一个画家的作品按照时间顺序排列,就会发现每幅画所用的技巧,都是建立在上一幅作品学到的东西之上。某幅作品如果 特别出众,你往往能在更早期的作品中找到类似的版本。软件工程师也是通过实践来学些编程,并且所进行的工作也是具有原创性的,通常不会有他人的完美的成果 可以依靠,如果有的话我们为什么还要再造轮子呢?

创作者的另一个学习途径是通过范例。对于画家而言,临摹大师的作品一直是传统美术教育的一部分,因为临摹迫使你仔细观察一幅画是如何完成的。软件工 程师也可以通过学习优秀的程序源码来学会编程,不是看其执行结果,而是看源码实现思路和风格。优秀的软件一定都是在软件工程师对软件美的不懈追求中实现 的,现如今有众多优秀的开源软件存在,如果你查看优秀软件的内部,就会发现,即使在那些不被人知的部分,也同样被优美的实现着。

所以说,代码是有画面感的,看一段代码就可以了解一个软件工程师的风格,进而塑造出该工程师在你心目中的印象。工作中,我们每天都在阅读同事们的代 码,进而对不同的同事产生不同的印象,对各种不同印象的感受也在不断影响着自身风格的塑造。代码的印象派,说的就是,你想让你的同事对你产生何种印象呢?

笔者不能自诩为我就是那类有着良好的编程风格,并且代码品味高雅的软件工程师,只能说,我还在向这个目标努力着。风格和品味不是一朝一夕就能养成 的,世间存在多少种风格我们也无法列举,而说某种风格比另一种风格要好也会陷入无意的争辩。况且,软件工程师多少都会有点自恋情节,在没有见到更好的代码 之前,始终都会感觉自己写出的代码就是好代码,并且有时不管你说什么,咱就是这个味儿!

我个人总结了几点关于优雅代码风格的描述:

  • 代码简单:不隐藏设计者的意图,抽象干净利落,控制语句直截了当。
  • 接口清晰:类型接口表现力直白,字面表达含义,API 相互呼应以增强可测试性。
  • 依赖项少:依赖关系越少越好,依赖少证明内聚程度高,低耦合利于自动测试,便于重构。
  • 没有重复:重复代码意味着某些概念或想法没有在代码中良好的体现,及时重构消除重复。
  • 战术分层:代码分层清晰,隔离明确,减少间接依赖,划清名空间,理清目录。
  • 性能最优:局部代码性能调至最优,减少后期因性能问题修改代码的机会。
  • 自动测试:测试与产品代码同等重要,自动测试覆盖 80% 的代码,剩余 20% 选择性测试。

下面,我会列举一些我在工作中遇到的不同的编程风格,用切身的体会来感悟代码的风格和品味。当然,吐槽为主,因为据说在 Code Review 时记录说 "我擦" 的数量就可以衡量代码的好坏。

变量

关于变量,很遗憾,不得不提变量的命名。时至今日,在 Code Review 中仍然可以看到下面这样的代码。

复制代码
 1   public class Job
 2   {
 3     private DateTime StartTime;
 4     private DateTime mStartTime;
 5     private DateTime m_StartTime;
 6     private DateTime _StartTime;
 7     public DateTime endTime;
 8     private Command pCommand;
 9     private long schId;
10   }
复制代码

有各种奇葩的前缀出现,有时同一个人的命名居然也不统一。虽然,眼睛和大脑在重 复的观察变量名后会自动学习以忽略前缀,并不会太影响阅读。实际上,使用前缀的目的主要是为了在局部代码中区分全局变量和局部变量。使用类似于 C# 这样的高级语言,我们已经不再需要为变量添加前缀了,可以利用 this 关键字来区分。如果非要添加的话,建议使用 "_" 单下划线前缀,促进大脑更快速的忽略。

复制代码
 1   public class Job
 2   {
 3     private DateTime _startTime; // use _ as prefix
 4     private DateTime endTime; // no prefix
 5 
 6     public DateTime StartTime
 7     {
 8       get { return _startTime; }
 9       set { _startTime = value; }
10     }
11 
12     public DateTime EndTime
13     {
14       get { return endTime; }
15       set { endTime = value; }
16     }
17 
18     public long ScheduleId { get; private set; } // or no field needed
19   }
复制代码

将 Field 标记为 public 应该是没有分清 Field 与 Property 的作用,进而推测对面向对象编程中的封装概念理解也不会有多好。

使用 "p" 前缀的显然有 C/C++ 编程情节,想描述这个变量是一个指针,好吧,这种写法在 C# 中只能称为不伦不类。

使用缩写,这里的 "sch" 其实是想代表 "schedule",但在没有上下文的条件下谁能想的出来呢?我个人是绝对不推荐使用缩写的,除非是普世的理解性存在,例如 "obj", "ctx", "pwd", "ex", "args" 这样非常常见的缩写。

使用拼音和有拼写错误的单词作为变量名会直接拉低工程师的档次。使用合适单词描述可以直接提高代码的质量,比如通常 "Begin", "End" 会成对儿出现,上面的示例代码中涉及到了时间,"StartTime" 和 "BeginTime" 是同义词,所以我们参考了 Outlook Calendar 中的默认术语,也就是 "StartTime" 和 "EndTime",也就是找范例。

在局部变量的使用中,我认为有一种使用方式是值得推荐的,那就是 "解释性变量"。当今的编程风格中流行使用 Fluent API,这样会产生类似于下面这样的代码。

复制代码
1       if(DateTimeOffset.UtcNow >= 
2         period
3           .RecurrenceRange.StartDate.ConvertTime(period.TimeZone)
4           .Add(schedule.Period.StartTime.ConvertTime(period.TimeZone).TimeOfDay))
5       {
6         // do something
7       }
复制代码

这一串 "." 看着好帅气,但我是理解不了这是要比较什么。可以简单重构为解释性变量。

复制代码
1       var firstOccurrenceStartTime = 
2         period
3           .RecurrenceRange.StartDate.ConvertTime(period.TimeZone)
4           .Add(schedule.Period.StartTime.ConvertTime(period.TimeZone).TimeOfDay);
5 
6       if(DateTimeOffset.UtcNow >= firstOccurrenceStartTime)
7       {
8         // do something
9       }
复制代码

构造函数

很多工程师还没有理解好构造函数的功效和使用方式,在选择依赖注入方式时,更倾向于使用属性依赖注入。个人认为,使用 "属性依赖注入" 是懒惰的一种表现,其不仅打破了信息隐藏的封装,而且还可以暴露了本不需要暴露的部分。使用构造函数进行依赖注入是最正确的方式,我们应该竭尽全力将代码 重构到这一点。

好的,你说的,我信了!并且,我也开始这么做了!绝对纯净的构造函数注入!

复制代码
 1   public class Schedule
 2   {
 3     public Schedule(long templateId, long seriesId,
 4       long promotionId, bool isOnceSche, DateTime startTime, DateTime endTime,
 5       List<TimeRange> blackOutList, ScheduleAddtionalConfig addtionalConfig,
 6       IDateTimeProvider tProvider, IScheduleMessageProxy proxy,
 7       IAppSetting appSetting, RevisionData revisionData)
 8     {
 9     }
10   }
复制代码

好吧,你赢了!构造函数居然有 12 个参数,距离史上最长的构造函数不远了。

一般写成这样的代码已经表示没法看了,而且注定类的设计也不怎么样,这要是遗留下来的 Legacy Code,不知道维护者心情几何?

还有一类构造函数问题就是参数顺序,这直接体现了软件工程师最终他人的基本素养。因为构造函数生来就是为使用者准备的,而为使用者设计合理的参数顺序是类设计者的基本职责。

复制代码
 1   public class Command
 2   {
 3     public Command(int argsA, int argsB)
 4     {
 5     }
 6 
 7     public Command(int argsC, int argsB, int argsA)
 8     {
 9     }
10   }
复制代码

上面这种反人类思维的参数顺序,怎么描述呢?写成下面这样有多大难度?

复制代码
 1   public class Command
 2   {
 3     public Command(int argsA, int argsB)
 4     {
 5     }
 6 
 7     public Command(int argsA, int argsB, int argsC)
 8     {
 9     }
10   }
复制代码

属性

蹩脚的属性设计常常彰显抽象对象类型的能力。以下面这个 Schedule 类为例,Schedule 业务上存在 Once 和 Recurring 两种状态。我们最初看到的类是这个样子的。

复制代码
1   public class Schedule
2   {
3     public Schedule(bool isOnceSchedule)
4     {
5       IsOnceSchedule = isOnceSchedule;
6     }
7 
8     public bool IsOnceSchedule { get; set; }
9   }
复制代码

看来这是想通过构造函数直接注入指定状态,但 IsOnceSchedule 属性的 set 又是 public 的允许修改,不仅暴露了封装,还没有起到隐藏的效果!

那么,稍微改进下,试图消灭 IsOnceSchedule 属性,引入继承机制。

复制代码
 1   public class Schedule
 2   {
 3   }
 4 
 5   public class OnceSchedule : Schedule
 6   {
 7   }
 8 
 9   public class RecurringSchedule : Schedule
10   {
11   }
复制代码

实现上在 OnceSchedule 和 RecurringSchedule 中均封装独立的实现。如果非要通过父类抽象暴露 Recurring 状态,可以在父类中通过属性暴露只读接口。

复制代码
 1   public class Schedule
 2   {
 3     public Schedule()
 4     {
 5       this.IsRecurring = false;
 6     }
 7 
 8     public bool IsRecurring { get; protected set; }
 9   }
10 
11   public class OnceSchedule : Schedule
12   {
13     public OnceSchedule()
14       : base()
15     {
16       this.IsRecurring = false;
17     }
18   }
19 
20   public class RecurringSchedule : Schedule
21   {
22     public RecurringSchedule()
23       : base()
24     {
25       this.IsRecurring = true;
26     }
27   }
复制代码

函数

我们或许都知道,函数命名要动词开头,如需要可与名词结合。而函数设计的要求是尽量只做一件事,这件事有时简单有时困难。

简单的可以像下面这种一句话代码:

1     internal bool CheckDateTimeWeekendDay(DateTimeOffset dateTime)
2     {
3       return dateTime.DayOfWeek == DayOfWeek.Saturday 
4         || dateTime.DayOfWeek == DayOfWeek.Sunday;
5     }

复杂的见到几百行的函数也不新奇。拆解长函数的方法有很多,这么不做赘述。这里推荐一种使用 C# 语法糖衍生出的函数设计方法。

上面的小函数其实是非常过程化的代码,其是为类 DateTimeOffset 服务,我们可以使用 C# 中的扩展方法来优化这个小函数。

复制代码
1   internal static class DateTimeOffsetExtensions
2   {
3     internal static bool IsWeekendDay(this DateTimeOffset dateTime)
4     {
5       return dateTime.DayOfWeek == DayOfWeek.Saturday
6         || dateTime.DayOfWeek == DayOfWeek.Sunday;
7     }
8   }
复制代码

这样,我们就可以像下面这样使用了,感觉会不会好一些?

1       if(DateTimeOffset.Now.IsWeekendDay())
2       {
3         // do something
4       }

在设计函数时,我们时常犹豫的是,到底应该返回一个 null 值还是应该抛出一个异常呢?

答案就是,如果你总是期待函数返回一个值时,而值不存在则应该抛出异常;如果你期待函数可以返回一个不存在的值,则可以返回 null。总之,不要因为懒惰而使得应该设计抛出异常的函数最终返回了 null,不幸的是,这种懒惰经常出现。

正常的代码是不需要 try..catch.. 的,异常就应该一抛到底直至应用程序崩溃,当然,这是开发阶段。一抛到底有利于发现已有代码路径中 的错误,毕竟异常在正常逻辑中是不应该产生的。我们要做的是,合理期待某调用可能会产生某类异常,则直接 catch 该特定异常,如 catch (System.IO.FileNotFoundException ex)。

实际上,遇到这种抉择场景,我们可以在函数命名上下些功夫,以变相解决问题。

1     object FindObjectOrNull(string key);
2     object FindObjectOrThrow(string key);
3     object FindObjectOrCreate(string key, object dataNeededToCreateNewObject);
4     object FindObjectOrDefault(string key, Object defaultReturnValue); 

单元测试

在开始写代码的时候就开始考虑测试问题,有利于产生易于测试的代码。幸运的是,对测试友好的设计会很自然的产生良好的代码。

测试驱动开发(TDD)是一种编程风格,包含 TDD 三定律:

  1. 在编写不能通过的单元测试前,不能编写生产代码;
  2. 只编写刚好无法通过的单元测试,不能编译不算通过;
  3. 只编写刚好通过当前失败测试的生产代码;

我们显然可以循规蹈矩的遵循上述 TDD 三定律风格编程,但 TDD 只是通过测试来保证代码质量,驱动良好设计的一种风格,我们没有必要完全强迫自己遵循上述定律,找到适合自己的过程可能效率更高,所以重点在于,要写单元 测试,通过写代码时思考测试这件事来帮助把代码写的更好。

测试代码不是二等公民,它和生产代码一样重要。他需要被思考、被设计、被维护,并且要像生产代码一样保持优雅的风格。

单元测试测什么?

在单元测试中,可通过两种方式来验证代码是否正确地工作。一种是基于结果状态的测试,一种是基于交互行为的测试。这两种方式在文章《单元测试的两种方式》中有描述,这里就不再赘述。

单元测试的可读性

在测试代码中,可读性仍然很重要。如果测试代码的可读性良好,使其更易于后期的维护和修改,不至于是测试代码腐化以致被删除。

下面是一些良好测试的关键点:

  • 测试越简明越好,每个测试只关注一个点。
  • 如果测试运行失败,则其应发出有帮助性的错误消息或提示。
  • 使用简单明确的测试输入条件。
  • 给测试用例取一个可描述的名字。

那么,具体什么样的单元测试用例名称,算是好名称呢?这里推荐两种:

  • 第一种:使用 Test_<ClassName>_<FunctionName>_<Situation> 风格;
  • 第二种:使用 Given_<State>_When_<Behavior>_Then_<SomethingHappen> 风格;

第二种实际上是 BDD 风格,其不仅可以应用于单元测试,在更高级的 Component LevelSystem Level 的测试中同样有效。

实际上,单元测试用例代码的内部实现也是有风格可遵循,常见的就是 Arrange-Act-Assert (AAA) 模式。

第三方组件代码不便于测试

在文章《类依赖项的不透明性和透明性》中描述了依赖项对单元测试的影响,实践中,我们碰到最多的是调用其他类库的代码而导致的不可测试性。

复制代码
 1   public class MyClass
 2   {
 3     private Job _job;
 4 
 5     public MyClass()
 6     {
 7       _job = new Job();
 8     }
 9 
10     public void ExecuteJob()
11     {
12       _job.Execute();
13     }
14   }
15 
16   public sealed class Job
17   {
18     public void Execute()
19     {
20       // do something heavy
21     }
22   }
复制代码

上面的代码,如果写一个 TestCase 的话,可能是下面这种情况。

复制代码
1   [Test]
2   public void Test_MyClass_ExecuteJob()
3   {
4     MyClass instance = new MyClass();
5     instance.ExecuteJob();
6 
7     // what should we assert?
8   } 
复制代码

这样,调用了 instance.ExecuteJob() 导致了不知道如何验证。同时,由于 Job 类使用了 sealed 关键字,并且没有实现任何接口,所以也无法通过 mocking 库来 mock。

解决办法,增加中间层。

复制代码
 1   public class MyClass
 2   {
 3     private IJob _job;
 4 
 5     public MyClass(IJob job)
 6     {
 7       _job = job;
 8     }
 9 
10     public void ExecuteJob()
11     {
12       _job.Execute();
13     }
14   }
15 
16   public class JobProxy : IJob
17   {
18     private Job _realJob;
19 
20     public JobProxy(Job job)
21     {
22       _realJob = job;
23     }
24 
25     public void Execute()
26     {
27       _realJob.Execute();
28     }
29   }
30 
31   public interface IJob
32   {
33     void Execute();
34   }
35 
36   // third-party Job Class
37   public sealed class Job
38   {
39     public void Execute()
40     {
41       // do something heavy
42     }
43   }
复制代码

这样,我们在测试 MyClass 类时,就可以通过 IJob 接口注入 Mock 对象。这里选用的 Mocking Library 是 NSubstitute,参考《NSubstitute完全手册索引》。

复制代码
 1   [Test]
 2   public void Test_MyClass_ExecuteJob()
 3   {
 4     IJob job = Substitute.For<IJob>();
 5 
 6     MyClass instance = new MyClass(job);
 7     instance.ExecuteJob();
 8 
 9     // assert
10     job.Received(1).Execute();
11   }
复制代码

依赖时间的测试

还有一种较难测试的代码是依赖于时间的代码。比如,我们有一个依赖于时间的 Trigger 类,简写为这个样子。

复制代码
 1   public class Trigger
 2   {
 3     public Trigger(DateTime triggeredTime)
 4     {
 5       this.TriggeredTime = triggeredTime;
 6     }
 7 
 8     public DateTime TriggeredTime { get; private set; }
 9 
10     public bool TryExecute()
11     {
12       if (DateTime.Now >= TriggeredTime)
13       {
14         // do something
15         return true;
16       }
17 
18       return false;
19     }
20   }
复制代码

测试时,我可能会挑一些特定时间进行测试,特定时间有可能在很远的未来。

复制代码
 1   [Test]
 2   public void Test_Trigger_TryExecute_AfterTriggeredTime()
 3   {
 4     DateTime triggeredTimeInFuture = 
 5       new DateTime(2016, 2, 29, 8, 0, 0, DateTimeKind.Local);
 6 
 7     Trigger trigger = new Trigger(triggeredTimeInFuture);
 8     bool result = trigger.TryExecute();
 9 
10     // assert
11     Assert.IsTrue(result);
12   }
复制代码

好吧,这个 TestCase 应该是到 2016 年才能执行成功,显然不是我们期待的。改进的办法还是增加中间层,增加 IClock 接口用于提供时间。

复制代码
 1   public class Trigger
 2   {
 3     private IClock _clock;
 4 
 5     public Trigger(IClock clock, DateTime triggeredTime)
 6     {
 7       _clock = clock;
 8       this.TriggeredTime = triggeredTime;
 9     }
10 
11     public DateTime TriggeredTime { get; private set; }
12 
13     public bool TryExecute()
14     {
15       if (_clock.Now() >= TriggeredTime)
16       {
17         // do something
18         return true;
19       }
20 
21       return false;
22     }
23   }
24 
25   public interface IClock
26   {
27     Func<DateTimeOffset> UtcNow { get; }
28     Func<DateTimeOffset> Now { get; }
29   }
30 
31   public class Clock : IClock
32   {
33     public Func<DateTimeOffset> UtcNow { get { return () => DateTimeOffset.UtcNow; } }
34     public Func<DateTimeOffset> Now { get { return () => DateTimeOffset.Now; } }
35   }
复制代码

这样,我们就可以在 TestCase 代码中使用 Mocking 类库来替换 IClock 的实例,进而指定时间。

复制代码
 1   [Test]
 2   public void Test_Trigger_TryExecute_AfterTriggeredTime()
 3   {
 4     IClock clock = Substitute.For<IClock>();
 5 
 6     clock.Now
 7       .Returns<Func<DateTimeOffset>>(() =>
 8       {
 9         return DateTimeOffset.Parse(
10           "2016-02-29T08:00:01.0000000", CultureInfo.CurrentCulture);
11       });
12 
13     DateTime triggeredTimeInFuture =
14       new DateTime(2016, 2, 29, 8, 0, 0, DateTimeKind.Local);
15 
16     Trigger trigger = new Trigger(clock, triggeredTimeInFuture);
17     bool result = trigger.TryExecute();
18 
19     // assert
20     Assert.IsTrue(result);
21   }
复制代码

本篇文章《代码的印象派》由 Dennis Gao 原创发表自博客园个人博客,未经作者本人同意禁止以任何的形式转载,任何自动的或人为的爬虫转载行为均为耍流氓。

133
3
(请您对文章做出评价)
« 上一篇:K-Means 聚类算法
posted @ 2015-05-04 16:22 Dennis Gao 阅读(10303) 评论(86) 编辑 收藏

#51楼 2015-05-05 18:32 netfocus  

写的很好,很有共鸣。

#52楼 2015-05-05 23:14 田园里的蟋蟀  

很不错!

#53楼[楼主] 2015-05-06 09:27 Dennis Gao  

@crazyair
@蝼蚁
@倚天照海- -
感谢各位的支持~~

#54楼[楼主] 2015-05-06 09:27 Dennis Gao  

@netfocus
@田园里的蟋蟀
感谢支持,能有所共鸣更好~~

#55楼 2015-05-06 09:56 microtry  

@ChuckLu

引用
我想知道这2种写法有什么区别?
前提是Number属性的get和set,仅仅是return和赋值,不涉及到逻辑的。

我会觉得第一种写法更简单。第二种方法,在get和set没有额外的逻辑处理的时候,感觉太臃肿了。字段少的时候,还可以接受,但是字段多了,每个字段都弄一个属性的话,太麻烦了吧

我的回复只针对C#:
1.微软明确表示:当class内部完全不依赖于字段时,可以直接字段公开,而没有必要属性,但是这样做并不能获得额外的性能优势
2.属性访问器会自动生成get_属性名()和set_属性名()两个函数,
使用过C++或者JS等早期OOPL的人大概都有这样的经历,要写很多的get/set,属性访问器只不过是语法糖而已,
当你确定绝不可能set/get隔离的时候,那就不隔离呗,
而另一种情况是当get/set的逻辑比较复杂的时候,还不如自己写get/set,更加方便

#56楼 2015-05-06 10:29 ChuckLu  

@microtry

引用 @ChuckLu
引用引用
我想知道这2种写法有什么区别?
前提是Number属性的get和set,仅仅是return和赋值,不涉及到逻辑的。

我会觉得第一种写法更简单。第二种方法,在get和set没有额外的逻辑处理的时候,感觉太臃肿了。字段少的时候,还可以接受,但是字段多了,每个字段都弄一个属性的话,太麻烦了吧

我的回复只针对C#:
1.微软明确表示:当class内部完全不依赖于字段时,可以直接字段公开,而没有必要属性,但是这样做并不能获得额外的性能优势
2.属性访问器会自动生成get_属性名()和set_属性名()两个函数,
使用过C++或者JS等早期O...

额,第一点能理解,不指望性能优势,只是希望简化编码。因为在字段很多,比如10个字段,如果不public的话。每个字段加属性,还要get和set。太麻烦了。[当然前提还是根本没有get和set的逻辑]
get和set逻辑复杂的时候?我不是很理解。
我如果需要用get和set的话,必然是需要自己写逻辑的。
不会傻瓜式的
get{return 字段;}
set{字段 = value;}
如果不涉及逻辑,单纯写这个,貌似太傻瓜了。

记得现在的编译器,可以简写的直接 get;set;就可以

#57楼 2015-05-06 10:34 妖居  

我好像看到你们新的job manager的代码

#58楼[楼主] 2015-05-06 10:44 Dennis Gao  

@妖居

引用 我好像看到你们新的job manager的代码

对啊,文章本身就是实战出来的,我们在实现新的 Job Scheduler 时遇到了不少问题。
咱不带这么爆料自黑的哦~~ 我们写的很好的,哈哈哈~~~

#59楼[楼主] 2015-05-06 10:48 Dennis Gao  

@妖居

引用 我好像看到你们新的job manager的代码

园龄:10年1个月
哥,别吓我,您老是特意上来为我捧场的吗~~

#60楼 2015-05-06 11:06 火星老蒋  

@ChuckLu
简化输入的话,在vs中prop+tab,基本上敲击键盘次数与public xxx yyyy比起来是差不多的,甚至前者更有一种因莫名节奏而带来的输入畅快感;

另外还有一个小的不值得一提的事,由于属性实际被编译为get和set方法,作为类库对外暴露的接口是可替换的,比如原来你的类库A里有个
public class A
{
public int Threshold;
public void Run()
{
}
}

调用的项目BCDEF...Z代码是:
var a = new A();
a.Threshold= 50000;
a.Run();
...
var a = new A();
a.Threshold= 50;
a.Run();

那么一旦Threshold由于某些不可预料的问题必须改为属性的时候,比如以下方式:

private int _threshold;
public int Threshold
{
get
{
return _threshold;
}
set
{
_threshold = value > 1000 ? value : 1000;
}
}

所有客户端项目BCDEF...Z都是必须重新编译的。

反之如果一开始该字段就是属性,那么无论get set怎么改实现,至少直接替换dll是可以跑起来的。

#61楼 2015-05-06 13:11 ChuckLu  

@火星老蒋

引用 @ChuckLu
简化输入的话,在vs中prop+tab,基本上敲击键盘次数与public xxx yyyy比起来是差不多的,甚至前者更有一种因莫名节奏而带来的输入畅快感;

另外还有一个小的不值得一提的事,由于属性实际被编译为get和set方法,作为类库对外暴露的接口是可替换的,比如原来你的类库A里有个
public class A
{
public int Threshold;
public void Run()
{
}
}

调用的项目BCDEF...Z代码是:
var a = new A();
a.Threshold= 50000;
a.Run(...

我说的是确保不会有get和set的扩展,这个是前提。

另外,就你说的情况。看不出字段替换dll为什么不能跑起来。
class A
{
public int Number;
}
class A
{
private int number;
public int Number
{
get;
set;
}
}
这2中写法,一个是字段,一个是属性。即便是你从第一种方式,改变成第二种方式。
对于调用方来说,是没有区别的。

#62楼 2015-05-06 13:28 wdwwtzy  

我好像看到你们新的job manager的代码

#63楼 2015-05-06 14:01 火星老蒋  

@ChuckLu
第一点咱们没法预测未来不是吗,所有的public字段都要考虑一下是否以后有可能修改才选用Property或者Field不是太累了吗?
第二点看不出来的话新建几个项目试试就知道了,MissingFieldException。

#64楼 2015-05-06 14:22 ChuckLu  

@火星老蒋

引用 @ChuckLu
第一点咱们没法预测未来不是吗,所有的public字段都要考虑一下是否以后有可能修改才选用Property或者Field不是太累了吗?
第二点看不出来的话新建几个项目试试就知道了,MissingFieldException。

ok,我测试了下,确实会有字段缺少的异常。
在不考虑扩展的情况下,完全没必要加属性的,直接写字段就ok了。

#65楼 2015-05-06 14:31 沈赟  

thanks

#66楼 2015-05-06 16:28 AK47  

博主的总结,深有感触

#67楼 2015-05-06 16:43 张小驴同学  

点个赞

#68楼 2015-05-06 16:43 冲杀  

@天罡

引用博主,我是第一个点推荐的,这篇上了最多推荐&&有这么多评论,我也想发骚一下。
我的感觉是,往往是最开始写架构时的代码很优秀,后来随着时间的推移,维护的人可能换过N个,各自根据各自需求与风格进行了修改,最后导致了代码的四不像。除非来个大牛做大的重构,否则只能越来越糟了。可往往大牛看着不爽就直接重写了。
最后,问个隐私问题。楼主是我崇拜的,我多想达到博主的高度。能告诉下您的薪资范围吗,我也好有个奋斗目标。
年薪:
A:30W-50W
B:50W-70W
C:70W-100W
D:100W以上


其 实你说的优秀设计的框架,最后千疮百孔,个人总结的是:99%的程序员是以解决手上的问题为前提,也就是说,他们没只是关注东西做出来,而不是先看了当前 代码的框架设计后,才开始动手!!!!不是框架的问题。归根结底还是人的问题。有的人可以很贴切的根据当前项目的框架来写代码,但是有的人,直接横冲直 撞,选择自己会的方法去解决问题(其实很多时候是这些人理解不到设计者的意图或者本身水平就不够又或者不愿意去思考)。反正我是看透了!!!搞烂就搞烂, 只要按时交付了就OK,毕竟我不是甲方,对那些乱搞的程序员,你说了反而还得罪人!!!!因为老板只管你什么时候拿出来,至于质量,大家见过多少老板会对 代码质量关心的,见过的举手

#69楼 2015-05-06 17:21 ta_wuhen  

构造注入,参数多。有好的解决方式?

#70楼[楼主] 2015-05-06 17:45 Dennis Gao  

@ta_wuhen

引用构造注入,参数多。有好的解决方式?

我在之前的评论中提到了些解决方案。

首先,要优先选择使用构造函数注入依赖。

如果参数数量超过 4-5 个,建议考虑,是不是类设计的有问题,真的要有这么多的外部依赖吗?
好,考虑后发现确实需要这么多依赖,那么是否可以增加一个中间层,将这些外部依赖打包到一起呢?他们之间是否有这么的联系,在某些场景下就是一起工作的。那么你就可以简化成一个依赖。
好,还不行,不能打包,那解决办法就是通过引入 Factory 模式,通过将所有依赖注入到类型的工厂中,通过工厂构造对象,这样,从你的代码中将构造该对象的实例的过程隔离出去。

如何?

#71楼 2015-05-06 18:01 shark_c  

我觉得构造函数注入不是绝对,属性注入更加灵活

#72楼 2015-05-06 18:02 ta_wuhen  

@Dennis Gao
嗯,我试试。3q

#73楼[楼主] 2015-05-06 18:25 Dennis Gao  

@shark_c

引用 我觉得构造函数注入不是绝对,属性注入更加灵活

灵活真不好说,属性注入的要求是在类API中暴露属性,而如果注入的依赖不应当向外暴露,则直接破坏了类的设计。

一般我只会在处理循环引用问题时才使用属性注入。

#74楼 2015-05-06 20:39 前端小王子  

深圳JAVA技术交流群 281463795, 不错,哈哈哈哈

#75楼 2015-05-06 21:21 Adming  


我觉得这样也还可以啊!

#76楼 2015-05-06 22:40 马非码  

总结得不错,即使是C#,不同类型的变量加个前缀还是很有必要的,当你的代码复杂了,能一眼就看到不同的信息,而不是鼠标放上去才知道,会省事很多

#77楼 2015-05-07 08:50 何镇汐  

非常感谢楼主的Substitute系列,我之前一直在使用Rhino Mocks,用得非常痛苦,看了楼主的Substitute系列,果断采用,不仅测试代码更简单,而且解决了之前的很多棘手问题,比如Rhino Mocks不允许覆盖测试设置
楼主为.NET敏捷开发贡献卓越

#78楼[楼主] 2015-05-07 11:22 Dennis Gao  

@何镇汐

引用非常感谢楼主的Substitute系列,我之前一直在使用Rhino Mocks,用得非常痛苦,看了楼主的Substitute系列,果断采用,不仅测试代码更简单,而且解决了之前的很多棘手问题,比如Rhino Mocks不允许覆盖测试设置
楼主为.NET敏捷开发贡献卓越

为识货者点赞!

#79楼 2015-05-07 13:44 前端小王子  

java技术群281463795, 大神解决开发问题,不错,哈哈哈哈

#80楼[楼主] 2015-05-07 21:28 Dennis Gao  

@Adming
这写的还真整齐,也可以赞了~

#81楼 2015-05-12 14:11 xumenger  

本 人大学生,最近在开发一个项目,也知道需要考虑异常处理、考虑项目的健壮性,但是因为经验较少,所以虽然在开发的过程中一直在注意这些问题、也通过一些途 径去尽可能的了解应该怎么规范的开发。结果最后开发出来的项目虽然可以实现需求中的功能,但是明显感觉有太多的漏洞、不足。所以我对这件事情真的很困惑

#82楼[楼主] 2015-05-12 14:39 Dennis Gao  

@xumenger

引用 本人大学生,最近在开发一个项目,也知道需要考虑异常处理、考虑项目的健壮性,但是因为经验较少,所以虽然在开发的过程中一直在注意这些问题、也通过一些 途径去尽可能的了解应该怎么规范的开发。结果最后开发出来的项目虽然可以实现需求中的功能,但是明显感觉有太多的漏洞、不足。所以我对这件事情真的很困惑

时间长就好了,经验和教训都是积累出来的,要不何来成长?

#83楼 2015-05-13 17:11 杨盛超  

@冲杀
这 个看公司项目性质吧,如果是给客户用的,确实BUG少就行了,至于代码质量没人关注,交付给客户不出问题就完事了。但是如果做出来是给自己公司用的产品 线,而且周期很长,经历了很多年,那就不一样了,如果代码写得很烂,后期维护及新需求开发很痛苦,这样代码质量的作用就体现出来了。一个项目做很长时间不 断改不断有新需求而项目不会发生软件危机这就体现出水平了。

#84楼 2015-05-13 22:54 瘸腿狼  

DRY很重要
以后工作中也要注意测试代码的编写啊!

#85楼 2015-05-22 21:23 ArvinZhang  

博主写的很好。写代码一定要养成良好的风格,这个风格可以在后期的学习中不断完善。用良好的编码风格,才能有利于后期的维护,并且最好在必要的时候加上详细的注释。

#86楼[楼主] 2015-05-25 22:40 Dennis Gao  

@ArvinZhang
@瘸腿狼
@杨盛超

感谢各位的支持~~

发表评论

昵称:

评论内容:
引用 粗体 链接 缩进 代码 图片

注销 订阅评论

[使用Ctrl+Enter键快速提交]