1

我正在使用第 3 方程序集为我的项目提供 sftp 功能,如下所示。多个线程需要实现此功能,因此第 3 方程序集在静态帮助程序类中使用。

除非遇到不寻常的错误情况,否则第 3 方程序集运行良好,例如 sftp 服务器存在,但我试图将文件放入的目录被删除。当发生这种情况时,会引发一个有效的异常,但是似乎第 3 部分组件间歇性地没有正确清理自身,因为没有新的客户端实例可以工作。

static readonly object _locker = new object();

    static void WorkerMethodCalledByThread()
    {
        lock(_locker)
        {
            var client = new ThirdPartyAssembly();
            bool hasConnection = false;
            try
            {
                client.Connect("some connection details");
                hasConnection = true;
                // do some work - 
                //but if an issue happens here then no new instances of ThirdPartAssembly work
            }
            catch(Exception e)
            {
                Logger.LogError(e);
            }
            finally
            {
                if (hasConnection) { client.Close(); }
            }
        }
    }

我没有办法改变第 3 方大会,所以我正在寻找一种处理它可能占用的任何资源的方法。但是,被调用的第 3 方类本身并不实现 IDisposable。

所以..我想弄清楚是否可以释放第 3 方使用的资源?

我能想到的两种可能的方法是

1:添加一个包装类,它以 3rd 方类作为其基类,并且还实现了 IDisposable(如下所示) - 但是我很确定这不会清理基类

2:对第 3 方类使用 Wea​​kReference(如下所示) - 但我不知道这是否可行

包装类

public class WrapperClass : ThirdPartyClass, IDisposable
{
    ~WrapperClass()
    {
        Dispose();
    }

    public void Dispose()
    {
        GC.SuppressFinalize(this);
    }
}

弱引用

static readonly object _locker = new object();

    static void WorkerMethodCalledByThread()
    {
        lock(_locker)
        {
            var client = new WeakReference(new ThirdPartyAssembly());
            bool hasConnection = false;
            try
            {
                ((ThirdPartyAssembly)client.Target).Connect("some connection details");
                hasConnection = true;
                // do some work - 
                //but if an issue happens here then no new instances of ThirdPartAssembly work
            }
            catch(Exception e)
            {
                Logger.LogError(e);
            }
            finally
            {
                if (hasConnection) { ((ThirdPartyAssembly)client.Target).Close(); }
            }
        }
    }
4

1 回答 1

1

所以问题WeakReference是:

同时仍然允许垃圾收集回收该对象。

它不能保证 GC 会得到它,因为它显然没有在内部很好地管理自己。

更好的方法是建立类并实现IDisposable. 这将允许您在using语句中利用您的包装器。然而,需要注意的是,真的没有办法绕过它,你将不得不使用它Reflection来挖掘这个类并清理它。

这将需要很多断点,并且在Watch窗口中进行大量挖掘,但你可以完成它!

于 2013-08-02T13:25:29.747 回答