草庐IT

c# - 从静态对象中包含的 C# 字典中释放内存

coder 2024-05-21 原文

我在使用 WCF Web 服务时遇到了一些问题(一些转储、内存泄漏等),并且我运行了一个 profillng 工具(ANTS 内存配置文件)。

只是为了发现即使处理结束(我运行特定测试然后停止),第 2 代也有 25% 的内存用于 Web 服务。我追踪这段内存,发现我有一个充满 (null, null) 项目的字典对象,哈希码为 -1。

Web 服务的工作流意味着在特定的处理过程中,项目被添加然后从字典中删除(只是简单的 AddRemove )。没什么大不了的。但似乎在删除所有项目后,字典中充满了 (null, null) KeyValuePair秒。实际上有数千个,以至于它们占据了很大一部分内存并最终发生溢出,相应的强制应用程序池回收和 DW20.exe 获得了它可以获得的所有 CPU 周期。

字典其实是Dictionary<SomeKeyType, IEnumerable<KeyValuePair<SomeOtherKeyType, SomeCustomType>>> ( System.OutOfMemoryException because of Large Dictionary ) 所以我已经检查过是否有某种引用资料。

字典包含在一个静态对象中(通过处理使不同的处理线程可以访问它)所以从这个问题和更多(Do static members ever get garbage collected?)我理解为什么那个字典在第 2 代中。但这也是那些(null,null)的原因?即使我从字典中删除项目,内存中也会始终占用某些内容吗?

这不是这个问题中的速度问题 Deallocate memory from large data structures in C# .似乎内存永远不会被回收。

我能做些什么来真正从字典中删除项目,而不仅仅是继续用 (null, null) 对填充它吗? 还有什么我需要检查的吗?

最佳答案

