和我一起作Tess的windbg lab - Lab3, Memory

原文地址:http://blogs.msdn.com/b/tess/archive/2008/02/15/net-debugging-demos-lab-3-memory.aspx

操作步骤:

1、产生压力:tinyget -srv:localhost -uri:/BuggyBits/Links.aspx -loop:4000

2、观察taskmgr的输出,w3wp的内存每秒钟大概增长100M。

3、内存到700M左右的时候,抓一个hang dump

4、由于这是一个memory的问题,所以我们要先看GC Heap的情况,运行命令:!eeheap -gc,结果如下:

  GC Heap Size  0x2b307720(724596512)

  由于dump一共870M,而GC占用了720M左右,所以我们的重点在于托管内存的分析。

5、看heap的整体状况,运行!dumpheap -stat,结果如下:

  790fd8c4    49787    721599752 System.String

  嗯,720M的托管内存中,String占用了绝大多数。

6、看一下string的情况,根据2/8原则,大小相同的string也许会很多,这里我们过滤一下,看看10K以上大小的字符串,运行命令:!dumpheap -mt 790fd8c4  -min 10000

  0331d6dc 790fd8c4    20020    
  03322534 790fd8c4    20020    
  0332738c 790fd8c4    20020    
  0332c1e4 790fd8c4    20020    
  0333103c 790fd8c4    20020    

  大部分都是20K的字符串,随便找一个,我们需要看它被谁分配的

7、运行!gcroot 0331d6dc,结果如下:

  Scan Thread 16 OSTHread 318
  Scan Thread 18 OSTHread c38
  Scan Thread 19 OSTHread a40
  Scan Thread 20 OSTHread c00
  Scan Thread 24 OSTHread 998
  Scan Thread 14 OSTHread 4cc
  Finalizer queue:Root:0331d6b8(Link)->
  0331d6c8(System.Text.StringBuilder)->
  0331d6dc(System.String)

  在14号线程中,Link引用了这个字符串。而且我们看到,link是在Finalizer Queue中的。

8、查看finalizequeue,输出如下:

  057e0bcc    35998       575968 Link

  一共35998个Link对象。由于该对象存在于Finalizequeue中,所以一定显示的实现了Finalize方法。

9、查看该方法,代码如下:

  ~Link()
  {
        //some long running operation when cleaning up the data
        Thread.Sleep(5000);
  }

10、换个方向,看上面步骤7中的那个Link对象,!do 0331d6b8,输出结果如下:

        MT    Field   Offset                 Type VT     Attr    Value Name
  790fdc5c  4000006        4 ...ext.StringBuilder  0 instance 0331d6c8 url
  790fd8c4  4000007        8        System.String  0 instance 029bb0b8 name
  再看第一个url对象,运行!do 0331d6c8 ,结果如下:

        MT    Field   Offset                 Type VT     Attr    Value Name
  791016bc  40000b1        8        System.IntPtr  1 instance    dc1d8 m_currentThread
  79102290  40000b2        c         System.Int32  1 instance 2147483647 m_MaxCapacity
  790fd8c4  40000b3        4        System.String  0 instance 0331d6dc m_StringValue

  注意最后一行的那个m_StringValue,对比一下步骤7中的!gcroot输入。

  从这里我们看到,Link中包含了一个StringBuilder,而StringBuilder中包含了一个20K的字符串。

11、看代码:

  public Link(string name, string url)
      {
          this.name = name;
          this.url.Append(url);
      }
  可以看到,Link对象的构造方法中,引用了字符串。

12、再回头看上面的步骤9,Link自作聪明的实现了Finalize方法,但是该方法执行的时间太长(这里是5秒钟),导致垃圾回收的时候,迟迟不能把该对象回收掉。因为Link引用了字符串url,所以相应的字符串也无法被回收。这样内存就上涨的很快了。

 

Over

posted @ 2010-06-27 13:30  鞠强  阅读(1008)  评论(0编辑  收藏  举报

hello

world