青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

C++分析研究  
C++
日歷
<2014年3月>
2324252627281
2345678
9101112131415
16171819202122
23242526272829
303112345
統計
  • 隨筆 - 92
  • 文章 - 4
  • 評論 - 4
  • 引用 - 0

導航

常用鏈接

留言簿

隨筆檔案

文章檔案

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

 

  boost.property_tree可以用來解析xml和json文件,我主要用它來解析xml文件,它內部封裝了號稱最快的xml解析器rapid_xml,其解析效率還是很好的。但是在使用過程中卻發現各種不好用,歸納一下不好用的地方有這些:

  獲取不存在的節點時就拋出異常

  獲取屬性值時,要排除屬性和注釋節點,如果沒注意這一點就會拋出異常,讓人摸不著頭腦。

  內存模型有點怪。

  默認不支持中文的解析。解析中文會亂碼。

  ptree獲取子節點

  獲取子節點接口原型為get_child(node_path),這個node_path從當前路徑開始的全路徑,父路徑和子路徑之間通過“.”連接,如“root.sub.child”。需要注意的是get_child獲取的是第一個子節點,如果我們要獲取子節點列表,則要用路徑“root.sub”,這個路徑可以獲取child的列表。如果獲取節點的路徑不存在則會拋出異常,這時,如果不希望拋出異常則可以用get_xxx_optional接口,該接口返回一個optional的結果出來,由外面判斷是否獲取到結果托福答案 www.jamo123.com

  //ptree的optional接口

  auto item = root.get_child_optional("Root.Scenes");

  該接口返回的是一個optional,外面還要判斷該節點是否存在,optional對象通過bool操作符來判斷該對象是否是無效值,通過指針訪問

  符"*"來訪問該對象的實際內容。建議用optional接口訪問xml節點。

  //ptree的optional接口

  auto item = root.get_child_optional("Root.Scenes");

  if(item)

  cout<<"該節點存在"<

  ptree的內存模型

  ptree維護了一個pair的子節點列表,first指向的是該節點的TagName,second指向的才是ptree節點,因此在遍歷ptree子節點時要注意迭代器的含義。

  for (auto& data : root)

  {

  for (auto& item : data.second) //列表元素為pair,要用second繼續遍歷

  {

  cout<

  }

  }

  需要注意的是ptree.first可能是屬性("")也可能是注釋(""),只有非注釋類型的節點才能使用獲取屬性值、子節點等常用接口。

  ptree獲取屬性值

  通過get(attr_name)可以獲取屬性的值,如果想獲取屬性的整形值的話,可以用get("Id"),返回一個整數值。有一點要注意如果ptree.first為""時,是沒有屬性值的,可以通過data()來獲取注釋內容。如果這個ptree.first不為時需要在屬性名稱前面加".",即get(".Id")才能正確獲取屬性值。可以看到獲取屬性值還是比較繁瑣的,在后面要介紹的幫助類中可以簡化屬性值的獲取。如果要獲取節點的值則用get_value()接口,該接口用來獲取節點的值,如節點:2通過get_value()就可以獲取值"2"。

  解析中文的問題

  ptree只能解析窄字符的xml文件,如果xml文件中含有unicode如中文字符,解析出來就是亂碼。解析unicode要用wptree,該類的接口均支持寬字符并且接口和ptree保持一致。要支持中文解析僅僅wptree還不夠,還需要一個unicode轉換器的幫助,該轉換器可以實現寬字符和窄字符的轉換,寬窄的互相轉換函數有很多實現,不過c++11中有更簡單統一的方式實現寬窄字符的轉換。

  c++11中寬窄字符的轉換:

  std::wstring_convert> conv

  (newstd::codecvt("CHS"));

  //寬字符轉為窄字符

  string str = conv.to_bytes(L"你好");

  //窄字符轉為寬字符

  string wstr = conv.from_bytes(str);

  boost.property_tree在解析含中文的xml文件時,需要先將該文件轉換一下。

  boost解決方法:

  #include "boost/program_options/detail/utf8_codecvt_facet.hpp"

  void ParseChn()

  {

  std::wifstream f(fileName);

  std::locale utf8Locale(std::locale(), new boost::program_options::detail::utf8_codecvt_facet());

  f.imbue(utf8Locale); //先轉換一下

  //用wptree去解析

  property_tree::wptree ptree;

  property_tree::read_xml(f, ptree);

  }

  這種方法有個缺點就是要引入boost的libboost_program_options庫,該庫有二十多M,僅僅是為了解決一個中文問題,卻要搞得這么麻煩,有點得不償失。好在c++11提供更簡單的方式,用c++11可以這樣:

  void Init(const wstring& fileName, wptree& ptree)

  {

  std::wifstream f(fileName);

  std::locale utf8Locale(std::locale(), new std::codecvt_utf8);

  f.imbue(utf8Locale); //先轉換一下

  //用wptree去解析

  property_tree::read_xml(f, ptree);

  }

  用c++11就不需要再引入boost的libboost_program_options庫了,很簡單。

  property_tree的幫助類

  property_tree的幫助類解決了前面提到的問題:

  用c++11解決中文解析問題

  簡化屬性的獲取

  增加一些操作接口,比如一些查找接口

  避免拋出異常,全部返回optional對象

  隔離了底層繁瑣的操作接口,提供統一、簡潔的高層接口,使用更加方便。

  下面來看看這個幫助類是如何實現的吧:

  #include

  #include

  using namespace boost;

  using namespace boost::property_tree;

  #include

  #include

  #include

  #include

  using namespace std;

  const wstring XMLATTR = L"";

  const wstring XMLCOMMENT = L"";

  const wstring XMLATTR_DOT = L".";

  const wstring XMLCOMMENT_DOT = L".";

  class ConfigParser

  {

  public:

  ConfigParser() : m_conv(new code_type("CHS"))

  {

  }

  ~ConfigParser()

  {

  }

  void Init(const wstring& fileName, wptree& ptree)

  {

  std::wifstream f(fileName);

  std::locale utf8Locale(std::locale(), new std::codecvt_utf8);

  f.imbue(utf8Locale); //先轉換一下

  wcout.imbue(std::locale("chs")); //初始化cout為中文輸出格式

  //用wptree去解析

  property_tree::read_xml(f, ptree);

  }

  // convert UTF-8 string to wstring

  std::wstring to_wstr(const std::string& str)

  {

  return m_conv.from_bytes(str);

  }

  // convert wstring to UTF-8 string

  std::string to_str(const std::wstring& str)

  {

  return m_conv.to_bytes(str);

  }

  //獲取子節點列表

  auto Descendants(const wptree& root, const wstring& key)->decltype(root.get_child_optional(key))

  {

  return root.get_child_optional(key);

  }

  //根據子節點屬性獲取子節點列表

  template

  vector GetChildsByAttr(const wptree& parant, const wstring& tagName, const wstring& attrName, const T& attrVal)

  {

  vector v;

  for (auto& child : parant)

  {

  if (child.first != tagName)

  continue;

  auto attr = Attribute(child, attrName);

  if (attr&&*attr == attrVal)

  v.push_back(child.second);

  }

  return v;

  }

  //獲取節點的某個屬性值

  template

  optional Attribute(const wptree& node, const wstring& attrName)

  {

  return node.get_optional(XMLATTR_DOT + attrName);

  }

  //獲取節點的某個屬性值,默認為string

  optional Attribute(const wptree& node, const wstring& attrName)

  {

  return Attribute(node, attrName);

  }

  //獲取value_type的某個屬性值

  template

  optional Attribute(const wptree::value_type& pair, const wstring& attrName)

  {

  if (pair.first == XMLATTR)

  return pair.second.get_optional(attrName);

  else if (pair.first == XMLCOMMENT)

  return optional();

  else

  return pair.second.get_optional(XMLATTR_DOT + attrName);

  }

  //獲取value_type的某個屬性值,默認為string

  optional Attribute(const wptree::value_type& pair, const wstring& attrName)

  {

  return Attribute(pair, attrName);

  }

  //根據某個屬性生成一個的multimap

  template>

  multimap MakeMapByAttr(const wptree& root, const wstring& key, const wstring& attrName, F predict = [](wstring& str){return true; })

  {

  multimap resultMap;

  auto list = Descendants(root, key);

  if (!list)

  return resultMap;

  for (auto& item : *list)

  {

  auto attr = Attribute(item, attrName);

  if (attr&&predict(*attr))

  resultMap.insert(std::make_pair(*attr, item.second));

  }

  return resultMap;

  }

  private:

  using code_type = std::codecvt;

  std::wstring_convert m_conv;

  };

  View Code

  測試文件test.xml和測試代碼:

  void Test()

  {

  wptree pt; pt.get_value()

  ConfigParser parser;

  parser.Init(L"test1.xml", pt); //解決中文問題,要轉換為unicode解析

  auto scenes = parser.Descendants(pt, L"Root.Scenes"); //返回的是optional

  if (!scenes)

  return;

  for (auto& scene : *scenes)

  {

  auto s = parser.Attribute(scene, L"Name"); //獲取Name屬性,返回的是optional

  if (s)

  {

  wcout << *s << endl;

  }

  auto dataList = parser.Descendants(scene.second, L"DataSource"); //獲取第一個子節點

  if (!dataList)

  continue;

  for (auto& data : *dataList)

  {

  for (auto& item : data.second)

  {

  auto id = parser.Attribute(item, L"Id");

  auto fileName = parser.Attribute(item, L"FileName");

  if (id)

  {

  wcout << *id << L" " << *fileName << endl; //打印id和filename

  }

  }

  }

  }

  }

  測試結果:


  可以看到通過幫助類,無需使用原生接口就可以很方便的實現節點的訪問與操作。使用者不必關注內部細節,根據統一而簡潔的接口就可以操作xml文件了。

  一點題外話,基于這個幫助類再結合linq to object可以輕松的實現linq to xml:

  //獲取子節點SubNode的屬性ID的值為0x10000D的項并打印出該項的Type屬性

  from(node.Descendants("Root.SubNode")).where([](XNode& node)

  {

  auto s = node.Attribute("ID");

  return s&&*s == "0x10000D";

  }).for_each([](XNode& node)

  {

  auto s = node.Attribute("Type");

  if (s)

  cout << *s << endl;

  });

