大部分人用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;
不僅基礎(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實例來取得Type:
JiaoAnDataBag^ hdl = nullptr;
System::Console::WriteLine(typeof(hdl));
System::Console::WriteLine(typeof<JiaoAnDataBag>()->FullName);
從實例得到Type^,不能用->,因為實例可能是無效的空handle,但利用模版則可以,如
JiaoAnDataBag^ hdl = nullptr;
Type^ t = hdl->GetType(); //crash
從列表初始化CLI容器
在C#中,從列表可以初始化為CLI容器:
var arr =new List<int>{1,2,3,4};
但在C++/CLI中,花括號代表為native中的初始化列表.對應于std::initializer_list,這是一個native class,在List<T>里沒有這個構(gòu)造函數(shù).不得不gcnew List<T>以后,通過Add加入到集合.
下面這個輔助函數(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);
考慮到擴展,可以增加類型:
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;
}
是不是有點殊途同歸的味道了.你還可以享受自己造輪子的樂趣.猶如智力游戲.
void foo(List<people^>^ peoples)
{
...
}
people^ p0 = ...;
people^ p1 = ...;
people^ p2 = ...;
foo(ArrayToList(p0,p1,p2));
請享用!