装箱拆箱续
原来写了一篇有关装箱拆箱的文章,这几天看到这个例子,觉得更能说明一些问题,首先看下面的例子
namespace InBoxUnInBoxTest
{
internal struct Point
{
private int m_x, m_y;
public Point(int x, int y)
{
m_x = x;
m_y = y;
}
public void Change(int x, int y)
{
m_x = x;
m_y = y;
}
public override string ToString()
{
return String.Format("({0},{1})", m_x, m_y);
}
}
class Program
{
static void Main(string[] args)
{
Point p = new Point(1, 1);
Console.WriteLine(p);
p.Change(2, 2);
Console.WriteLine(p);
object o = p;
Console.WriteLine(o);
((Point)o).Change(3, 3);
Console.WriteLine(o);
}
}
}
其结果是:
(1,1)
(2,2)
(2,2)
(2,2)
前三个输出没有什么问题,来说说第四个输出:o 是一个 Object 类型,首先进行的是拆箱的操作。这时系统就要把 o 复制到一堆栈上的一个临时空间中。对象 o 本身还是存储在 托管对上的,所以这是调用 Change 函数对于 o 来说没有起任何作用。所以才出现了上述结果。
再来看看下面的程序:
namespace InBoxTest
{
internal interface IChangeBoxedPoint
{
void Change(int x, int y);
}
internal struct Point : IChangeBoxedPoint
{
private int m_x, m_y;
public Point(int x, int y)
{
m_x = x;
m_y = y;
}
public void Change(int x, int y)
{
m_x = x;
m_y = y;
}
public override string ToString()
{
return String.Format("({0},{1})", m_x, m_y);
}
}
class Program
{
static void Main(string[] args)
{
Point p = new Point(1, 1);
Console.WriteLine(p);
p.Change(2, 2);
Console.WriteLine(p);
object o = p;
Console.WriteLine(o);
((Point)o).Change(3, 3);
Console.WriteLine(o);
((IChangeBoxedPoint)p).Change(4, 4);
Console.WriteLine(p);
((IChangeBoxedPoint)o).Change(5, 5);
Console.WriteLine(o);
Console.Read();
}
}
}
其结果是:
(1,1)
(2,2)
(2,2)
(2,2)
(2,2)
(5,5)
主要来看看最后两个结果:在这个程序中定义了一个接口,自定义的结构继承自这个类型。第五个结果输出的是(2,2),程序中进行的是装箱处理,这样 p 就以引用类型存储形势被复制到托管对上,这样在进行修改,实际上只是修改托管对上的对象。 p 本身没有变化。如果我们再看看托管对上的对象是否进行修改了,加入如下代码测试一下:
IChangeBoxedPoint test = (IChangeBoxedPoint)p;
test.Change(4, 4);
Console.WriteLine(p);
Console.WriteLine(test);
其结果:(4,4)显示出来了
然后再来看看最后一个结果,系统对对象 o 实际上就是强制类型的转换,并不是装箱拆箱的操作。所以操作的还是 o 本身。
最后说一句:装拆箱要在堆栈和托管对上进行对象复制(这当中包括值类型和引用类型存储形式的变化),性能和安全性降低,所以在系统开发中不应该大量使用,当然我想可以尽量使用泛形来解决问题。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构