建站.Net常识 详谈值类型和引用类型

所属分类: 软件教程 / 编程开发 阅读数: 385
收藏 0 赞 0 分享

值类型和引用类型是.net里面的一个基本概念
在面试的时候也经常遇到
关于这个概念有很多误解,经常听到下面的说法
1.两者的区别是值类型分配在堆栈上,引用类型分配在堆上
  这句话不对,至少不准确
2.值类型性能更好,
  这句话要考虑情况
先补充一些背景资料
常见的值类型有:大部分原生类型,例如int float long 各种自己定义的结构体等等
常见的引用类型有:string 各种Class 数组(包括int[]这种的)
堆栈:在这里指的是执行堆栈
堆:在这里指的是托管堆,就是LOH+G0+G1+G2
让我们先来看看第一点:两者的区别是值类型分配在堆栈上,引用类型分配在堆上
1.假设在一个方法里面有一个语句是 var obj = new object();
首先 new 出来的Object将被存放在堆中
obj在堆栈上,其内容是一个指针,指向new 出来的那个Object
2.然后假设在一个方法里面有一个语句是 var i =1 ;
这里的 i 在堆栈上, 其值是1 (int 类型)
3.类中的值类型成员,例如以下一个定义
public class ClassA
{
private int i = 1;
}
假设在一个方法里面有一个语句是 var obj = new ClassA();
首先 new 出来的ClassA将被存放在堆中
obj在堆栈上,其内容是一个指针,指向new 出来的那个ClassA
ClassA中的成员 i 这个时候也在堆上
假设有一个有一个其他语句使用到ClassA.i 这个i的值才会被拷贝到堆栈上(大部分默认的情况)
4.将引用类型放在堆栈上
unsafe
{
var obj = stackalloc int[100];
}
stackalloc是用来在堆栈上分配内存的keyword
上面的4个例子正好证明了 引用类型和值类型都可以存在在堆和堆栈上
不过大部分时候都是情况1和2, 所以大部分引用类型都在堆上,大部分
让我们先来看看第二点:值类型性能更好
就上面的情况1,2而言
a.在取一个对象的时候,情况1先读取obj的值, 这是一个地址,然后要重新读取该地址的真正的对象Object
情况2读取obj的值,这就是真正的值了,所以相对数据比较快
b.在堆中的对象受到GC的影响,需要额外的CPU资源;(堆栈中的对象,出栈以后释放掉了)
c.在堆中的对象需要等到GC后才被释放,所以暂用内存时间较久
其他情况:
1.考虑一些情况,装箱拆箱;这是值类型在堆栈和对中拷贝时特有的操作,该操作还是非常消耗资源的
  那么如果无法避免装箱拆箱,就要考虑避免使用值类型了
2.值类型传递的时候每次都是值拷贝,如果某个值类型很大(例如自己定义的struct) 那么这个性能也是个问题;(而且还要考虑到堆栈有大小限制)
  所以一般情况下比较复杂的类型都只能用class
3.许多时候,引用比较都比值比较来的快,因为引用比较只要看看两个地址是否相等
  而值比较却要考虑里面真实的值
值类型和引用类型的区别其实从他们的名字上就看的出来
在传递值类型的时候传递的是真实值,这也就意味着原来的值被复制了一份到新的位置
而在传递引用类型的时候传递的是引用(地址),这里并没有复制一份原来值的动作,因此两个引用都指向一个对象
Ref
在没有Ref的时候传递参数,CLR会为每个参数弄一个临时变量出来,存储值类型的值或者是引用类型的指针
这种情况下修改值类型或者引用类型的值不会影响到原来的值
但是修改引用类型的成员会影响到原来的值,下面两个例子分别是修改参数成员和修改参数本身
?
public class ClassA
{
public string Name { get; set; }
}
class Program
{
static void Main(string[] args)
{
ClassA a = new ClassA();
Test(a);
Console.WriteLine(a.Name);//这里会输出mark
}
private static void Test(ClassA a)
{
a.Name = "Mark";
}
}
?
public class ClassA
{
public string Name { get; set; }
}
class Program
{
static void Main(string[] args)
{
ClassA a = new ClassA();
Test(a);
Console.WriteLine(a.Name);//这里不会输出Liu
}
private static void Test(ClassA a)
{
a = new ClassA() { Name = "Liu" };
}
}
  
