我有一个COM类型(使用tlbimp.exe创建)和一个包装此对象的C#类。我想在我的C#包装器的终结器中执行一些清理工作。遵循指南here我可能会这样写:
public class MyClass : IDisposable
{
private IMyComObject comObject;
public MyClass()
{
comObject = new MyComObject();
}
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this);
}
~MyClass()
{
Dispose(false);
}
protected virtual void Dispose(bool disposing)
{
// Be tollerant of partially constructed instances
if (comObject != null)
{
comObject.Cleanup();
// Account for object being disposed twice
comObject = null;
}
}
// Other bits go here...
}
我知道终结器可以以任何顺序运行,所以我不应该尝试使用任何实现终结器的对象,但是据我所知tlbimp生成的COM类型没有实现终结器,所以上面应该是行。
我无法找到任何关于此的官方文档,所以我的问题是在终结器中引用和使用COM对象是否安全?
答案 0 :(得分:-1)
我创建了一个抽象的com包装类,从中派生出所有的com包装类。我工作得很好。我的原始代码是VB,因为我需要后期绑定。我的应用程序的其余部分是用c#编写的。
public abstract class ComWrapper : IDisposable
{
protected internal object _comObject;
protected ComWrapper(object comObject)
{
_comObject = comObject;
}
#region " IDisposable Support "
private bool _disposed = false;
protected virtual void FreeManagedObjects()
{
}
protected virtual void FreeUnmanagedObjects()
{
ReleaseComObject(_comObject);
}
private void Dispose(bool disposing)
{
if (!_disposed) {
if (disposing) {
FreeManagedObjects();
}
FreeUnmanagedObjects();
_disposed = true;
}
}
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this);
}
protected override void Finalize()
{
Dispose(false);
base.Finalize();
}
#endregion
}
和
public static class Helpers
{
public static void ReleaseComObject(ref object o)
{
if ((o != null)) {
try {
Marshal.ReleaseComObject(o);
} catch {
} finally {
o = null;
}
}
}
public static string ToDotNetString(object comString)
{
if (comString == null) {
return string.Empty;
}
string s = comString.ToString();
ReleaseComObject(ref comString);
return s;
}
}