字典将项目存储在哈希表中。为此内部使用了一个数组。由于哈希表的工作方式,该数组必须始终大于实际存储的项目数(至少大 30%)。 Microsoft 使用 72% 的负载因子,即至少有 28% 的数组为空(参见 An Extensive Examination of Data Structures Using C# 2.0,尤其是 The System.Collections.Hashtable ClassThe System.Collections.Generic.Dictionary Class ) 因此 null/null 条目可以代表这个空闲空间。

如果数组太小,会自动增长;然而,当项目被删除时,数组不会收缩,但将释放的空间应该在插入新项目时重新使用。

如果你控制了这个字典,你可以尝试重新创建它以缩小它:

theDict = new Dictionary<TKey, IEnumerable<KeyValuePair<TKey2, TVal>>>(theDict);

但问题可能来自实际(非空)条目。你的字典是静态的,因此永远不会被垃圾收集器自动回收,除非你给它分配另一个字典或 null (theDict = new ...theDict = 空)。这仅适用于静态字典本身,而不适用于其条目。只要对已删除条目的引用存在于其他地方,它们就会持续存在。 GC 将回收任何无法通过某些引用访问的对象(较早或较晚)。无论此对象是否声明为静态,都没有区别。对象本身不是静态的,只有它们的引用。


正如@RobertTausig 友善地指出的那样,自 .NET Core 2.1 以来 有新的 Dictionary.TrimExcess(),这是您真正想要的,但没有存在于当时。

关于c# - 从静态对象中包含的 C# 字典中释放内存,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15022528/

有关c# - 从静态对象中包含的 C# 字典中释放内存的更多相关文章

  1. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  2. ruby-on-rails - Ruby net/ldap 模块中的内存泄漏 - 2

    作为我的Rails应用程序的一部分,我编写了一个小导入程序,它从我们的LDAP系统中吸取数据并将其塞入一个用户表中。不幸的是,与LDAP相关的代码在遍历我们的32K用户时泄漏了大量内存,我一直无法弄清楚如何解决这个问题。这个问题似乎在某种程度上与LDAP库有关,因为当我删除对LDAP内容的调用时,内存使用情况会很好地稳定下来。此外,不断增加的对象是Net::BER::BerIdentifiedString和Net::BER::BerIdentifiedArray,它们都是LDAP库的一部分。当我运行导入时,内存使用量最终达到超过1GB的峰值。如果问题存在,我需要找到一些方法来更正我的代

  3. ruby-on-rails - 按天对 Mongoid 对象进行分组 - 2

    在控制台中反复尝试之后,我想到了这种方法,可以按发生日期对类似activerecord的(Mongoid)对象进行分组。我不确定这是完成此任务的最佳方法,但它确实有效。有没有人有更好的建议,或者这是一个很好的方法?#eventsisanarrayofactiverecord-likeobjectsthatincludeatimeattributeevents.map{|event|#converteventsarrayintoanarrayofhasheswiththedayofthemonthandtheevent{:number=>event.time.day,:event=>ev

  4. ruby-on-rails - 如何验证非模型(甚至非对象)字段 - 2

    我有一个表单,其中有很多字段取自数组(而不是模型或对象)。我如何验证这些字段的存在?solve_problem_pathdo|f|%>... 最佳答案 创建一个简单的类来包装请求参数并使用ActiveModel::Validations。#definedsomewhere,atthesimplest:require'ostruct'classSolvetrue#youcouldevencheckthesolutionwithavalidatorvalidatedoerrors.add(:base,"WRONG!!!")unlesss

  5. Ruby 写入和读取对象到文件 - 2

    好的,所以我的目标是轻松地将一些数据保存到磁盘以备后用。您如何简单地写入然后读取一个对象?所以如果我有一个简单的类classCattr_accessor:a,:bdefinitialize(a,b)@a,@b=a,bendend所以如果我从中非常快地制作一个objobj=C.new("foo","bar")#justgaveitsomerandomvalues然后我可以把它变成一个kindaidstring=obj.to_s#whichreturns""我终于可以将此字符串打印到文件或其他内容中。我的问题是,我该如何再次将这个id变回一个对象?我知道我可以自己挑选信息并制作一个接受该信

  6. ruby-on-rails - 如果 Object::try 被发送到一个 nil 对象,为什么它会起作用? - 2

    如果您尝试在Ruby中的nil对象上调用方法,则会出现NoMethodError异常并显示消息:"undefinedmethod‘...’fornil:NilClass"然而,有一个tryRails中的方法,如果它被发送到一个nil对象,它只返回nil:require'rubygems'require'active_support/all'nil.try(:nonexisting_method)#noNoMethodErrorexceptionanymore那么try如何在内部工作以防止该异常? 最佳答案 像Ruby中的所有其他对象

  7. ruby-on-rails - 未在 Ruby 中初始化的对象 - 2

    我在Rails工作并有以下类(class):classPlayer当我运行时bundleexecrailsconsole然后尝试:a=Player.new("me",5.0,"UCLA")我回来了:=>#我不知道为什么Player对象不会在这里初始化。关于可能导致此问题的操作/解释的任何建议?谢谢,马里奥格 最佳答案 havenoideawhythePlayerobjectwouldn'tbeinitializedhere它没有初始化很简单,因为你还没有初始化它!您已经覆盖了ActiveRecord::Base初始化方法,但您没有调

  8. ruby - 如何在 Rails 4 中使用表单对象之前的验证回调? - 2

    我有一个服务模型/表及其注册表。在表单中,我几乎拥有服务的所有字段,但我想在验证服务对象之前自动设置其中一些值。示例:--服务Controller#创建Action:defcreate@service=Service.new@service_form=ServiceFormObject.new(@service)@service_form.validate(params[:service_form_object])and@service_form.saverespond_with(@service_form,location:admin_services_path)end在验证@ser

  9. ruby-on-rails - Ruby 中的内存模型 - 2

    ruby如何管理内存。例如:如果我们在执行过程中采用C程序,则以下是内存模型。类似于这个ruby如何处理内存。C:__________________|||stack|||------------------||||------------------|||||Heap|||||__________________|||data|__________________|text|__________________Ruby:? 最佳答案 Ruby中没有“内存”这样的东西。Class#allocate分配一个对象并返回该对象。这就是程序

  10. c# - 如何在 ruby​​ 中调用 C# dll? - 2

    如何在ruby​​中调用C#dll? 最佳答案 我能想到几种可能性:为您的DLL编写(或找人编写)一个COM包装器,如果它还没有,则使用Ruby的WIN32OLE库来调用它;看看RubyCLR,其中一位作者是JohnLam,他继续在Microsoft从事IronRuby方面的工作。(估计不会再维护了,可能不支持.Net2.0以上的版本);正如其他地方已经提到的,看看使用IronRuby,如果这是您的技术选择。有一个主题是here.请注意,最后一篇文章实际上来自JohnLam(看起来像是2009年3月),他似乎很自在地断言RubyCL

随机推荐