大部分人用C++/CLI都是為了包裝,甚至有人說(shuō)用托管C++的時(shí)候充滿了噪音,什么時(shí)候要用pin_ptr,什么時(shí)候用interor_ptr,什么時(shí)候value class,什么時(shí)候ref class,用prediction的時(shí)候還不能用lambda,在managed code來(lái)用不了lambda,new/gcnew,&和%,*和^...有沒(méi)有帶著鐐銬跳舞的感覺(jué)?
如果這叫噪音的話,是不是因?yàn)槲覀円恢北蝗诉^(guò)分地照顧所致?
在此,分享幾個(gè)tips,讓我們逐漸去除編程語(yǔ)言中的那些噪音.
typeof
csharp中有個(gè)typeof,c++/cli則沒(méi)有,只有一個(gè)typeid,而且是被智能提示所忽略.
比如:
Type^ t0 = int::typeid;
...
ref class people
{}
Type^ t1 = people::typeid;
不僅基礎(chǔ)類型,而且可以是自定義類型.
定義如下模版,可以模擬出類似C#中的typeof:
template<typename T>
inline System::Type^ typeof() { return T::typeid; }
template<typename T>
inline System::Type^ typeof(T t) { return T::typeid; }
可以用handle實(shí)例來(lái)取得Type:
JiaoAnDataBag^ hdl = nullptr;
System::Console::WriteLine(typeof(hdl));
System::Console::WriteLine(typeof<JiaoAnDataBag>()->FullName);
從實(shí)例得到Type^,不能用->,因?yàn)閷?shí)例可能是無(wú)效的空handle,但利用模版則可以,如
JiaoAnDataBag^ hdl = nullptr;
Type^ t = hdl->GetType(); //crash
從列表初始化CLI容器
在C#中,從列表可以初始化為CLI容器:
var arr =new List<int>{1,2,3,4};
但在C++/CLI中,花括號(hào)代表為native中的初始化列表.對(duì)應(yīng)于std::initializer_list,這是一個(gè)native class,在List<T>里沒(méi)有這個(gè)構(gòu)造函數(shù).不得不gcnew List<T>以后,通過(guò)Add加入到集合.
下面這個(gè)輔助函數(shù)可以做到類似的效果:
template<typename T>
auto ArrayToList(... cli::array<T>^ arr)
{
auto ret = gcnew List<T>;
for each(auto cur in arr)
{
ret->Add(cur);
}
return ret;
}
//使用方法
List<int>^ arr = ArrayToList(1,2,3,4);
考慮到擴(kuò)展,可以增加類型:
template<typename T,typename Coll_t = System::Collections::Generic::List<T>>
auto ArrayToList(... cli::array<T>^ arr)
{
auto ret = gcnew Coll_t;
for each(auto cur in arr)
{
ret->Add(cur);
}
return ret;
}
是不是有點(diǎn)殊途同歸的味道了.你還可以享受自己造輪子的樂(lè)趣.猶如智力游戲.
void foo(List<people^>^ peoples)
{
...
}
people^ p0 = ...;
people^ p1 = ...;
people^ p2 = ...;
foo(ArrayToList(p0,p1,p2));
請(qǐng)享用!