在有REF的情况下传递参数,CLR就会把原来的变量的地址传递过去,如果修改了该变量会影响到原来的成员
?
public class ClassA
{
public string Name { get; set; }
}
class Program
{
static void Main(string[] args)
{
ClassA a = new ClassA();
Test(ref a);
Console.WriteLine(a.Name);//这里会输出Liu
}
private static void Test(ref ClassA a)
{
a = new ClassA() { Name = "Liu" };
}
}
  
备注1:如何确定一个对象在堆上或者是堆栈上
刚才说的都是理论,这边是验证
使用WinDBG+SOS附加到一个.net程序上;然后查看堆中的情况;
具体指令就不说了,大家查看一下帮助

更多精彩内容其他人还在看

Unity3D摄像机跟随小球移动而不旋转的设置方法

Unity3D中怎样让摄像机跟随小球移动而不旋转?Unity3D中使用摄像机拍小球,它在移动的过程中,自身是不断旋转的,所以,摄像机也会跟着一起旋转,怎么让摄像机不转呢?请看下文详细介绍
收藏 0 赞 0 分享

Myeclipse断点不起作用出现斜线该怎么办呢?

Myeclipse断点不起作用出现斜线该怎么办呢?Myeclipse断点处如果出现斜线,那就说明断点不起作用了,该怎么处理这种情况呢?下面我们一起来看看详细的解决办法,需要的朋友可以参考下
收藏 0 赞 0 分享

Visual Studio中无法查找或打开PDB文件怎么办?

Visual Studio中无法查找或打开PDB文件怎么办?在VS底部的“输出”框中提示“无法查找或打开 PDB 文件”,出现这个问题该怎么解决呢?下面我们来看看详细的解决办法,需要的朋友可以参考下
收藏 0 赞 0 分享

VS2013调试代码时怎么避免加载符号?

VS2013调试代码时怎么避免加载符号?VS2013调试代码时候,当电脑接入网络后系统会自动从Microsoft 符号服务器加载PDB符号文件,一旦加载符号就会变得很慢,下面我们来看看详细的解决办法
收藏 0 赞 0 分享

eclipse怎么更换工作空间?eclipse更换工作区的方法

eclipse怎么更换工作空间?eclipse经常编辑代码,eclipse默认的的工作区在C盘,经常使用eclipse写代码会导致C盘空间缩小,该怎么更换工作区呢?下面我们来看看eclipse更换工作区的方法
收藏 0 赞 0 分享

vs2010怎么更换默认的起始页?vs2010起始页添加命令按钮的方法

vs2010怎么更换默认的起始页?vs2010起始页就是vs启动的时候默认打开的页面,当然我们可以根据自己的需要来设置起始页,下面我们来看看vs2010起始页添加命令按钮的方法
收藏 0 赞 0 分享

eclipse中sdk与adt版本不兼容该怎么解决? 三种方法帮你搞定

eclipse中sdk与adt版本不兼容该怎么解决?出现这种问题基本上是因为sdk版本过高,今天我们来看看三种比较简单的解决办法,图文教程很简单,需要的朋友可以参考下
收藏 0 赞 0 分享

vb怎么输出菱形图案?vb绘制菱形的教程

vb怎么输出菱形图案?vb编辑的代码可以实现很多功能,今天我们就来看看怎么使用vb绘制菱形图案,这篇教程很简单,适合新手学习,需要的朋友可以参考下
收藏 0 赞 0 分享

Android Studio怎么导出设置?

Android Studio怎么导出设置?Android Studio中设置了很多符合自己习惯的东西,但是如果重新安装软件,之前设置的东西都会消失,该怎么将这些设置导出来呢?下面我们一起来看看详细教程
收藏 0 赞 0 分享

怎么使用arcgis对进行地图投影?

怎么使用arcgis对进行地图投影?没有投影过的地图该怎么投影?我们想将是西安80的shp数据,但是一般的搜不带号,想转换成带号的,就需要使用arcgis对进行地图投影,详细教程请看下文
收藏 0 赞 0 分享
查看更多