一、簡單介紹
JSON 的全稱爲:JavaScript Object Notation,JSON 是用於標記 Javascript 對象的,JSON 官方的解釋爲:JSON 是一種輕量級的數據傳輸格式。
二、第三方庫
jsoncpp:它 是比較出名的 C++ JSON 解析庫。地址:http://sourceforge.net/projects/jsoncpp
三、使用方法
jsoncpp 主要包含三個class:Value、Reader、Writer。jsoncpp 中所有對象、類名都在 namespace Json 中,只需要包含 json.h 即可。注意Json::Value 只能處理 ANSI 類型的字符串,如果 C++ 程序是用 Unicode 編碼的,最好加一個 Adapt 類來適配。
1.Json::Value 可以表示裏所有的類型,比如int,string,object,array等。
Json::Value使用
Json::Value json_temp; // 臨時對象,供如下代碼使用
json_temp["name"] = Json::Value("huchao");
json_temp["age"] = Json::Value(26);
Json::Value root; // 表示整個 json 對象
root["key_string"] = Json::Value("value_string"); // 新建一個 Key(名爲:key_string),賦予字符串值:"value_string"。
root["key_number"] = Json::Value(12345); // 新建一個 Key(名爲:key_number),賦予數值:12345。
root["key_boolean"] = Json::Value(false); // 新建一個 Key(名爲:key_boolean),賦予bool值:false。
root["key_double"] = Json::Value(12.345); // 新建一個 Key(名爲:key_double),賦予 double 值:12.345。
root["key_object"] = Json_temp; // 新建一個 Key(名爲:key_object),賦予 json::Value 對象值。
root["key_array"].append("array_string"); // 新建一個 Key(名爲:key_array),類型爲數組,對第一個元素賦值爲字符串:"array_string"。
root["key_array"].append(1234); // 爲數組 key_array 賦值,對第二個元素賦值爲:1234。
Json::ValueType type = root.type(); // 獲得 root 的類型,此處爲 objectValue 類型。
注:跟C++ 不同,JavaScript 數組可以爲任意類型的值,所以 jsoncpp 也可以。
如上幾個用法已經可以滿足絕大部分 json 應用了,當然 jsoncpp 還有一些其他用法,比如說設置註釋、比較 json 大小、交換 json 對象等。
2.Json::Reader 將json文件流或字符串解析到Json::Value, 主要函數有Parse。
查看 json 內容,使用 Writer 類即可。
Jsoncpp 的 Json::Writer 類是一個純虛類,並不能直接使用。在此我們使用 Json::Writer 的子類:Json::FastWriter、Json::StyledWriter、Json::StyledStreamWriter。
Json::FastWriter fast_writer;
std::cout << fast_writer.write(root) << std::endl;
輸出結果爲:
{"key_array":["array_string",1234],"key_boolean":false,"key_double":12.3450,"key_number":12345,"key_object":{"age":26,"name":"name1"},"key_string":"value_string"}
用 Json::StyledWriter 是格式化後的 json,下面我們來看看 Json::StyledWriter 是怎樣格式化的。
Json::StyledWriter styled_writer;
std::cout << styled_writer.write(root) << std::endl;
輸出結果爲:
{
"key_array" : [ "array_string", 1234 ],
"key_boolean" : false,
"key_double" : 12.3450,
"key_number" : 12345,
"key_object" : {
"age" : 26,
"name" : "name1"
},
"key_string" : "value_string"
}
3.Json::Writer 與Json::Reader相反,將Json::Value轉化成字符串流,注意它的兩個子類:Json::FastWriter和Json::StyleWriter,分別輸出不帶格式的json和帶格式的json。
Json::Reader 是用於讀取的,用於將字符串轉換爲 Json::Value 對象的,
Json::Reader reader;
Json::Value json_object;
const char* json_document = "{\"age\" : 26,\"name\" : \"123\"}";
if (!reader.parse(json_document, json_object))
return 0;
std::cout << json_object["name"] << std::endl;
std::cout << json_object["age"] << std::endl;
輸出結果爲:
"123"
例如:
字符串方式:
- 01.int ParseJsonFromString()
- 02.{
- 03. const char* str = "{\"uploadid\": \"UP000000\",\"code\": 100,\"msg\": \"\",\"files\": \"\"}";
- 04.
- 05. Json::Reader reader;
- 06. Json::Value root;
- 07. if (reader.parse(str, root)) // reader將Json字符串解析到root,root將包含Json裏所有子元素
- 08. {
- 09. std::string upload_id = root["uploadid"].asString(); // 訪問節點,upload_id = "UP000000"
- 10. int code = root["code"].asInt(); // 訪問節點,code = 100
- 11. }
- 12. return 0;
- 13.}
從文件解析json
- 01.{
- 02. "uploadid": "UP000000",
- 03. "code": "0",
- 04. "msg": "",
- 05. "files":
- 06. [
- 07. {
- 08. "code": "0",
- 09. "msg": "",
- 10. "filename": "1D_16-35_1.jpg",
- 11. "filesize": "196690",
- 12. "width": "1024",
- 13. "height": "682",
- 14. "images":
- 15. [
- 16. {
- 17. "url": "fmn061/20111118",
- 18. "type": "large",
- 19. "width": "720",
- 20. "height": "479"
- 21. },
- 22. {
- 23. "url": "fmn061/20111118",
- 24. "type": "main",
- 25. "width": "200",
- 26. "height": "133"
- 27. }
- 28. ]
- 29. }
- 30. ]
- 31.}
解析代碼:
- 01.int ParseJsonFromFile(const char* filename)
- 02.{
- 03. // 解析json用Json::Reader
- 04. Json::Reader reader;
- 05. // Json::Value是一種很重要的類型,可以代表任意類型。如int, string, object, array...
- 06. Json::Value root;
- 07.
- 08. std::ifstream is;
- 09. is.open (filename, std::ios::binary );
- 10. if (reader.parse(is, root))
- 11. {
- 12. std::string code;
- 13. if (!root["files"].isNull()) // 訪問節點,Access an object value by name, create a null member if it does not exist.
- 14. code = root["uploadid"].asString();
- 15.
- 16. // 訪問節點,Return the member named key if it exist, defaultValue otherwise.
- 17. code = root.get("uploadid", "null").asString();
- 18.
- 19. // 得到"files"的數組個數
- 20. int file_size = root["files"].size();
- 21.
- 22. // 遍歷數組
- 23. for(int i = 0; i < file_size; ++i)
- 24. {
- 25. Json::Value val_image = root["files"][i]["images"];
- 26. int image_size = val_image.size();
- 27. for(int j = 0; j < image_size; ++j)
- 28. {
- 29. std::string type = val_image[j]["type"].asString();
- 30. std::string url = val_image[j]["url"].asString();
- 31. }
- 32. }
- 33. }
- 34. is.close();
- 35. return 0;
- 36.}
Json中插入Json
- 01.Json::Value arrayObj; // 構建對象
- 02.Json::Value new_item, new_item1;
- 03.new_item["date"] = "2011-12-28";
- 04.new_item1["time"] = "22:30:36";
- 05.arrayObj.append(new_item); // 插入數組成員
- 06.arrayObj.append(new_item1); // 插入數組成員
- 07.int file_size = root["files"].size();
- 08.for(int i = 0; i < file_size; ++i)
- 09. root["files"][i]["exifs"] = arrayObj; // 插入原json中
輸出Json
- 01.// 轉換爲字符串(帶格式)
- 02.std::string out = root.toStyledString();
- 03.// 輸出無格式json字符串
- 04.Json::FastWriter writer;
- 05.std::string out2 = writer.write(root);
四、其它方式
Boost property_tree解析json,
property_tree可以解析xml,json,ini,info等格式的數據,用property_tree解析這幾種格式使用方法很相似。
解析json很簡單,命名空間爲boost::property_tree,reson_json函數將文件流、字符串解析到ptree,write_json將ptree輸出爲字符串或文件流。其餘的都是對ptree的操作。
解析json需要加頭文件:
#include <boost/property_tree/ptree.hpp>
#include <boost/property_tree/json_parser.hpp>