【译】.NET中六个重要的概念:栈、堆、值类型、引用类型、装箱和拆箱

为什么要翻译

  一来是为了感觉国外优秀技术社区知名博主的高质量文章,二来是为了复习对.NET技术的基础拾遗达到温故知新的效果,最后也是为了锻炼一下本身的英文读写能力。由于是首次翻译英文文章(哎,原谅我这个菜比,弱爆了!),因此确定会有不少问题(有些语句理解不透彻,翻译出来也不通顺,还请不吝赐教),也请各位园友多多指正,谢谢!编程

关于原文做者

原文做者:Shivprasad koirala
原文地址:http://www.codeproject.com/Articles/76153/Six-important-NET-concepts-Stack-heap-value-types

内容导读

  • 概述
  • 当你声明一个变量背后发生了什么?
  • 堆和栈
  • 值类型和引用类型
  • 哪些是值类型,哪些是引用类型?
  • 装箱和拆箱
  • 装箱和拆箱的性能问题

1、概述

  本文会阐述六个重要的概念:堆、栈、值类型、引用类型、装箱和拆箱。本文首先会经过阐述当你定义一个变量以后系统内部发生的改变开始讲解,而后将关注点转移到存储双雄:堆和栈。以后,咱们会探讨一下值类型和引用类型,并对有关于这两种类型的重要基础内容作一个讲解。函数

  本文会经过一个简单的代码来展现在装箱和拆箱过程当中所带来的性能上的影响,请各位仔细阅读。工具

1

2、当你声明一个变量背后发生了什么?

  当你在一个.NET应用程序中定义一个变量时,在RAM中会为其分配一些内存块。这块内存有三样东西:变量的名称变量的数据类型以及变量的值性能

  上面简单阐述了内存中发生的事情,可是你的变量究竟会被分配到哪一种类型的内存取决于数据类型。在.NET中有两种可分配的内存:。在接下来的几个部分中,咱们会试着详细地来理解这两种类型的存储。测试

2

3、存储双雄:堆和栈

  为了理解栈和堆,让咱们经过如下的代码来了解背后到底发生了什么。spa

public void Method1()
{
    // Line 1
    int i=4;

    // Line 2
    int y=2;

    //Line 3
    class1 cls1 = new class1();
}

  代码只有三行,如今咱们能够一行一行地来了解到底内部是怎么来执行的。pwa

  • Line 1:当这一行被执行后,编译器会在栈上分配一小块内存。栈会在负责跟踪你的应用程序中是否有运行内存须要
  • Line 2:如今将会执行第二步。正如栈的名字同样,它会将此处的一小块内存分配叠加在刚刚第一步的内存分配的顶部。你能够认为栈就是一个一个叠加起来的房间或盒子。在栈中,数据的分配和解除都会经过LIFO (Last In First Out)即先进后出的逻辑规则进行。换句话说,也就是最早进入栈中的数据项有可能最后才会出栈。
  • Line 3:在第三行中,咱们建立了一个对象。当这一行被执行后,.NET会在栈中建立一个指针,而实际的对象将会存储到一个叫作“堆”的内存区域中。“堆”不会监测运行内存,它只是可以被随时访问到的一堆对象而已。不一样于栈,堆用于动态内存的分配。
  • 这里须要注意的另外一个重要的点是对象的引用指针是分配在栈上的。 例如:声明语句 Class1 cls1; 其实并无为Class1的实例分配内存,它只是在栈上为变量cls1建立了一个引用指针(而且将其默认职位null)。只有当其遇到new关键字时,它才会在堆上为对象分配内存。
  • 离开这个Method1方法时(the fun):如今执行控制语句开始离开方法体,这时全部在栈上为变量所分配的内存空间都会被清除。换句话说,在上面的示例中全部与int类型相关的变量将会按照“LIFO”后进先出的方式从栈中一个一个地出栈。
  • 须要注意的是:这时它并不会释放堆中的内存块,堆中的内存块将会由垃圾回收器稍候进行清理。

