使用C++处理JSON数据交换格式
一、摘要
JSON的全称为:JavaScript Object Notation,顾名思义,JSON是用于标记Javascript对象的,JSON官方的解释为:JSON是一种轻量级的数据传输格式。本文并不详细介绍JSON本身的细节,旨在讨论如何使用C++语言来处理JSON。关于JSON更具体的信息,可参见JSON官网:http://www.json.org。http://json.org/json-zh.html
二、本文选择处理JSON的C++库
本文选择一个第三方库jsoncpp来解析JSON。jsoncpp是比较出名的C++ JSON解析库。在JSON官网也是首推的。下载地址为:http://sourceforge.net/projects/jsoncpp。本文使用的jsoncpp版本为:0.5.0。
三、jsoncpp在Windows下的编译
方法一:使用Jsoncpp生成的lib文件
解压上面下载的Jsoncpp文件,在jsoncpp-src-0.5.0/makefiles/vs71目录里找到jsoncpp.sln,用VS2008版本编译,默认生成静态链接库。 在工程中引用,只需要包含include/json下的头文件及生成的.lib文件即可。 如何包含lib文件:在.cpp文件中#pragma comment(lib."json_vc71_libmt.lib"),
在工程属性中Linker下Input中Additional Dependencies写入lib文件名字(Release下为json_vc71_libmt.lib,Debug为json_vc71_libmtd.lib)
注意:Jsoncpp的lib工程编译选项要和VS工程中的编译选项保持一致。如lib文件工程编译选项为MT(或MTd),VS工程中也要选择MT(或MTd),否则会出现编译错误问题,debug和release下生成的lib文件名字不同,注意不要看错了,当成一个文件来使用(我就犯了这个错误)。
方法二:使用Jsoncpp包中的.cpp和.h文件 解压上面下载的Jsoncpp文件,把jsoncpp-src-0.5.0文件拷贝到工程目录下,将jsoncpp-src-0.5.0\jsoncpp-src-0.5.0\include\json和jsoncpp-src-0.5.0\jsoncpp-src-0.5.0\src\lib_json目录里的文件包含到VS工程中,在VS工程的属性C/C++下General中Additional Include Directories包含头文件目录.\jsoncpp-src-0.5.0\include。在使用的cpp文件中包含json头文件即可,如:#include "json/json.h"。将json_reader.cpp、json_value.cpp和json_writer.cpp三个文件的Precompiled Header属性设置为Not Using Precompiled Headers,否则编译会出现错误。jsoncpp 使用详解
jsoncpp 主要包含三种类型的 class:Value、Reader、Writer。jsoncpp 中所有对象、类名都在 namespace Json 中,包含 json.h 即可。
Json::Value 只能处理 ANSI 类型的字符串,如果 C++ 程序是用 Unicode 编码的,最好加一个 Adapt 类来适配。
本实验采用方法二, 这样方便调试,同时也方便学习该开源库的代码。 本代码不为商用, 所以以学习为主。
四\ 需要解析的数据,face++ 调动返回json数据,数据个数如下:
{ "face": [ { "attribute": { "age": { "range": 5, "value": 23 }, "gender": { "confidence": 99.9999, "value": "Female" }, "glass": { "confidence": 99.945, "value": "None" }, "pose": { "pitch_angle": { "value": 17 }, "roll_angle": { "value": 0.735735 }, "yaw_angle": { "value": -2 } }, "race": { "confidence": 99.6121, "value": "Asian" }, "smiling": { "value": 4.86501 } }, "face_id": "17233b4b1b51ac91e391e5afe130eb78", "position": { "center": { "x": 49.4, "y": 37.6 }, "eye_left": { "x": 43.3692, "y": 30.8192 }, "eye_right": { "x": 56.5606, "y": 30.9886 }, "height": 26.8, "mouth_left": { "x": 46.1326, "y": 44.9468 }, "mouth_right": { "x": 54.2592, "y": 44.6282 }, "nose": { "x": 49.9404, "y": 38.8484 }, "width": 26.8 }, "tag": "" } ], "img_height": 500, "img_id": "22fd9efc64c87e00224c33dd8718eec7", "img_width": 500, "session_id": "38047ad0f0b34c7e8c6efb6ba39ed355", "url": "http://www.faceplusplus.com.cn/wp-content/themes/faceplusplus/assets/img/demo/1.jpg?v=4" }
五、遇到问题,曾经尝试多次,找了很多博客都是不能解决问题,
仔细观察后,发现有[ ], 这个不能忽视。http://blog.163.com/pei_hua100/blog/static/80569759201333114010800/ 通过这个博客启发。发现[] 为里面的数据。所以更改代码如下。
<span style="color:#333333;">Json::Value detect; Json::Value face; if (!DetectResult1.parse(DetectResult, detect)) { return -1; } int face_size = detect["face"].size(); // 遍历face 个数 for (int i = 0; i < face_size; i++) </span><span style="color:#ff0000;"> 这里很关键,学会使用该方法。</span><span style="color:#333333;"> { Json::Value attribute; attribute = detect["face"][i]["attribute"]; int attribute_size = attribute.size(); int age = attribute["age"]["value"].asInt(); string gender = attribute["gender"]["value"].asString(); int smiling = attribute["smiling"]["value"].asInt(); string race = attribute["race"]["value"].asString(); } for (int i = 0; i < face_size; i++) { Json::Value position; position = detect["face"][i]["position"]; int centerX = position["center"]["x"].asInt(); int centerY = position["center"]["y"].asInt(); int eye_leftx = position["eye_left"]["x"].asInt(); int eye_lefty = position["eye_left"]["y"].asInt(); int eye_rightx = position["eye_right"]["x"].asInt(); int eye_righty = position["eye_right"]["y"].asInt(); int height = position["height"].asInt(); cout << "centerX" << centerX << "centerY" << centerY << endl; int mouth_leftx = position["mouth_left"]["x"].asInt(); int mouth_lefty = position["mouth_left"]["y"].asInt(); int mouth_rightx = position["mouth_right"]["x"].asInt(); int mouth_righty = position["mouth_right"]["y"].asInt(); int nosex = position["nose"]["x"].asInt(); int nosey = position["nose"]["y"].asInt(); int width = position["width"].asInt(); } int img_height = detect["img_height"].asInt(); cout << "img_height" << img_height << endl; int img_width = face["img_width"].asInt(); </span>
通过这个方式,发现curl 返回的代码 里面有\n 换行符, 想办法去掉他。
本实验的实现方式如下:
// 去掉返回值中的 \n 换行符 string::iterator it; for (it = HTTPRESULT.begin(); it != HTTPRESULT.end(); ++it) { if (*it == '\n') { // *it = '\\r\\n'; HTTPRESULT.erase(it); } }
本次实验所有代码如下:// face++.cpp : 定义控制台应用程序的入口点。 // #include "stdafx.h" #include <string> #include <iostream> using namespace std; #include "HttpClient.h" #include "curl/curl.h" #include "curl/easy.h" std::string strResult; #include <opencv2\core\core.hpp> #include <opencv2\highgui\highgui.hpp> #include "json/json.h" using namespace cv; // #pragma comment(lib, "libcurl.lib") std::string HTTPRESULT; const char* detectResult; void Write_data(void* buffer, size_t size, size_t nmemb, void* user_p){ cout << "(const char*)buffer" << (const char*)buffer << endl; HTTPRESULT += (const char*)buffer; detectResult = (const char*)buffer; } int _tmain(int argc, _TCHAR* argv[]) { CURL *curl = curl_easy_init(); CURLcode res = curl_global_init(CURL_GLOBAL_WIN32); struct curl_httppost *formpost = NULL; struct curl_httppost *lastptr = NULL; // struct curl_slist *headerlist=NULL; // static const char buf[] = "Expect:"; curl_formadd(&formpost, &lastptr, CURLFORM_COPYNAME, "api_key", CURLFORM_COPYCONTENTS, "d45344602f6ffd77baeab05b99fb7730", CURLFORM_END); curl_formadd(&formpost, &lastptr, CURLFORM_COPYNAME, "api_secret", CURLFORM_COPYCONTENTS, "jKb9XJ_GQ5cKs0QOk6Cj1HordHFBWrgL", CURLFORM_END); char* file_data = NULL; long file_size = 0; string imageName = "d:\\mqx.jpg"; FILE* fp = fopen("d:\\mqx.jpg", "rb"); if (fp) { fseek(fp, 0, SEEK_END); file_size = ftell(fp); fseek(fp, 0, SEEK_SET); file_data = new char[file_size + 1]; fread(file_data, 1, file_size, fp); cout << file_data << endl; fclose(fp); } curl_formadd(&formpost, &lastptr, CURLFORM_COPYNAME, "img", CURLFORM_BUFFER, "test.jpg", CURLFORM_BUFFERPTR, file_data, CURLFORM_BUFFERLENGTH, file_size, CURLFORM_CONTENTTYPE, "image/jpeg", CURLFORM_END); if (curl) { // what URL that receives this POST curl_easy_setopt(curl, CURLOPT_URL, "http://apicn.faceplusplus.com/v2/detection/detect"); curl_easy_setopt(curl, CURLOPT_VERBOSE, 0); curl_easy_setopt(curl, CURLOPT_HTTPPOST, formpost); curl_easy_setopt(curl, CURLOPT_WRITEDATA, &HTTPRESULT); curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, Write_data); cout << "CURLOPT_WRITEFUNCTION" << CURLOPT_WRITEFUNCTION << endl; char error[1024]; curl_easy_setopt(curl, CURLOPT_ERRORBUFFER, error); res = curl_easy_perform(curl); if (res != CURLE_OK) cout << endl << error << endl; } curl_easy_cleanup(curl); curl_formfree(formpost); cout << endl << HTTPRESULT<< endl; // cout << endl << "测试结果如下" << endl; // cout << endl << "年龄:" << HTTPRESULT.find("value") << endl; // cout << endl << "年龄:" << HTTPRESULT.substr(HTTPRESULT.find("age", 0)) << endl; // cout << endl << "年龄:" << HTTPRESULT.substr(HTTPRESULT.find("gender",0)) << endl; // cout << endl << "年龄:" << HTTPRESULT.substr(HTTPRESULT.find("race", 0)) << endl; // cout << endl << "年龄:" << HTTPRESULT.substr(HTTPRESULT.find("smiling", 0)) << endl; // 得到年龄性别等。 int agePos = HTTPRESULT.find("age", 0); string age = HTTPRESULT.substr(agePos + 70, 2); int genderPos = HTTPRESULT.find("gender", 0); string gender = HTTPRESULT.substr(genderPos + 84, 6); int racePos = HTTPRESULT.find("race", 0); string race = HTTPRESULT.substr(racePos + 83, 5); int smilingPos = HTTPRESULT.find("smiling", 0); string smiling = HTTPRESULT.substr(smilingPos + 41, 5); // end 结束 /* //读入图像 Mat img = imread(imageName, CV_LOAD_IMAGE_COLOR); //如果读入图像失败 if (img.empty()) { cout << "Could not open or find the image!" << endl; return -1; } //创建窗口 namedWindow("face", CV_WINDOW_AUTOSIZE); //显示图像 imshow("face", img); //等待按键,按键盘任意键返回 waitKey(-1); // while (HTTPRESULT != " ") // { // agePos[numage] = HTTPRESULT.find("value"); // HTTPRESULT // } // int agePos = HTTPRESULT.find("value"); if (file_data != NULL) delete[] file_data; */ //HTTPRESULT = HTTPRESULT.replace(/ \\n / g, "\\n"); // 去掉返回值中的 \n 换行符 string::iterator it; for (it = HTTPRESULT.begin(); it != HTTPRESULT.end(); ++it) { if (*it == '\n') { // *it = '\\r\\n'; HTTPRESULT.erase(it); } } const char * DetectResult = HTTPRESULT.c_str(); Json::Reader DetectResult1; Json::Value detect; Json::Value face; if (!DetectResult1.parse(DetectResult, detect)) { return -1; } int face_size = detect["face"].size(); // 遍历face 个数 for (int i = 0; i < face_size; i++) { Json::Value attribute; attribute = detect["face"][i]["attribute"]; int attribute_size = attribute.size(); int age = attribute["age"]["value"].asInt(); string gender = attribute["gender"]["value"].asString(); int smiling = attribute["smiling"]["value"].asInt(); string race = attribute["race"]["value"].asString(); } for (int i = 0; i < face_size; i++) { Json::Value position; position = detect["face"][i]["position"]; int centerX = position["center"]["x"].asInt(); int centerY = position["center"]["y"].asInt(); int eye_leftx = position["eye_left"]["x"].asInt(); int eye_lefty = position["eye_left"]["y"].asInt(); int eye_rightx = position["eye_right"]["x"].asInt(); int eye_righty = position["eye_right"]["y"].asInt(); int height = position["height"].asInt(); cout << "centerX" << centerX << "centerY" << centerY << endl; int mouth_leftx = position["mouth_left"]["x"].asInt(); int mouth_lefty = position["mouth_left"]["y"].asInt(); int mouth_rightx = position["mouth_right"]["x"].asInt(); int mouth_righty = position["mouth_right"]["y"].asInt(); int nosex = position["nose"]["x"].asInt(); int nosey = position["nose"]["y"].asInt(); int width = position["width"].asInt(); } int img_height = detect["img_height"].asInt(); cout << "img_height" << img_height << endl; int img_width = face["img_width"].asInt(); system("pause"); return 0; }
下面是从网上找的代码示例: 1. 从字符串解析json
const char * str = "{\"uploadid\": \"UP000000\",\"code\": 100,\"msg\": \"\",\"files\": \"\"}"; Json::Reader reader; Json::Value root; if (reader.parse(str, root)) // reader将Json字符串解析到root,root将包含Json里所有子元素 { std:: string upload_id = root["uploadid"].asString(); // 访问节点,upload_id = "UP000000" int code = root["code"].asInt(); // 访问节点,code = 100 }2. 从文件解析jsonint ReadJsonFromFile( const char * filename) { Json::Reader reader; // 解析json用Json::Reader Json::Value root; // Json::Value是一种很重要的类型,可以代表任意类型。如int, string, object, array std::ifstream is ; is .open (filename, std::ios::binary ); if (reader.parse( is , root, FALSE)) { std:: string code; if (!root["files"].isNull()) // 访问节点,Access an object value by name, create a null member if it does not exist. code = root["uploadid"].asString(); code = root. get ("uploadid", "null").asString(); // 访问节点,Return the member named key if it exist, defaultValue otherwise. int file_size = root["files"].size(); // 得到"files"的数组个数 for ( int i = 0; i < file_size; ++i) // 遍历数组 { Json::Value val_image = root["files"][i]["images"]; int image_size = val_image.size(); for ( int j = 0; j < image_size; ++j) { std:: string type = val_image[j]["type"].asString(); std:: string url = val_image[j]["url"].asString(); printf("type : %s, url : %s \n", type.c_str(), url.c_str()); } } } is .close(); return 0; }3. 向文件中插入json
void WriteJsonData( const char * filename) { Json::Reader reader; Json::Value root; // Json::Value是一种很重要的类型,可以代表任意类型。如int, string, object, array std::ifstream is ; is .open (filename, std::ios::binary ); if (reader.parse( is , root)) { Json::Value arrayObj; // 构建对象 Json::Value new_item, new_item1; new_item["date"] = "2011-11-11"; new_item1["time"] = "11:11:11"; arrayObj.append(new_item); // 插入数组成员 arrayObj.append(new_item1); // 插入数组成员 int file_size = root["files"].size(); for ( int i = 0; i < file_size; ++i) root["files"][i]["exifs"] = arrayObj; // 插入原json中 std:: string out = root.toStyledString(); // 输出无格式json字符串 Json::FastWriter writer; std:: string strWrite = writer.write(root); std::ofstream ofs; ofs.open("test_write.json"); ofs << strWrite; ofs.close(); } is .close(); }
要使用第三方源码库,第一步少不了的就是编译,将源码文件编译成我们方便使用的动态链接库、静态链接库或者静态导入库[1]。
jsconcpp进行JSON解析的源码文件分布在include/json、src/lib_json下。其实jsoncpp源码并不多,为了方便产品管理,此处没必要将其编译为动态链接库或者静态导入库,所以我们选择使用静态链接库[2]。
jsoncpp已经处理的很完善了,所有编译选项都已经配置好,打开makefiles/vs71/jsoncpp.sln便可以开始编译(默认是使用VS2003编译器的,打开时直接按照VS2005提示转换即可)。
四、jsoncpp使用详解
jsoncpp主要包含三种类型的class:Value、Reader、Writer。jsoncpp中所有对象、类名都在namespace Json中,包含json.h即可。
Json::Value只能处理ANSI类型的字符串,如果C++程序是用Unicode编码的,最好加一个Adapt类来适配。
1、Value
Json::Value是jsoncpp中最基本、最重要的类,用于表示各种类型的对象,jsoncpp支持的对象类型可见Json::ValueType枚举值。
可如下是用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、Writer
如上说了Json::Value的使用方式,现在到了该查看刚才赋值内容的时候了,查看json内容,使用Writer类即可。
Jsoncpp的Json::Writer类是一个纯虚类,并不能直接使用。在此我们使用Json::Writer的子类:Json::FastWriter、Json::StyledWriter、Json::StyledStreamWriter。
顾名思义,用Json::FastWriter来处理json应该是最快的,下面我们来试试。
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":"huchao"},"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" : "huchao"}, "key_string" : "value_string" }
3、Reader
Json::Reader是用于读取的,说的确切点,是用于将字符串转换为Json::Value对象的,下面我们来看个简单的例子。
Json::Reader reader;
json::Value json_object;
const char* json_document = "{\"age\" : 26,\"name\" : \"huchao\"}";
if (!reader.parse(json_document, json_object))
return 0;
std::cout << json_object["name"] << std::endl;
std::cout << json_object["age"] << std::endl;
输出结果为:
"huchao"
26
可见,上述代码已经解析出了json字符串。
--------------------------------------
[1]:使用第三方源码最简单的方法是直接将文件加入工程,但这样不利于源码、软件产品管理,对于一般软件开发来说,不建议使用。
[2]:如果真需要编译成动态链接库、静态导入库的话,可以使用VS新建一个工程属性,然后在Project --> Properties中进行相应的设置即可。
转载地址:http://hi.baidu.com/awz_tiger/blog/item/d165970b4ca967fc36d122a4.html
另一个比较有用的地址为:http://blog.csdn.net/vagrxie/article/details/5754179
参考资料: http://blog.csdn.net/chen19870707/article/details/39646173
http://www.cppblog.com/wanghaiguang/archive/2013/12/26/205020.html
http://bbs.csdn.net/topics/370006412