posted on 2014-03-11 16:27 HAOSOLA 閱讀(5148) 評論(0)  編輯 收藏 引用
 
Copyright © HAOSOLA Powered by: 博客園 模板提供:滬江博客
PK10開獎 PK10開獎
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            欧美高清在线| 欧美国产日产韩国视频| 亚洲国产美女精品久久久久∴| 欧美日韩伦理在线免费| 免费不卡视频| 樱桃视频在线观看一区| 欧美日本在线播放| 亚洲大胆女人| 欧美精品久久久久久| 欧美a级大片| 欧美午夜一区二区福利视频| 亚洲国产欧美一区| 亚洲性夜色噜噜噜7777| 国产精品永久免费视频| 欧美在线二区| 欧美丰满高潮xxxx喷水动漫| 久久久免费精品| 国产精品免费久久久久久| 亚洲精品一区二区三区樱花| 久久综合色88| 久久精品人人做人人爽电影蜜月 | 欧美中文字幕不卡| 亚洲精品一区二区三区蜜桃久| 欧美一区二区在线视频| 在线观看欧美精品| 久久综合国产精品| 久久夜色精品国产| 免费国产一区二区| 久久全国免费视频| a4yy欧美一区二区三区| 欧美亚洲午夜视频在线观看| 欧美亚洲一区在线| 欧美一区二区视频在线观看| 91久久国产精品91久久性色| 国产欧美精品va在线观看| 欧美久久综合| 国产精品久久久对白| 狠狠色丁香久久婷婷综合丁香| 国语自产在线不卡| 99伊人成综合| 久久久久久久波多野高潮日日| 蜜臀av性久久久久蜜臀aⅴ| 欧美激情视频免费观看| 夜夜爽夜夜爽精品视频| 午夜视频久久久| 欧美激情视频给我| 国产亚洲一二三区| 99精品欧美一区二区三区综合在线| 一区二区三区四区在线| 女人色偷偷aa久久天堂| 亚洲欧美第一页| 欧美激情1区2区3区| 国产在线精品一区二区夜色| 亚洲欧美日韩一区二区| 亚洲电影自拍| 久久综合电影| 久久精品在线观看| 欧美一区二区久久久| 亚洲午夜极品| 欧美午夜在线观看| 亚洲一二三级电影| 亚洲视频在线观看一区| 欧美日韩亚洲成人| 亚洲一区二区四区| 亚洲午夜电影网| 欧美三级在线| 欧美亚洲三区| 狂野欧美激情性xxxx| 日韩视频精品在线观看| 欧美亚洲一级| 欧美女同在线视频| 久久久久久穴| 欧美激情免费在线| 午夜一区在线| 久久婷婷蜜乳一本欲蜜臀| 最新日韩在线视频| 久久成人久久爱| 亚洲精品国产精品国自产观看浪潮| 欧美成年人网站| 欧美精品 国产精品| 在线观看精品| 99热在线精品观看| 国产日韩欧美综合一区| 欧美激情综合色| 国产亚洲激情视频在线| 欧美激情精品久久久久久黑人| 欧美精品免费看| 久久久国际精品| 欧美日韩精品综合| 亚洲在线观看视频| 国产精品高潮久久| 夜夜嗨一区二区| 亚洲国产精品尤物yw在线观看| 亚洲视频播放| 亚洲视频在线观看三级| 久久躁日日躁aaaaxxxx| 亚洲午夜激情在线| 麻豆精品传媒视频| 夜夜嗨av一区二区三区四季av | 亚洲黄色在线视频| 亚洲精品一区二区三区樱花| 国产亚洲激情在线| 香蕉久久夜色精品国产| 妖精视频成人观看www| 亚洲欧美日韩精品久久久久| 欧美激情亚洲国产| 国产在线精品自拍| 亚洲午夜精品一区二区| 亚洲欧美国产高清va在线播| 国产精品有限公司| 亚洲午夜影视影院在线观看| 亚洲一区二区四区| 欧美日韩精品久久| 亚洲欧美中文另类| 一本一本a久久| 国产精品一区二区久久久| 一区二区高清在线| 久久久夜夜夜| 欧美精选在线| 亚洲欧美激情精品一区二区| 久久久久久久999精品视频| 亚洲国产裸拍裸体视频在线观看乱了中文 | 久久久视频精品| 亚洲激情影院| 亚洲欧美日韩精品久久久| 国产人成精品一区二区三| 久久久亚洲欧洲日产国码αv| 亚洲国产日本| 欧美一区国产在线| 日韩亚洲视频| 国产综合久久久久久| 国产精品久久久一区二区三区| 欧美大片在线看| 久久天堂国产精品| 久久久久久久久一区二区| 亚洲国产高清aⅴ视频| 国产美女精品视频免费观看| 欧美激情第五页| 欧美激情综合色| 欧美顶级大胆免费视频| 老鸭窝91久久精品色噜噜导演| 亚洲一区二区三区中文字幕在线| 黄色资源网久久资源365| 久久精品国产99| 久久爱www.| 欧美人与性动交a欧美精品| 免费不卡在线观看| 欧美精品一区二区三区久久久竹菊| 欧美成人精品三级在线观看 | 亚洲精品日韩一| 亚洲第一网站免费视频| 91久久极品少妇xxxxⅹ软件| 亚洲国产合集| 亚洲伊人色欲综合网| 欧美一级午夜免费电影| 老司机一区二区| 亚洲视频在线观看三级| 久久人人爽国产| 欧美精品亚洲一区二区在线播放| 欧美日韩国产影片| 国产日韩欧美综合在线| 99在线精品视频| 亚洲国产精品久久久| 亚洲天堂免费在线观看视频| 久久资源在线| 国产精品www网站| 99国内精品久久久久久久软件| 欧美一级久久久| 亚洲国产精品成人综合色在线婷婷| 亚洲网站在线| 欧美国产一区二区在线观看| 国产精品综合色区在线观看| 国产精品美女久久| 午夜伦理片一区| 欧美日韩免费在线观看| 亚洲精品国产系列| 母乳一区在线观看| 亚洲毛片在线观看.| 欧美激情aaaa| 亚洲私人影吧| 亚洲摸下面视频| 欧美精品日韩www.p站| 亚洲第一中文字幕在线观看| 久久久久综合一区二区三区| 久久人体大胆视频| 亚洲激情偷拍| 亚洲一区激情| 韩国三级在线一区| 欧美成人激情视频| 欧美日本国产精品| 亚洲欧美一区二区三区久久| 亚洲欧美日韩另类| 在线观看一区视频| 亚洲欧美中文另类| 亚洲精品乱码久久久久| 久久婷婷色综合| 亚洲精品视频二区| 午夜国产不卡在线观看视频| 伊人久久婷婷| 久久久久久久精|