3

 

  如今咱们许多的开发者朋友必定很好奇为何会有两种不一样类型的存储?咱们为何不能将全部的内存块分配只到一种类型的存储上?翻译

  若是你观察足够仔细,基元数据类型并不复杂,他们仅仅保存像 ‘int i = 0’这样的值。对象数据类型就复杂了,他们引用其余对象或其余基元数据类型。换句话说,他们保存其余多个值的引用而且这些值必须一一地存储在内存中。对象类型须要的是动态内存而基元类型须要静态内存。若是需求是动态内存的话,那么它将会在堆上为其分配内存,相反,则会在栈上为其分配。指针

4

4、值类型和引用类型

  既然咱们已经了解了栈和堆的概念了,是时候了解值类型和引用类型的概念了。值类型将数据和内存都保存在同一位置,而一个引用类型则会有一个指向实际内存区域的指针。code

  经过下图,咱们能够看到一个名为i的整形数据类型,它的值被赋值到另外一个名为j的整形数据类型。他们的值都被存储到了栈上。

  当咱们将一个int类型的值赋值到另外一个int类型的值时,它其实是建立了一个彻底不一样副本。换句话说,若是你改变了其中某一个的值,另外一个不会发生改变。因而,这些种类的数据类型被称为“值类型”。

5

  当咱们建立一个对象而且将此对象赋值给另一个对象时,他们彼此都指向了以下图代码段所示的内存中同一块区域。所以,当咱们将obj赋值给obj1时,他们都指向了堆中的同一块区域。换句话说,若是此时咱们改变了其中任何一个,另外一个都会受到影响,这也说明了他们为什么被称为“引用类型”。

5、哪些是值类型,哪些是引用类型?

  在.NET中,变量是存储到栈仍是堆中彻底取决于其所属的数据类型。好比:‘String’或‘Object’属于引用类型,而其余.NET基元数据类型则会被分配到栈上。下图则详细地展现了在.NET预置类型中,哪些是值类型,哪些又是引用类型。

6

6、装箱和拆箱

  如今,你已经有了很多的理论基础了。如今,是时候了解上面的知识在实际编程中的使用了。在应用中最大的一个意义就在于:理解数据从栈移动到堆的过程当中所发生的性能消耗问题,反之亦然。

  考虑一下如下的代码片断,当咱们将一个值类型转换为引用类型,数据将会从栈移动到堆中。相反,当咱们将一个引用类型转换为值类型时,数据也会从堆移动到栈中。

  不论是在从栈移动到堆仍是从堆中移动到栈上都会不可避免地对系统性能产生一些影响。

  因而,两个新名词横空出世:当数据从值类型转换为引用类型的过程被称为“装箱”,而从引用类型转换为值类型的过程则被成为“拆箱”。

  若是你编译一下上面这段代码而且在ILDASM(一个IL的反编译工具)中对其进行查看,你会发如今IL代码中,装箱和拆箱是什么样子的。下图则展现了示例代码被编译后所产生的IL代码。

8

7、装箱和拆箱的性能问题

  为了弄明白到底装箱和拆箱会带来怎样的性能影响,咱们分别循环运行10000次下图所示的两个函数方法。其中第一个方法中有装箱操做,另外一个则没有。咱们使用一个Stopwatch对象来监视时间的消耗。

  具备装箱操做的方法花费了3542毫秒来执行完成,而没有装箱操做的方法只花费了2477毫秒,整整相差了1秒多。并且,这个值也会由于循环次数的增长而增长。也就是说,咱们要尽可能避免装箱和拆箱操做。在一个项目中,若是你须要装箱和装箱,请仔细考虑它是不是绝对必不可少的操做,若是不是,那么尽可能不用。

10

  虽然以上代码段没有展现拆箱操做,但其效果一样适用于拆箱。你能够经过写代码来实现拆箱,而且经过Stopwatch来测试其时间消耗。

 

相关文章
相关标签/搜索