敬业的IT人 >> 编程开发 >> C/C++ >> C++/CLI的“值类型的强类型装箱实例”

C++/CLI的“值类型的强类型装箱实例”

敬业的IT人 互联网 佚名 2008-1-3 19:44:02

  近来接到几个朋友问Visual C++2005 (C++/CLI) Webcast中讲的“值类型的强类型装箱实例”是什么?

  讲座比较匆忙,因此对这个技术点只是点了一下,没有详细展开。这里借blog把它展开说一下。

  首先来看下面的C#代码:

using System;
using System.Collections;
struct MyClass
{
  public  int data;
        
}
class Test
{
  public static void Main()
  {
    MyClass myClass1 = new MyClass();
    MyClass myClass2=new MyClass();
    ArrayList list=new ArrayList();
    list.Add(myClass1);
    list.Add(myClass2);
    Print(list);
    for(int i=0;i<list.Count;i++)
    {
      MyClass temp=(MyClass)list[i];
      temp.data=i+1;
      list[i]=temp;// 注意这句话
    }
    Print(list);
  }
  public static void Print(ArrayList list)
  {
    for(int i=0;i<list.Count;i++)
    {
      MyClass temp=(MyClass)list[i];
      Console.WriteLine(temp.data);
    }
  }
    
}

  其中list[i]=temp这句话很重要,否则无法实施改变,因为temp是一个“和list中hold的boxed的值实例无关的”stack上的实例。

  但是如果使用C++/CLI,我们就可以这么做:

using namespace System;
using namespace System::Collections;
value class MyClass {
public:
  int data;
};
void Print(ArrayList^ list);
int main() {
  MyClass^ hMyClass1 = gcnew MyClass;
  MyClass^ hMyClass2 = gcnew MyClass;
  ArrayList^ list=gcnew ArrayList();
  list->Add(hMyClass1);
  list->Add(hMyClass2);
  Print(list);
  for(int i=0;i<list->Count;i++)
  {
    MyClass^ temp=(MyClass^)list[i];
    temp->data=i+1; // 注意这里无需再有list[i]=temp这句话,因为temp引用的就是“list中hold的boxed的值实例”
  }
  Print(list);
}
void Print(ArrayList^ list)
{
  for(int i=0;i<list->Count;i++)
  {
    MyClass^ temp=(MyClass^)list[i];
    Console::WriteLine(temp->data);
  }
}

  其实在C#中,我们使用MyClass temp=(MyClass)list[i]; 中间就发生了一个unbox和一个copy动作。

文章地址: 进入讨论组讨论。

  而在C++/CLI中,我们使用MyClass^ temp=(MyClass^)list[i]; 中间只发生了unbox,而没有发生copy动作——注意只发生unbox,而数据还位于managed heap中,但是类型却是MyClass^,这样我们就可以直接获取MyClass的data成员——这就是我所说的“值类型的强类型装箱实例”。

  这在C#中是不能做到的,只能获得类型为Object的一个“弱类型”——有一种办法是采用interface来间接实现,因为interface是一个引用类型。

  “值类型的强类型装箱实例”当然不仅仅意味着“少写代码”,实际上它带来很好的性能提升——因为unbox代价很小,与copy动作是分离的。不像box,它是代价比较高的操作,因为本身包含copy动作。

  其实C++/CLI对类型的强大描述能力不仅仅体现在这一个地方,还有很多,比如interior_ptr,这将是我今后将要剖析C++/CLI的一个重点, 也是我去年撰写“漫谈C++/CLI中的几种指针和引用(1)”(http://blog.dreambrook.com/jzli/archive/2004/11/20/352.aspx)系列的规划——可惜后来由于太忙,没有集中的时间展开。展开讨论它要相当大的篇幅,因为它牵扯到对托管对象模型的深入剖析。下面我会随着自己对Shared Source CLI (Rotor)源代码的剖析,来展开这一系列的blog。

文章地址: 进入讨论组讨论。
粤ICP备06119539号
Copyright CiscoSky.Org,Some Rights Reserved.
Email:me1228#tom.com