Bootstrap

C#的装箱和拆箱

一、装箱和拆箱概念

装箱:将值类型转换为引用类型称为装箱
拆箱:将引用类型转换为值类型称为拆箱

1、在C#中,值类型的数据被保存在栈(stack)上,而引用类型的数据被保存在堆(heap)上,在栈上只是保存了这些数据在堆内存中的首地址

using Ststem.Collections;
using System.Collections.Generic;
using UnityEngine;

class Animal
{
	public int age;

	public Animal(int age)
	{
		this.age = age;
	}
}

public class s1 : MonoBehaviour {
	void Start () {

	int x = 3;
	Vector3 v = new Vector3(1,2,3);
	Animal a = new Animal(3);
	
	void Update () {
	
	}
}

以上这段在Unity中的C#代码,在Start()中定义了三个变量,分别是整型x、向量类型v、Animal类型a。这3中类型的初始化方式,从表面上看,v和a是相同的,都是用new运算符进行初始化变量的,而变量x是直接进行赋值的。但实际上,在数据的存储方式上,反而是x和v是相同的,它们都是存储在栈上的值类型,而只有Animal a是存储在堆上的引用类型
图示

2、装箱拆箱的时机

using Ststem.Collections;
using System.Collections.Generic;
using UnityEngine;

class Animal
{
	public int age;

	public Animal(int age)
	{
		this.age = age;
	}
}

public class s2 : MonoBehaviour {
	void Start () {

	int x = 12;
	print(x);

	print(new Vector3(1,1,1));
	
	print("Metaverse");
	
	print(new Animal(3));
	
	void Update () {
	
	}

主要用print方法来向console窗口打印不同类型的数据。首先print()方法的参数是一个object类型,object类型是其他所有类型的父类,因此任意一个C#中的变量都满足is a objcet的关系,所以print方法可以用于打印所有类型的数据。

第1个print语句用来打印x变量,x是一个int类型的数据,这个调用会将int类型转换为object类型,因此是将值类型转换为了引用类型,属于装箱。

第2个print语句用来打印了一个向量,在Unity中,Vector是用struct定义的,因此属于值类型,这里依然是将值类型转换为引用类型。

第3个print语句打印了一个Animal类型的变量,用class定义Animal,因此是引用类型。这里将Animal类型转换为object类型是类型的转换,而不是装箱

二、总结

1、普遍的场景是调用一个含类型为Objcet的参数的方法,该Object可支持任意类型,以便通用。当你需要将一个值类型传入时,需要装箱。

2、一个非泛型的容器,同样是为了保证通用,而将元素类型定义为Objcet。于是,要将值类型数据加入容器,需要装箱。

装箱时生成的是全新的引用对象,这就会有时间损耗,造成效率降低,所以应该尽量避免装箱。前面提到的两种情况,都可以避免,在第1中情况下,可以通过重载函数来避免,而第2种情况则可以通过泛型来避免。

;