问:能否告诉我,在托管 C++ 中,使用 delete 操作符销毁托管对象是否安全?
Bernie Sanders
答:是安全的,可以在托管 C++ 中删除托管对象,不过要知道,delete 所做的就是调用对象的析构函数,所以必须显式定义它。调用 delete 并没有释放对象的存储空间。只有垃圾回收器才能够释放存储空间。图 1 所示的是一个简单的程序,它定义了一个托管类,这个类有一个析构函数,当运行这个析构函数时会显示一条消息。TESTDTOR 分配了两个 ManagedClass 实例。它显式删除第一个实例,但没有删除第二个。当您运行 TESTDTOR 时,就会得到如下所示的结果:
以下是引用片段: Begin main ManagedClass(04A712D4)::ctor ManagedClass(04A712D4)::dtor ManagedClass(04A712E0)::ctor End main ManagedClass(04A712E0)::dtor |
它表明当 delete 语句执行时,第一个对象的析构函数就会立即执行,而第二个对象(位于 04A712E0)并没有被销毁,直到控制离开 main,系统终止代码调用垃圾回收器释放未完成对象时才被销毁。
图 2 testdtor 的重要部分
要是对 .NET 环境中所发生的事情不确定,随时可以编写一些代码,对其进行编译,然后检查生成的 Microsoft 中间语言 (MSIL)。如图 2 所示,定义一个析构函数会使编译器生成两个方法:一个是 Finalize 方法,它包含您的实现(在本例中为调用 printf);另一个是 __dtor 方法,它调用 System.GC::SuppressFinalize,然后调用 Finalize。当您删除一个对象时,编译器会调用这个特定的 __dtor 方法。如果您带 /FAs 参数编译 TESTDTOR,产生随源代码列出的汇编语言,您就会发现 delete 语句按照以下方式进行编译:
以下是引用片段: delete pmc; ldloc.0 ; _pmc$ call ..1ManagedClass@@$$FQ$AAM@XZ |
那些奇怪的神秘符号就是析构函数 (__dtor) 的托管名称。
经验丰富的 C++ 编程人员也可能会疑惑不解,想知道如果调用 delete 不释放对象,那调用它有什么用呢?这个问题提得好。调用 delete 的唯一目的是回收类使用的任何非托管资源。例如,如果您的对象打开文件或创建一个数据库连接,则可以编写一个析构函数,当对象使用完毕时,可以用这个析构函数关闭它的资源,然后使用 delete 来调用它。释放托管类中的资源的更好方式是通过实现 Dispose 模式、IDisposable 和 auto_dispose 模式(如果您采用托管 C++ 编写)来调用它。有关更多信息,请参考 Tomas Restrepo 在 2002 年 2 月的 MSDN?Magazine 中撰写的“Tips and Tricks to Bolster Your Managed C++ Code in Visual Studio .NET”。
如果您实现 dispose 模式,.NET 的其他用户就可以使用它。如果您在析构函数中进行清理工作,其他语言就无法显式调用您执行清理工作的代码。在 C# 和 Visual Basic 中没有 delete 操作符。
所以虽然您可以调用 delete 来调用您的析构函数,但不提倡将执行清理工作的代码放在析构函数中。较好的方法是实现 IDisposable,所有的语言都可以使用它。不过要注意,这种行为在 Visual C++? 2005 中有所变化。有关更多信息,请参考 Andy Rich 在 Deterministic Finalization IV - Benefits, part II 中关于 deterministic finalization 的讨论,以及当前的 C++/CLI Language Specification Standard。
问:我有一个非托管函数,它返回一个包含 char* 字符串的链接列表:
以下是引用片段: struct blah { int a, b; char *a, *b; struct blah *next; }; struct blah *getmystruct(); |
因为 getmystruct() 分配了内存,所以当我使用完这个函数后,我需要调用 freemystruct(struct blah *b)。我试图生成一个包装,将它转换成托管类型的集合,但我不知道当这些指针需要释放时如何处理它们。您能指点我一下吗?
钢琴王子 ぎ孤独浪子ぎ 丽日和风/jd 同在天涯 ヤ繏擇ɑ愛伱 ╲ˊ.輪迴. 蕭遙☆朤髭 井中月/tp ▕&掛唸° | 舞~▲★~動 轻笑 。 raffle /‘糸統消息/ ヾぺ靈戀ズ♂ βrave Boy *红海* o2jam 叼煙d尐P孩 ǐ灆┊樰狐゛ 篍天dě諷 嬲你滴滴针蚤 蓝天ぁ的祁视 ⒈哖逅ˇ… ヴ梦潇泓波 永恒的诺言 ˊ谜薍﹏丶 ¤濸嗨鎦響◎ ヤ蓜ミ角﹖。 《铭簰娚朲》 ..↘我 .︶ㄣ鱌尐ǜ 悔 囧♀ 缘 ♀囧 ??昣愛恛忔х 程琪 20〇9莪娶祢 西门吹泡泡 栤葑Dě專情 ╰☆←珦祐赱 &野/mg百合 ◇◆.佐.铒﹎ 傷/aiq悩荕﹖ 〥.〥 吥洅泘嗳 冷月 唯叶 ︵St〇p戀﹏ QQ/mn小天使 _縞縞鎭 心情马桶 Sparerib小生 Ⅲ朝メ飘逸ジ 秋雨 《{( )}》 mai 花妖芭比 路西法 【彭☆少】 ㊣魂㊣ ~寶☆崽~b哭 L.-_-# 為妳╃→心動 右耳逝去爱 黑色幽默 ヤ`僮孓 你是我的特例 丑小鸭 倾城→恋 Wing ゛癡⌒礙 梅川&内酷 繒→戀過/xs ぜ3歲が亼粅 縌鎏魚oο/lh _べ小宝☆. ‘私乆專屬 -,潵啦嘿哟° 寻~* lorlin 死亡降临 §阳光男孩§ じ☆ve豪 oO黄天明Oo 嗜血残鹰 そ靈號黨舗ひ ヤ骑士の忠诚 ﹊寳寳佉褕歡 一晴空 寳少 ざ(_蒍祢戀伱 娟灬怺遠哒愛 V埪墹殺掱げ 小矮人 5chao@ 拿什么去经营 ╱╳灬萭衹鹤 ↘流星dè泪 ·羽·新· 哆唻M童衣阁 γ?ǒ?の航 ︷繒俓撮坏ヤ ☆鎏曐^o^☆ 心有灵犀 傷心ん楓枼 星期八值班 乄酷酷哋尐鱻 壊ㄨiǎo孩? 忘却的纪念 べ`鈊巳凍_oo "━ 活死人 '.縵縵' 东轩☆/:P 極度鬧心/dk 唓佡de沐腢 风逍天 憂鬱 ╰ァ認眞嗳妳 ╭ァ茳會軍ヤ .丶粉z!° 〃锘の唁⒅、 貝者 衤申 呼呼猪/aiq 笑面︶ㄣ二郎 耲壊ャ孩孓氣 ˊ;嗯.娶莪﹖ 狗狗的小主人 ????艷翎 忠義★無名 我爱足球 尐蕣鱫╭╮ 奶油小生 独善无忧 づ雲談風清つ ☆從新開始☆ °﹡尋 ミ壊壊ヤDê _10086咨询 (_菰獨奶嘴 寻宝 敟當/莪de愛 浪漫的蓝影子 衤果`ㄖ垂`衶 ♂YQ缘倩♀ ..日式宅男`` 己 ┧醜囡↖ ^@^趴趴*_* 七月 `′鉺乄 ﹎訞.釹_ 獨领峰彩 メ君臨天下メ _ . 嘜釾赯|_ 才思/aiq敏捷 *゛ 酃ぃ孁▽╰亟 無唁啲の傑 僾/love伱