我正在搞乱来自两个COM组件的两个对象,但是它们基本上是相同的,具有不同的方法并且几乎没有(但从技术上讲它们是不同的类)
说我要处理的对象a
和b
:
var app = Marshal.GetActiveObject("MotherClassOfA&B");
NamespaceA.A a = app.OpenAsA("random-filepath") ;
NamespaceB.B b = app.OpenAsB("random-filepath") ;
我想访问一些a
的成员,也可以从b
获得,但是,他们仍然是不同的类
public void DoWorkA(NamespaceA.A target){
DealWith(target.member1);
Check(target.member2);
BlahBlah(target.member3);
... // many others
}
public void DoWorkB(NamespaceB.B target){
DealWith(target.member1);
Check(target.member2);
BlahBlah(target.member3);
... // many others same as DoWorkA
}
虽然它们有相同的名字,typeof target.member1
的a
是NamespaceA.someClassA
,而typeof target.member1
的b
是NamespaceB.someClassB
,但它们有相同的名称(对于大多数成员而言)
问:DoWorkA
和DoWorkB
几乎相同,是否有可能为DoWork
和a
写一个通用方法b
?
附:我尝试了DoWork<T>(T target) where T:A,B
,但它无法构建,因为Visual Studio无法告诉target.member1
正在调用someClassA
或someClassB
错误
member1
是NamespaceA.someClassA.member1
和NamespaceB.someClassB .member1
之间的模糊参考
对于任何想知道的人,具体来说,他们是:
a
是AutoCAD.AcadDocument
,b
是AXDBLib.AxDbDocument
并且没有类或界面IDocument
使a is IDocument
和b is IDocument
两个true
响应your comment并在@JeroenMostert's suggestion上构建,您可以为所有方法使用一个界面,或者为每种方法使用一个界面,或者根据您的需要使用其中任何一种方法。让我们为所有方法创建一个......
interface IDocumentAdapter
{
bool Member1
{
get; set;
}
int Member2
{
get; set;
}
string Member3
{
get; set;
}
}
对于您正在调整的每种类型,您将需要一个实现此接口的类,并包含对适应类型的实例的引用...
class AcadDocumentAdapter : IDocumentAdapter
{
public AcadDocumentAdapter(AutoCAD.AcadDocument document)
{
Document = document;
}
private AutoCAD.AcadDocument Document
{
get;
}
public bool Member1
{
get => Document.Member1;
set => Document.Member1 = value;
}
public int Member2
{
get => Document.Member2;
set => Document.Member2 = value;
}
public string Member3
{
get => Document.Member3;
set => Document.Member3 = value;
}
}
class AxDbDocumentAdapter : IDocumentAdapter
{
public AxDbDocumentAdapter(AXDBLib.AxDbDocument document)
{
Document = document;
}
private AXDBLib.AxDbDocument Document
{
get;
}
public bool Member1
{
get => Document.Member1;
set => Document.Member1 = value;
}
public int Member2
{
get => Document.Member2;
set => Document.Member2 = value;
}
public string Member3
{
get => Document.Member3;
set => Document.Member3 = value;
}
}
然后,您只需要一个DoWork()
方法,通过IDocumentAdapter
接口执行其逻辑...
public void DoWork(IDocumentAdapter documentAdapter)
{
DealWith(documentAdapter.Member1);
Check(documentAdapter.Member2);
BlahBlah(documentAdapter.Member3);
}
或者,您可以简单地编写一个带有DoWork()
参数的dynamic
方法...
public void DoWork(dynamic target)
{
DealWith(target.member1);
Check(target.member2);
BlahBlah(target.member3);
}
这样做的好处就是不必为数百个成员编写适配器代码,但是缺点是在使用dynamic
变量时你不会得到任何编译器/ Intellisense帮助,因为成员访问不会被评估/检查/绑定直到运行。