惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

Attack and Defense Labs
Attack and Defense Labs
The GitHub Blog
The GitHub Blog
C
Check Point Blog
博客园_首页
MongoDB | Blog
MongoDB | Blog
N
Netflix TechBlog - Medium
F
Full Disclosure
Microsoft Security Blog
Microsoft Security Blog
爱范儿
爱范儿
Recent Announcements
Recent Announcements
阮一峰的网络日志
阮一峰的网络日志
G
GRAHAM CLULEY
Cyber Security Advisories - MS-ISAC
Cyber Security Advisories - MS-ISAC
T
Threat Research - Cisco Blogs
C
Cybersecurity and Infrastructure Security Agency CISA
V
Vulnerabilities – Threatpost
K
Kaspersky official blog
博客园 - 司徒正美
S
Schneier on Security
T
The Exploit Database - CXSecurity.com
Project Zero
Project Zero
云风的 BLOG
云风的 BLOG
Cisco Talos Blog
Cisco Talos Blog
Know Your Adversary
Know Your Adversary
雷峰网
雷峰网
V
V2EX - 技术
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
Spread Privacy
Spread Privacy
罗磊的独立博客
K
KPMG report finds enterprise disconnect between AI and its ROI | CIO
S
Security Affairs
SecWiki News
SecWiki News
Schneier on Security
Schneier on Security
O
OpenAI News
Jina AI
Jina AI
PCI Perspectives
PCI Perspectives
Cyberwarzone
Cyberwarzone
Y
Y Combinator Blog
Apple Machine Learning Research
Apple Machine Learning Research
B
Blog RSS Feed
I
InfoQ
D
Docker
P
Palo Alto Networks Blog
Recorded Future
Recorded Future
M
MIT News - Artificial intelligence
博客园 - Franky
B
Blog
Scott Helme
Scott Helme
博客园 - 叶小钗
D
DataBreaches.Net

博客园 - 漫步人生

c# 使用GetOleDbSchemaTable获取access数据库结构 xp上vs2010+IE8无法调试脚本 iis 上部署asp.net程序出错排查 windows2008的共享 使用sqlite绿色部署问题 在vs中使用正则表达式 - 漫步人生 事件不应该有返回值 adf for dotnet中servercontext的释放 修改Windows中ASPNET账户的密码 ae显示arcserver的地图服务 arcgis 编辑时独占mdb - 漫步人生 adf for dotnet中动态给ArcGis Server服务添加图层 js中在不知道属性名时获取属性对象 html文档中的name属性 ie中js a光标位置 - 漫步人生 adf中UpdatePanel刷新地图 windows局域网内以arcgis server local的方式连接arcserver服务器 - 漫步人生 c#与 如何调试 - 漫步人生 C#与C/C++的互操作 - 漫步人生
ae的com对象在dotnet中的释放问题
漫步人生 · 2011-11-11 · via 博客园 - 漫步人生

ae的com对象是需要释放的,不然就可能会锁住一些基础设备(如mdb文件等),这里研究了一下ae锁mdb的情况

释放方法一般是,Marshal.ReleaseComObject或Marshal.FinalReleaseComObject

但要在什么时候释放com对象呢,这就需要了解dotnet跟com交互的实现方法:运行库可调用包装(RCW)

每次将 COM 接口指针映射到该运行时可调用包装时,此引用计数都将递增。这是msdn中Marshal.ReleaseComObject 方法 描述里的一句话。那这句话是什么意思呢,什么样的操作才会导致将COM接口指针映射到运行时可调用包装。做了一个简单的实验,打开一个workspace,并打开一个featureclass,使用方法Marshal.ReleaseComObject释放COM,查看引用计数(不知道如何查看引用计数,只能通过Marshal.ReleaseComObject方法),以此来判断是否执行了将COM接口指针映射到运行时可调用包操作。

情况一:新声明一个workspace局部变量,将原来的workspace值赋给新变量,释放workspace,查看workspace引用计数

IWorkspaceFactory wsf = new AccessWorkspaceFactoryClass();
IWorkspace ws = wsf.OpenFromFile(mdbPath, 0);
IFeatureClass fls = ((IFeatureWorkspace)ws).OpenFeatureClass(featureClassName);
IWorkspace ws1 = ws;
int i = Marshal.ReleaseComObject(ws);

i的值为0。

情况二:新声明一个workspace局部变量,通过IDataset.Workspace属性给新变量赋值,释放workspace,查看workspace引用计数

