重载与重写 | 静态构造函数 | 只读代理 | 同步代理
资源管理 | 构造函数中的虚函数 | 强制针对接口编程 | 抗变与协变
重载与重写问题
日常讨论中,术语的不统一带来些许混乱
惯用的表达
overload 重载 函数名称相同,参数不同(严格的定义还有其它一些限制) 静态决议 override 重写(覆写,覆盖,改写) 子类重新定义父类定义过的虚函数(个别语言允许返回值,访问级别可以不同) 动态决议示例
class Base {
}
class Derived:Base {
}
class Client {
void Test(Base obj){
Console.WriteLine("base");
}
void Test(Derived obj){
Console.WriteLine("derived");
}
static void Main(string[] args) {
Base obj = new Derived();
new Client().Test(obj); //输出“base”
}
}
静态构造函数问题
1,在工具类中,通常有一些初始化需要在任何静态方法被调用前进行,如配置信息的读取
2,普通类中的复杂的静态信息,需要在任何实例方法被调用前初始化
我见过的解决方法
1,在每个静态方法中都调用必需的初始化步骤public class SomeUtilClass {
private SomeUtilClass(){
}
private static void Init(){
//....
}
public static string GetUID(){
Init();
return uid;
}
public static string GetConnectionString(){
Init();
return connString;
}
}
2,在普通构造函数中初始化public class SomeMapperClass{
private static Hashtable types;
public SomeMapperClass(){
if(types == null){
types = new Hashtable();
types.Add("RED", Color.Red);
types.Add("GREEN", Color.Green);
types.Add("BLUE", Color.Blue);
}
}
public Color GetColor(string color){
return (Color)types[color];
}
}
我推荐的解决方法
使用静态构造函数(C#),或静态初始化块(Java)
[C#]
public class SomeClass {
static SomeClass(){
Init();
types = new Hashtable();
types.Add(...);
types.Add(...);
}
}
[Java]
public class SomeClass {
static{
Init();
types = new HashMap();
types.put("", "");
types.put("", "");
}
}
效果
1,Once,only once
2,定义中对异常处理等有要求,可参考规范
2,多线程时是否有问题,我不清楚,讨论一下
只读代理问题对象内部有一个集合,由这个对象来控制其元素的增加删除,但客户需要访问该集合取得自己想要的信息,而对象不可能为所有的客户都提供对应的方法,因此需要返回内部的这个集合,但不允许客户增加或删除其元素
我见过的解决方法
直接返回代表集合的成员引用,仅在文档中要求客户不能增删集合中的元素public class SomeClass {
private List attrs;
public List GetAttributes(){
return attrs;
}
}
我推荐的解决方法
1,首选语言提供的功能
2,次选类库提供的功能
3,自己包装代理类,或返回深度拷贝,或使用AOP
[C++]
class config
{
public:
const list<string> & get_attributes(){
return attrs;
}
private:
list<string> attrs;
};
[C#]
public class SomeClass {
private IList attrs;
public IList GetAttributes(){
return ArrayList.ReadOnly(attrs);
}
}
[Java]
public class SomeClass {
private List attrs;
public List getAttributes(){
return Collections.unmodifiableList(attrs);
}
}
效果
1,语言提供的功能可帮助在编译期进行检查,确保程序中连试图增删元素的代码都不存在;但对有意无意的const转型无能为力
2,类库提供的功能可帮助在运行期进行检查,确保程序中试图增删元素的操作都抛出异常
同步代理问题为了对象的线程安全引入了同步机制,却使对象在单线程环境下付出了不必要的性能上的代价,曾经的例子如写时拷贝COW
我见过的解决方法
就是视而不见,不做任何处理,使用同步原语
[C#]
public class SomeClass {
[MethodImplAttribute(MethodImplOptions.Synchronized)]
public void Add(string name){
attrs.Add(name);
}
}
[Java]
public class SomeClass {
public synchronized void Add(string name){
attrs.add(name);
}
}
我推荐的解决方法
参考类库的实现,提供没有同步的原始类,及有同步的代理类;早期的JDK中Vector及HashTable都是同步的类,新的ArrayList及HashMap都不是同步的,Collections提供了静态方法返回同步代理;当在多线程环境中需要更改集合时,使用代理类
[C#,多线程环境中使用同步代理的客户类代码]
public class SomeClass {
public SomeClass(IList source){
attrs = ArrayList.Synchronized(source);
}
public void Add(string name){
attrs.Add(name);
}
public void Remove(string name){
attrs.Remove(name);
}
}
[C#,单线程环境中使用同步代理的客户类代码]
public class OtherClass{
public OtherClass(IList source){
attrs = source;
}
public void Add(string name){
attrs.Add(name);
}
public void Remove(string name){
attrs.Remove(name);
}
}
[Java,多线程环境中使用同步代理的客户类代码]
public class SomeClass {
public SomeClass (List source){
attrs = Collections.synchronizedList(source);
}
public void add(string name){
attrs.add(name);
}
}
[Java,单线程环境中使用同步代理的客户类代码]
public class OtherClass{
public OtherClass(List source){
attrs = source;
}
public void add(string name){
attrs.add(name);
}
}
效果
不必为不需要的功能付出额外的代价
资源管理问题有时需要精确的控制资源分配和释放的时机,保证资源的异常安全,避免资源泄漏,导致死锁,文件丢失,数据库连接过多等
我见过的解决方法
在缺乏真正的局部对象和析构函数的语言中,try/catch/finally充斥在代码中
使用中间件可帮助解决部分资源管理,如数据库连接等
可能会出现基于AOP的资源管理框架
我推荐的解决方法
在C++中,自动化的资源管理是与生俱来的,即B.S.提出的“资源管理即初始化”(RAII)
在C#中,可使用using+IDispose取得近似RAII的效果
在Java中,我不知道,讨论一下
[C++,RAII,仅仅示例,操作文件应首选std::fstream等]
class File
{
public:
explicit File(string path){
pf = fopen(path.c_str(), "rwb");
}
~File(){
fclose(pf);
}
operator FILE* (){
return pf;
}
private:
FILE* pf;
};
[C++,RAII的客户代码,仅仅示例,操作文件应首选std::fstream等]
void test()
{
File file("auto.txt");
char buf[256];
fread(buf, 0, 256, file);//即使这个操作会抛出异常,文件依然会被关闭
}
[C#,仅仅示例]
public class File:IDisposable {
private FileStream fs;
public File(string path){
fs = new FileStream(path, FileMode.Create);
}
public static implicit operator FileStream(File file) {
return file.fs;
}
public void Dispose() {
fs.Close();
}
}
[C#,仅仅示例]
public class Test{
void test(){
using(File file = new File("auto.txt")){
//some read, write, etc.
}
//文件已经被关闭,即使某步操作抛出异常
}
}
效果
1,资源管理自动化,不局限于内存
2,C++中使用模板,可统一定义大部分资源的包装类,目前的C#只能为每种资源定义单独的类,或者使用AOP
构造函数中的虚函数语言特性[C++]
虚函数与对象状态有关,与访问权限(public/protected/private)无关
只要子类对象构造出来了,就可以调用重写的方法,不管访问权限
[Java, C#]
虚函数与对象状态无关,与访问权限(public/protected/private/default/internal)有关
只要访问权限允许,就可以调用重写的方法,不管子类对象构造出来没有
后果
[C++]
在基类构造函数/析构函数里调用的方法永远都是基类的实现,不会调到子类;在其它方法里面虚函数永远都是调到子类的覆写实现,不管是不是private
[Java, C#]
在基类构造函数里调用方法,只要子类覆写了该方法,就会调到子类的实现
解决方法
慎重的在构造函数中调用虚函数,尤其是在Java和C#中,至少应该在注释中说明理由
强制针对接口编程问题尽管“针对接口编程”做为一条原则已经广为流传,但实际应用中仍然随处可见HashMap,Vector等做为接口参数、返回值传来传去
我见过的解决方法
使用Factory Method返回接口,并最小化具体类构造函数的访问权限,或类本身的访问权限
我推荐的解决方法
Factory Method依然值得推荐,另外可以利用语言本身的特性来避免多写一个Factory Method
在C++中,override一个虚函数时可以任意改变它的访问权限,包括将它由public变为private;有人说这样会破坏封装,但只要语义正确,有意为之,也没什么问题
在C#中,可使用“显式接口成员实现”
在Java中,我不知道,讨论一下
[C++]
class ISomeInterface
{
public:
virtual void SomeMethod() = 0;
};
class SomeClass : public ISomeInterface
{
private:
void SomeMethod(){
std::cout << "Subclass\n";
}
};
int main(int argc, _TCHAR* argv[])
{
SomeClass obj;
obj.SomeMethod(); //Error
ISomeInterface& iobj = obj;
iobj.SomeMethod(); //Ok
return 0;
}
[C#]
public interface ISomeInterface {
void SomeMethod();
}
public class SomeClass:ISomeInterface {
//1,不要写访问修饰符;2,使用方法全名
void ISomeInterface.SomeMethod(){
System.Console.WriteLine("Subclass");
}
}
public class Test{
void test(){
SomeClass obj = new SomeClass();
obj.SomeMethod(); //Error;
ISomeInterface iobj = obj;
iobj.SomeMethod(); //Ok
}
}
效果
1,少写一个Factory Method
2,不需要控制构造函数的访问权限
抗变与协变问题在override虚函数时,子类有时想要返回或处理与父类函数参数和返回值略微不同的类型,比如假设“动物类”有一个“伴侣”的虚函数,其返回值类型为“动物类”,但子类“兔子”override“伴侣”时,需要把返回值改为“兔子”;假设“鸟类”有一个“进食”的虚函数,其参数类型为“谷类”,但子类“食铁鸟”override“进食”时,需要把参数改为“碱性食物”;这时,除了使用泛型可以解决外,就需要用到抗变与协变
定义
抗变:向父类的方向变化
协变:向子类的方向变化
语言支持
返回值抗变与参数协变会带来明显的类型安全问题,因此,常用的基本是返回值协变与参数抗变;对抗变与协变支持的最全面的是Eiffel,它同时提供了受束泛型来解决返回值抗变与参数协变带来的类型安全问题
[C++]
只支持返回值协变
class Animal
{
public:
virtual Animal const& Spouse() = 0;
};
class Rabbit : public Animal
{
public:
Rabbit const& Spouse(){
return Rabbit();
}
};
[C#]
不支持
[Java]
1.5之前不支持,1.5与泛型结合,有限度的支持协变;
另外迫于checked exception的蹩脚实现,Java支持异常声明的协变
【推荐参考资料】
1.C#标准:ECMA-334 : C# Language Specification
2.Java标准:The Java™ Language Specification Second Edition
3.C++标准:ISO/IEC 14882:2003 Programming Languages - C++
4.The C# Programming Language
5.The Java Programming Language
6.The C++ Programming Language
本文地址:http://com.8s8s.com/it/it28305.htm