避免将多个类放在一个文件里面
一个文件应该只有一个命名空间避免将多个命名空间放在同一个文件里面
一个文件最好不要超过行的代码(不包括机器产生的代码)
一个方法的代码长度最好不要超过行
避免方法中有超过个参数的情况使用结构来传递多个参数
每行代码不要超过个字符
不要手工的修改机器产生的代码
a) 如果需要编辑机器产生的代码编辑格式和风格要符合该编码标准
b) 尽可能使用分部类来提出需要维护的部分
避免利用注释解释显而易见的代码
a) 代码应该可以自解释好的代码由可读的变量和方法命名因此不需要注释
避免使用方法级的文档
a) 使用扩展的API文档说明之
b) 只有在该方法需要被其他的开发者使用的时候才使用方法级的注释(在C#中就是///)
不要硬编码数字的值总是使用构造函数设定其值
只有是自然结构才能直接使用const比如一个星期的天数
避免在只读的变量上使用const如果想实现只读可以直接使用readonly
public class MyClass
{
public readonly int Number;
public MyClass(int someValue)
{
Number = someValue;
}
public const int DaysInWeek = ;
}
每个假设必须使用Assert检查
a) 平均每行要有一次检查(Assert)
using SystemDiagnostics;
object GetObject()
{…}
object obj = GetObject()
DebugAssert(obj != null)
代码的每一行都应该通过白盒方式的测试
只抛出已经显示处理的异常
在捕获(catch)语句的抛出异常子句中(throw)总是抛出原始异常维护原始错误的堆栈分配
catch(Exception exception)
{
MessageBoxShow(exceptionMessage)
throw ; //和throw exception一样
}
避免方法的返回值是错误代码
尽量避免定义自定义异常类
当需要定义自定义的异常时
a) 自定义异常要继承于ApplicationException
b) 提供自定义的序列化功能
避免在单个程序集里使用多个Main方法
只对外公布必要的操作其他的则为internal
避免友元程序集因为它能增加间装配耦合
避免代码依赖于从特定位置运行的程序集
使应用程序集尽量为最小化代码(EXE客户程序)使用类库来替换包含的商务逻辑
避免给枚举变量提供显式的值
//正确方法
public enum Color
{
RedGreenBlue
}
//避免
public enum Color
{
Red = Green = Blue =
}
避免指定特殊类型的枚举变量
//避免
public enum Color : long
{
RedGreenBlue
}
即使if语句只有一句也要将if语句的内容用大括号扩起来
避免使用三目条件操作符
避免在条件语句中调用返回bool值的函数可以使用局部变量并检查这些局部变量
bool IsEverythingOK()
{…}
//避免
if (IsEverythingOK ())
{…}
//替换方案
bool ok = IsEverythingOK()
if (ok)
{…}
总是使用基于开始的数组
在循环中总是显式的初始化引用类型的数组
public class MyClass
{}
MyClass[] array = new MyClass[];
for(int index = ; index < arrayLength; index++)
{
array[index] = new MyClass()
}
不要提供public 和 protected的成员变量使用属性代替他们
避免在继承中使用new而使用override替换
在不是sealed的类中总是将public 和 protected的方法标记成virtual的
除非使用interop(COM+ 或其他的dll)代码否则不要使用不安全的代码(unsafe code)
避免显示的转换使用as操作符进行兼容类型的转换
Dog dog = new GermanShepherd()
GermanShepherd shepherd = dog as GermanShepherd;
if (shepherd != null )
{…}
当类成员包括委托的时候
a) Copy a delegate to a local variable before publishing to avoid concurrency race
condition
b) 在调用委托之前一定要检查它是否为null
public class MySource
{
public event EventHandler MyEvent;
public void FireEvent()
{
EventHandler temp = MyEvent;
if(temp != null )
{
temp(thisEventArgsEmpty)
}
}
}
不要提供公共的事件成员变量使用事件访问器替换这些变量
public class MySource
{
MyDelegate m_SomeEvent ;
public event MyDelegate SomeEvent
{
add
{
m_SomeEvent += value;
}
remove
{
m_SomeEvent = value;
}
}
}
使用一个事件帮助类来公布事件的定义
总是使用接口
类和接口中的方法和属性至少为:的比例
避免一个接口中只有一个成员
尽量使每个接口中包含个成员
接口中的成员不应该超过个
a) 实际情况可能限制为个
避免接口成员中包含事件
避免使用抽象方法而使用接口替换
在类层次中显示接口
推荐使用显式的接口实现
从不假设一个类型兼容一个接口
SomeType obj;
IMyInterface obj;
/* 假设已有代码初始化过obj接下来 */
obj = obj as IMyInterface;
if (obj != null)
{
objMethod()
}
else
{
//处理错误
}
表现给最终用户的字符串不要使用硬编码而要使用资源文件替换之
不要硬编码可能更改的基于配置的字符串比如连接字符串
当需要构建长的字符串的时候使用StringBuilder不要使用string
避免在结构里面提供方法
a) 建议使用参数化构造函数
b) 可以重裁操作符
总是要给静态变量提供静态构造函数
能使用早期绑定就不要使用后期绑定
使用应用程序的日志和跟蹤
除非在不完全的switch语句中否则不要使用goto语句
在switch语句中总是要有default子句来显示信息(Assert)
int number = SomeMethod()
switch(number)
{
case :
TraceWriteLine(Case :)
break;
case :
TraceWriteLine(Case :)
break;
default :
DebugAssert(false)
break;
}
除非在构造函数中调用其他构造函数否则不要使用this指针
// 正确使用this的例子
public class MyClass
{
public MyClass(string message )
{}
public MyClass() : this(hello)
{}
}
除非你想重写子类中存在名称沖突的成员或者调用基类的构造函数否则不要使用base来访问基类的成员
// 正确使用base的例子
public class Dog
{
public Dog(string name)
{}
virtual public void Bark( int howLong)
{}
}
public class GermanShepherd : Dog
{
public GermanShe pherd(string name) base (name)
{}
override public void Bark(int howLong)
{
base Bark(howLong)
}
}
基于模板的时候要实现Dispose()和Finalize()两个方法
通常情况下避免有从SystemObject转换来和由SystemObject转换去的代码而使用强制转换或者as操作
符替换
class SomeClass
{}
//避免
class MyClass<T>
{
void SomeMethod(T t)
{
object temp = t;
SomeClass obj = (SomeClass)temp;
}
}
// 正确
class MyClass<T> where T : SomeClass
{
void SomeMethod(T t)
{
SomeClass obj = t;
}
}
在一般情况下不要定影有限制符的接口接口的限制级别通常可以用强类型来替换之
public class Customer
{…}
//避免
public interface IList<T> where T : Customer
{…}
//正确
public interface ICustomerList : IList<Customer>
{…}
不确定在接口内的具体方法的限制条件
总是选择使用C#内置(一般的generics)的数据结构