IWorkspaceFactory wsf = new AccessWorkspaceFactoryClass();
IWorkspace ws = wsf.OpenFromFile(mdbPath, 0);
IFeatureClass fls = ((IFeatureWorkspace)ws).OpenFeatureClass(featureClassName);
IWorkspace ws1 = ((IDataset)fls).Workspace;
int i = Marshal.ReleaseComObject(ws);

i的值为1。

我大胆的得出结论(有可能不对):将COM接口指针映射到运行时可调用包装操作是在调用执行com对象方法并返回值时才会发生。

情况三:在一个方法里面打开featureclass,不释放workspace,新声明一个workspace局部变量,通过IDataset.Workspace属性给新变量负责,释放workspace,查看workspace引用计数

       private IFeatureClass getFclss(string path, string featureClassName)
        {
            IWorkspaceFactory wsf = new AccessWorkspaceFactoryClass();
            IWorkspace ws = wsf.OpenFromFile(path, 0);
            return ((IFeatureWorkspace)ws).OpenFeatureClass(featureClassName);
        }
        IFeatureClass fls = getFclss(mdbPath, featureClassName);
        IWorkspace ws = ((IDataset)fls).Workspace;
        int i = Marshal.ReleaseComObject(ws);

i的值为1。

情况四:跟情况三类似,不同的是调用了GC.Collect方法

       IFeatureClass fls = getFclss(mdbPath, featureClassName);
       GC.Collect();
        IWorkspace ws = ((IDataset)fls).Workspace;
       int i = Marshal.ReleaseComObject(ws);

i的值为0。

比较情况三跟情况四,可以得出结论,释放COM对象在dotnet中的映射对象的时候,引用计数会减一

情况五:调用情况三的获取featureclass的方法,释放得到的featureclass,删除mdb文件

IFeatureClass fls = getFclss(mdbPath, featureClassName);
Marshal.ReleaseComObject(fls);
File.Delete(mdbPath);

得到“文件“。。。”正由另一进程使用,因此该进程无法访问该文件。”的异常。

情况六:修改getFclss方法,在方法内释放掉worksapce,再像情况五一样操作

        private IFeatureClass getFclss(string path, string featureClassName)
        {
            IWorkspaceFactory wsf = new AccessWorkspaceFactoryClass();
            IWorkspace ws = wsf.OpenFromFile(path, 0);
            try
            {
                return ((IFeatureWorkspace)ws).OpenFeatureClass(featureClassName);
            }
            finally
            {
                Marshal.ReleaseComObject(ws);
            }
        }

这时可以删除掉文件。

情况七:调用情况六修改后的getFclss,不释放featureclass,直接删除文件

IFeatureClass fls = getFclss(mdbPath, featureClassName);
File.Delete(mdbPath);

得到情况五一样的异常。

情况八:调用情况六修改后的getFclss,新声明一个workspace局部变量,通过IDataset.Workspace属性给新变量赋值。

IFeatureClass fls = getFclss(mdbPath, featureClassName);
IWorkspace ws = ((IDataset)fls).Workspace;

这时得到的ws是可以查看属性的,释放后的com对象查看属性会得到提示为“COM 对象与其基础 RCW 分开后就不能再使用。”的异常。

情况九:workspace打开ifeatureclass2次

            IWorkspaceFactory wsf = new AccessWorkspaceFactoryClass();
            IWorkspace ws = wsf.OpenFromFile(mdbPath, 0);
            IFeatureClass fcls= ((IFeatureWorkspace)ws).OpenFeatureClass(featureClassName);
            IFeatureClass fcls1 = ((IFeatureWorkspace)ws).OpenFeatureClass(featureClassName);
            int i = Marshal.ReleaseComObject(fcls);

i的值为1。

那么对于访问COM对象属性的属性的情况会是怎么样呢。

情况十:创建getDataset方法,跟新的getFclss一样,只是返回的是IDataset,访问IDataset.Workspace,再访问IDataset.Workspace.PathName,释放workspace,查看引用计数

       private IDataset getDataset(string path, string featureClassName)
        {
            IWorkspaceFactory wsf = new AccessWorkspaceFactoryClass();
            IWorkspace ws = wsf.OpenFromFile(path, 0);
            try
            {
                return ((IFeatureWorkspace)ws).OpenFeatureClass(featureClassName) as IDataset;
            }
            finally
            {
                Marshal.ReleaseComObject(ws);
            }
        }
        IDataset ds = getDataset(mdbPath, featureClassName);
            IWorkspace ws = ds.Workspace;
            string s = ds.Workspace.PathName;
            int i = Marshal.ReleaseComObject(ws);

i的值为1。

所以,为了避免有的对象释放漏掉,最好不要使用IDataset.Workspace.PathName这种写法。