- 在线时间
- 31 小时
- 最后登录
- 2014-1-6
- 注册时间
- 2012-11-27
- 听众数
- 11
- 收听数
- 0
- 能力
- 0 分
- 体力
- 391 点
- 威望
- 0 点
- 阅读权限
- 30
- 积分
- 151
- 相册
- 1
- 日志
- 1
- 记录
- 2
- 帖子
- 78
- 主题
- 10
- 精华
- 0
- 分享
- 1
- 好友
- 13
升级   25.5% TA的每日心情 | 擦汗 2014-1-6 09:21 |
|---|
签到天数: 29 天 [LV.4]偶尔看看III
- 自我介绍
- calm down
 群组: web前端开发交流 |
# M, R9 J8 o% [作者: Paul Firth 来源: 外刊IT评论
# o& ]: u2 i+ m3 ?, Z4 }
& B6 V+ H g7 e- K 英文原文:10 steps to becoming a better programmer# t! F6 D. y: J
这篇文章要介绍的,是我作为专业程序员这些年来学到的能真正提高我的代码质量和整体工作效率的 10 件事情。
' }2 b3 T! ~! a( v: p 1. 永远不要复制代码* ]& e/ H; s0 Q- U) f w
不惜任何代价避免重复的代码。如果一个常用的代码片段出现在了程序中的几个不同地方,重构它,把它放到一个自己的函数里。重复的代码会导致你的同事在读你的代码时产生困惑。而重复的代码如果在一个地方修改,在另外一个地方忘记修改,就会产生到处是 bug,它还会使你的代码体积变得臃肿。现代的编程语言提供了很好的方法来解决这些问题,例如,下面这个问题在以前很难解决,而如今使用 lambda 却很好实现:
# W# F" h! t0 t6 P7 Y/ Y/ S/// <summary>0 A5 }9 }8 C( G& H9 | z# Y) x9 n) b# \# n
/// 一些函数含有部分重复代码
- m1 F* O2 g: ^4 }% G$ G/ X/// </summary>8 |% i8 j" X k0 ~+ H0 n
void OriginalA()( v7 X8 v l! J6 O$ u# ~ B# D
{
8 b- P2 l+ \& v5 `) _/ m- N DoThingsA();
) P" f. `: Q+ B% \ // unique code
# |8 Z3 y. e- L DoThingsB();
+ e9 e- a" K$ H) b1 ~: ]& E}
3 B" {& l8 L: I5 ]) n/// <summary>+ s( n; r/ T7 N5 t0 B
/// 另外一个含有部分重复代码的函数3 W. @. `# J3 q, ~
/// </summary>9 G* c9 }/ e; M$ b0 {+ n. f
void OriginalB()! k5 w) [7 ~5 V/ F# l2 }$ y, z
{* w% H" F+ Y: B9 ~# x+ t9 V+ e
DoThingsA();8 F5 D( l- ~$ h$ o4 X- f
// 没有重复的代码
! Y. Z; i, D7 y DoThingsB();& Q6 l4 M1 D$ ]: y2 {1 j: ^
}
" ?3 I: Y+ k% q% x8 L 现在我们重构含有部分相同代码的函数,用 delegate 模式重写它们:
! b4 C. W% p F2 B; |) B$ ~/ f/// <summary>
0 l% H2 Q. P9 @, R& R& ?) ]7 h+ R/// Encapsulate shared functionality# F& e$ L" K; E/ x# d+ b9 P2 y
/// </summary>
* d( n, E# n8 J$ E& ?2 Q7 D/// <param name="action">User defined action</param>
- K m; @ P. Rvoid UniqueWrapper(Action action)
$ `3 t# X0 q( E% B{6 G, ~* J. T; e2 E7 B
DoThingsA();
, K4 L$ v1 g9 h action();# E" d3 r8 O2 N4 o" n) l
DoThingsB();
5 U% e& T+ T! }0 p1 M4 A}* b' O0 L8 ^; H7 ?" v! N7 k/ ^ c9 j0 v
/// <summary>
& ]. \! t8 t4 a4 R/// New implmentation of A9 }8 d! ]' w/ h
/// </summary>
# z; {5 Q3 V# ^2 S9 ~( M, {void NewA()
0 T7 Q0 {- C% `9 h0 o" u5 ~+ r{
9 I1 P- f9 m: q6 X UniqueWrapper(() =>
1 u8 p$ F! t/ W! N. D% L$ C {
% O8 u9 U% I, K) D% U2 V: P6 K // unique code/ G# x2 k( M, T0 h& a! l, {1 w7 q+ D" X
});7 K4 N1 R3 w. R) i& T: ?* F% k
}
) W8 i6 ]" k6 L/// <summary>
" n- K, ]3 P+ E9 W) f9 G4 v8 A/// New implementation of B; t( L8 `* j3 U/ K+ P
/// </summary>( n# f& j$ w4 z) ]/ x5 e
void NewB()
" V! ?" {3 |+ i5 a- g1 ^{( \$ k. j' b* S( B; S j c- v
UniqueWrapper(() =>
) e4 K) w& a, o1 x {
& {! D, k$ I5 d" M // unique code
: [. ]6 N7 N2 J+ C });
4 Z/ H, t" z, ]! n1 \}
+ q0 k. _" \" f; B' k 2. 留意你开始分心的时候
5 m/ b4 ?: y \$ C 当你发现自己在浏览 facebook 或微博,而不是在解决问题,这通常是一种你需要短暂休息的信号。离开办公桌,去喝一杯咖啡,或去跟同事聊 5 分钟。尽管这样做看起来有点反直觉,但长久去看,它会提高你的工作效率。
7 b- ~. r- f& Q- H# Q 3. 不要匆忙赶任务而放弃原则
2 w. I5 R1 z, Z4 r1 w3 c, ~ 当带着压力去解决一个问题或修改一个 bug,你很容易失去自制,发现自己匆匆忙忙,甚至完全忘了一直坚持的重要的测试过程。这通常会导致更多的问题,会让你在老板或同事眼里显得很不专业。1 p8 n& _/ @2 p1 n" A7 @* e7 Z
4. 测试你完成的代码
- J5 [0 m' J5 t' D; F5 v- @) d 你知道你的代码能做什么,而且试了一下,它确实好用,但你实际上需要充分的验证它。分析所有可能的边界情况,测试在所有可能的条件下它都能如期的工作。如果有参数,传递一些预期范围外的值。传递一个 null 值。如果可能,让同事看看你的代码,问他们能否弄坏它。单元测试是到达这种目的的常规方法。1 _& Z2 b* Q8 C8 M
5. 代码审查
% D. b+ U0 V5 P5 z' b 提交你的代码之前,找个同事一起坐下来,向他解释你做了哪些修改。通常,这样做的过程中你就能发现代码中的错误,而不需要同事说一句话。这比自己审查自己的代码要有效的多得多。
' @7 V- U; Y {2 b7 O 6. 让代码更少
. n6 p, @0 u) x, @ 如果你发现写了大量的代码来解决一个简单的问题,你很可能做错了。下面的 boolean 用法是一个很好的例子:
: \3 l3 v8 v- z. ^3 _if (numMines > 0)
/ F, o: ~3 l* ^- }{. s. r) @: }3 M2 { U
enabled=true;% X4 j! d1 F7 o# m! t9 b& M
}1 e5 O7 b! g& V' |/ \- o
else% s* J+ F. ?; D( R6 D
{& r" ?6 y1 Y% H* S; x
enabled=false;
0 N+ k, [3 S( H. I& w2 _( x}
: G( N+ j( d! o4 ?9 i+ c5 J 这时你应该写成这样:& g7 ]' e1 ^( q* ?
enabled = numMines > 0;
. H N2 ^9 K8 C( V' L 代码越少越好。这会使 bug 更少,重构可能性更小,出错的几率更小。要适度。可读性同等重要,你可不能这样做而使代码丧失可读性。
; h$ r/ `) ?0 ^) \ 7. 为优雅的代码而努力3 J4 X3 h& S& Q
优雅的代码非常的易读,只用手边很少的代码、让机器做很少的运算就能解决问题。在各种环境中都做到代码优雅是很难的,但经过一段时间的编程,你会对优雅的代码是个什么样子有个初步的感觉。优雅的代码不会通过重构来获得。当你看到优雅的代码是会很高兴。你会为它自豪。例如,下面就是一个我认为是优雅的方式来计算多边形面积的方法:
0 ]" x7 K& a# D, I- astatic public double GetConvexPolygonArea (Vector2[] vertices)" K6 u1 C0 N {- P
{2 b: X4 N9 Y! R0 ~+ s( i
double area = 0;
. I7 B+ T: V; q, v for (int i = 0; i < vertices.Length; i++)
" c, }8 s6 x/ N& j3 ?# Z- F: X {
, o# |% l% M* k* P3 m Vector2 P0 = vertices[i];
. J0 L4 q4 e5 c Vector2 P1 = vertices[(i + 1) % vertices.Length];* S2 v4 R5 Q) K y/ t
area += P0.Wedge (P1);. Y9 y9 F+ ]6 \1 C- w
}
1 [5 d4 {* v% f) a7 N" ` return area / 2;
% _8 ]% W' _, G( D}
! J( c1 X; A5 f% C- P1 R$ q 8. 编写不言自明的代码4 V0 m( g- R- S& T, h* C
勿庸置疑,注释是编程中很重要的一部分,但能够不言自明的代码更胜一筹,因为它能让你在看代码时就能理解它。函数名变量名要慎重选择,好的变量/方法名字放到语言语义环境中时,不懂编程的人都能看懂。例如:
; z- c& w8 R2 f7 lvoid DamagePlayer (Player player, int damageAmount)
' Z6 v( D# x5 g/ `{2 u& r8 ^# @2 F/ C1 M, z4 _
if (!player.m_IsInvincible && !player.m_IsDead)
) U! g. g' H2 S, T/ v {/ H& u* s z+ ~/ O
player.InflictDamage ( damageAmount );
2 e) N R% d3 H$ K6 }, l }3 L* g# `0 P9 O N: G4 a1 Q
} 4 ]3 ]/ O6 ^* ^7 H
能自我说明的代码不能代替注释。注释是用来解释“为什么”的,而自我说明的代码是来描述“是什么”的。4 E, E! q( o9 o; H" M- ~
9. 不要使用纯数字8 G4 v& C$ G. H0 |7 Q2 \
直接把数字嵌入代码中是一种恶习,因为无法说明它们是代表什么的。当有重复时更糟糕——相同的数字在代码的多个地方出现。如果只修改了一个,而忘记了其它的。这就导致 bug。一定要用一个命名常量来代表你要表达的数字,即使它在代码里只出现一次。
' p! J. T; R( N& `; { 10. 不要做手工劳动
% Y! Y4 ?/ T3 t2 ^7 g% y/ [; N 当做一系列动作时,人类总是喜欢犯错误。如果你在做部署工作,并且不是一步能完成的,那你就是在做错事。尽量的让工作能自动化的完成,减少人为错误。当做工作量很大的任务时,这尤其重要。3 l2 Z# U$ _8 v/ F x( b
11. 避免过早优化
! h7 S0 V0 {0 s: `: j 当你要去优化一个已经好用的功能代码时,你很有可能会改坏它。优化只能发生在有性能分析报告指示需要优化的时候,通常是在一个项目开发的最后阶段。性能分析之前的优化活动纯属浪费时间,并且会导致 bug 出现。
- G( T( n- X4 v7 }. I9 Z- o! t 好吧,我说是 10 个,但你却得到了额外赠送的一个!
6 x. u8 z' s' D 这些就是我要说的,我希望它们能帮助你改进编程开发过程。
4 v0 t u, i( Q8 c6 W 下次再见!祝快乐!. {0 _( Q7 Y+ b/ E9 d7 C# B
Cheers, Paul. |
zan
|