一个比较有趣的问题:某程序在启动之前,需要对三种不同类型的数据的交叉引用(Cross Reference,xref)进行更新,现假设每种xref组件(xref component)都专门负责一种类型数据的交叉引用更新。交叉引用更新程序(XRefUpdator)在被构造的时候,会通过反射列举出当前assembly中的所有components,以便在更新的时候,逐一调用这些components的相应方法,完成对所有数据的更新。
很明显,为了强化类型和扩展方便,我们会定义一个xref component的泛型类,泛型类型就是交叉引用数据(XRefData)。由此,出现如下定义:
?
1234567891011121314 | public interface IXRefData { // TODO: add definition here } public class XRefComponent<TData> where TData : IXRefData { public void Update() { } } public class XRefPickData : IXRefData { } public class XRefPackData : IXRefData { } |
好了,现在考察XRefUpdator的定义。根据上面的设计思路,XRefUpdator需要包含一个xref component的列表,以便在需要的时候进行遍历从而逐一更新数据:
?
1234 | public class XRefUpdator { private List<XRefComponent<...... } |
OK,到这里就傻眼了,上面的省略号部分不知道该怎么写,到底是XRefComponent<XRefPickData>还是XRefComponent<XRefPackData>,事实上两者都不是。由此引出了一个非常明显但又容易忽视的概念:泛型表达的是一组完全不同的类型。既然是不同的类型,我们也不可能将它们统一地装入一个列表当中。
如何解决这样的问题?其实方法很简单,就是引入接口。废话不说了,看完下面的代码后,您就会豁然开朗:
?
public interface IXRefData { // TODO: add definition here } public interface IXRefComponent { void Update(); } public class XRefComponent<TData> : IXRefComponent where TData : IXRefData { public void Update() { } } public class XRefPickData : IXRefData { } public class XRefPackData : IXRefData { } public class XRefUpdator { private List<IXRefComponent> components = new List<IXRefComponent>(); public XRefUpdator { foreach (Type type in this .GetType().Assembly.GetExportedTypes()) { if ( typeof (IXRefComponent).IsAssignableFrom(type)) { components.Add((IXRefComponent)Activator.CreateInstance(type)); } } } } |
到这里,我想您应该觉得事情已经结束了。事实上并非如此。如果XRefComponent<TData>类中的Update方法需要用到TData类型的参数,或者返回值是TData类型,那么,你就无法简单地使用IXRefComponent接口去做抽象,因为你没有办法在这个接口中定义Update方法。如果你使用泛型接口,那又到了问题的起点:你遇到的是一堆完全不同的接口。在后续的文章中,我会介绍如何解决这个问题。
共同学习,写下你的评论
评论加载中...
作者其他优质文章