算半個作業文吧 (?) 不過是寫專案的時候碰到的問題
我們有一組Data Access Layer,之前為了讓他能方便吃下所有型態
所以Value的部分用boost::variant,如下這樣
typedef boost::variant<std::string, int, char*> ValueType;
ValueType get(const std::string&& key);
void set(const std::string&& key, ValueType value);
然後把它們存在一組map裡面
std::map<std::string, ValueType> map;
當然使用上並沒有想像中那麼方便啦,這是後話(尤其是get,真的是歐麥尬)
不過看起來也算能用就是了
對於boost::variant使用比較熟的同學應該就知道他能吃三個值了
(本例來講就是std::string, int, char*)
但是在做serialization的時候,發現boost::serialization對這種case簡直無解
std::stringstream ss;
boost::archive::text_oarchive ar(ss);
ar << map;
std::cout << ss.str() << std::endl;
會跑出一大組壯觀的錯誤,當然,身為專業的C++碼農,本來就不期待boost無痛搞好
所以就開始自己認命的寫serialization了
void SimpleDevDBSpace::serialize(const std::string&& folder) {
for(auto i : map) {
//Use reference to improve performance
const std::string& name = i.first;
ValueType& value = i.second;
switch(value.which()) {
當然這對大多數人來講都已經good enough了,但是我總覺得心裡癢癢的
value.watch()傳回來的是order,也就是0 = string, 1 = int, 2 = char*
但是這不夠泛用,如果今天把variant型別改成<int, std::string, float, long>
這樣的話還要記得去改value.which的case handle,這顯然是有點問題的
所以我的需求是,有沒有什麼方法可以讓std::string, int, char* 成為一個array
讓我需要增加支援type時,改一個地方即可,有點像這樣
TypeArray ARR = {std::string, int, char*};
typedef boost::variant<{ARR}> ValueType;
而且這個可以被輪詢
我知道value還額外一個提供value.type(),不過我想不出他對我的需求有什麼用
他用了一個很類似loki的Type2Int的方法,但是我找不到公開方法可以去轉
有人可以給我一點建議嗎?雖然我覺得這需求已經有點龜毛了.....