C++/CLI中小心得 之一

大部分人用C++/CLI都是为了包装,甚至有人说用托管C++的时候充满了噪音,什么时候要用pin_ptr,什么时候用interor_ptr,什么时候value class,什么时候ref class,用prediction的时候还不能用lambda,在managed code来用不了lambda,new/gcnew,&和%,*和^...有没有带着镣铐跳舞的感觉?
如果这叫噪音的话,是不是因为我们一直被人过分地照顾所致?
在此,分享几个tips,让我们逐渐去除编程语言中的那些噪音.

typeof

csharp中有个typeof,c++/cli则没有,只有一个typeid,而且是被智能提示所忽略.
比如:

Type^ t0 = int::typeid;
...
ref class people
{}
Type^ t1 = people::typeid;

不仅基础类型,而且可以是自定义类型.
定义如下模版,可以模拟出类似C#中的typeof:

template
inline System::Type^ typeof() { return T::typeid; }

template
inline System::Type^ typeof(T t) { return T::typeid; }

可以用handle实例来取得Type:

JiaoAnDataBag^ hdl = nullptr;
System::Console::WriteLine(typeof(hdl));
System::Console::WriteLine(typeof()->FullName);

从实例得到Type^,不能用->,因为实例可能是无效的空handle,但利用模版则可以,如

JiaoAnDataBag^ hdl = nullptr;
Type^ t = hdl->GetType();  //crash

从列表初始化CLI容器

在C#中,从列表可以初始化为CLI容器:

var arr =new List{1,2,3,4};

但在C++/CLI中,花括号代表为native中的初始化列表.对应于std::initializer_list,这是一个native class,在List里没有这个构造函数.不得不gcnew List以后,通过Add加入到集合.
下面这个辅助函数可以做到类似的效果:

template
auto ArrayToList(... cli::array^ arr)
{
    auto ret = gcnew List;
    for each(auto cur in arr)
    {
        ret->Add(cur);
    }
    return ret;
}
//使用方法
List^ arr = ArrayToList(1,2,3,4);

考虑到扩展,可以增加类型:

template>
auto ArrayToList(... cli::array^ arr)
{
    auto ret = gcnew Coll_t;
    for each(auto cur in arr)
    {
        ret->Add(cur);
    }
    return ret;
}

是不是有点殊途同归的味道了.你还可以享受自己造轮子的乐趣.犹如智力游戏.

void foo(List^ peoples)
{
 ...
}
people^ p0 = ...;
people^ p1 = ...;
people^ p2 = ...;
foo(ArrayToList(p0,p1,p2));

请享用!

你可能感兴趣的:(C++/CLI中小心得 之一)