随笔-87  评论-10  文章-1 

.NET垃圾回收机制 GC

线程栈 stuck:存值类型,和引用类型的引用  先进后出,链表形式,连续摆放 CLR(公共语言运行库(Common Language Runtime))启动进程,main函数为一个线程入口

进程堆heap:存引用类型 进程中的一块区域

IL:中间语言

对象的属性为值类型出现在堆里,方法里的值类型,由进程调用,出现在栈里

 

    /// <summary>
    /// class  引用类型
    /// </summary>
    public class ReferenceTypeClass
    {
        private int _valueTypeField;//堆:因为对象都在堆里,对象里面的属性也在堆里
        public ReferenceTypeClass()
        {
            _valueTypeField = 0;
        }
        public void Method()
        {
            int valueTypeLocalVariable = 0;//栈:全新的局部变量,线程栈来调用方法,然后分配内存
            new Process();//new对象,分配在堆里
        }
    }

 装箱拆箱(仅仅是说内存的拷贝动作):内存copy 也会浪费性能  通常都是因为object,

 装箱拆箱只能发生在父子类里面, 因为这样你才能转换呀

dynamic 是引用类型的语法糖

                string student = "123";
                string student2 = student;

                Console.WriteLine(student); //123
                Console.WriteLine(student2);//123

                student2 = "APP";
              
                Console.WriteLine(student);//123
                Console.WriteLine(student2);//APP            
     string student = "大山";
                string student2 = "APP";//共享
                student2 = "大山";
                Console.WriteLine(object.ReferenceEquals(student, student2));//true 竟然一样
                //就是同一个  享元模式 CLR内存分配字符串的时候,会查找相同值,有就重用了

 

 托管资源:.Net  New的类,出了作用域就访问不到了,自动释放了,存放在进程堆中的资源; 值类型变量,存放在进程栈中,出了作用域就是放了,

非托管资源:访问数据库,操作Excel,Word什么的.

 

析构函数 ~Class() 见下图

    public class Class : IDisposable
    {
        public int ClassId { get; set; }
        public string ClassName { get; set; }
        ~Class()
        {
            MyLog.Log($"执行{this.GetType().Name}Dispose");
        }
        public void Dispose()
        {
            MyLog.Log($"执行{this.GetType().Name}Dispose");
        }
    }

 

 主要是用来释放非托管资源,等着GC去把非托管资源释放掉 系统自动执行
 GC.Collect();//主动GC(释放资源) GC回收的时候,CLR一定调用析构函数

Dispose() 也是释放非托管资源的,主动释放,方法本身是没有意义的,我们需要在方法里面实现对资源的释放
GC不会调用,而是用对象时,使用者主动调用这个方法(using),去释放非托管资源

总结:

1.循环New100个对象,出了作用域,就访问不到了,没有引用,.Net GC就释放了,但少New对象,创建对象需要内存开辟空间

2.实现Idispose() 接口的 使用Using 或调用dispose() 方法

3.操作Word或Excel 的时候 时候需要注意释放资源

前一段时间做了一个小程序,Word批量转PDF,运行的时候内存持续增加,加上释放资源就好了,见下图

   /// <summary>
        /// word 操作类
        /// </summary>
        Microsoft.Office.Interop.Word.Application application =  new Microsoft.Office.Interop.Word.Application();
        /// <summary>
        ///word 转换成pdf
        /// </summary>
        /// <param name="sourcePath"></param>
        /// <param name="targetPath"></param>
        /// <returns></returns>
        public bool WordToPDF(string sourcePath, string targetPath)
        {

            bool result = false;

            Microsoft.Office.Interop.Word.Document _document = null;
            // application.
            try
            {
                application.Visible = false;
                _document = application.Documents.Open(sourcePath);
                _document.ExportAsFixedFormat(targetPath,
                    Microsoft.Office.Interop.Word.WdExportFormat.wdExportFormatPDF);
                result = true;
            }
            catch (Exception e)
            {
                Console.WriteLine(e.Message);
                result = false;
            }
            finally
            {
                var doc_close = (Microsoft.Office.Interop.Word._Document)_document;
                if (doc_close != null)
                {
                    doc_close.Close();
                }
                else
                {
                    result = false;
                }

                // _document.Close();
            }
            return result;
        }

 

4.访问数据库的时候还没看

 

posted on 2018-10-08 01:34 从未太晚 阅读(...) 评论(...) 编辑 收藏