json格式解析和libjson的用法介紹(關(guān)于cjson的使用方法)
在閱讀本文之前,請(qǐng)先閱讀下《Rss Reader實(shí)例開(kāi)發(fā)之系統(tǒng)設(shè)計(jì)》一文。
Rss Reader實(shí)例開(kāi)發(fā)中,進(jìn)行網(wǎng)絡(luò)數(shù)據(jù)交換時(shí)主要使用到了兩種數(shù)據(jù)格式:JSON與XML。本文主要介紹JSON格式的簡(jiǎn)單概念及JSON在Rss Reader中的應(yīng)用,XML格式的使用將在下一篇文章做介紹。
JSON簡(jiǎn)介:
JSON(JavaScript Object Notation) 是一種輕量級(jí)的數(shù)據(jù)交換格式,可以把JSON的結(jié)構(gòu)理解成無(wú)序的、可嵌套的key-value鍵值對(duì)集合,這些key-value鍵值對(duì)是以結(jié)構(gòu)體或數(shù)組的形式來(lái)組織的。同一級(jí)的key-value鍵值對(duì)之間是用一個(gè)“,”(逗號(hào))隔開(kāi),每個(gè)key-value鍵值對(duì)是由一個(gè)key后面緊接一個(gè)“:”(冒號(hào)),冒號(hào)后面是這個(gè)key對(duì)應(yīng)的value。Key是一個(gè)word,由大小寫(xiě)字母、下劃線及數(shù)字組成,可以由雙引號(hào)封閉,也可以不加雙引號(hào);而value的取值集為:Number、Boolean(true或false)、null、String、Object及Array,如圖一:
(圖一)
1、Number:數(shù)值,包括整形數(shù)與浮點(diǎn)數(shù),如:123、0.83、-2.7e10。其結(jié)構(gòu)如圖二:
(圖二)
2、String:字符串,是以雙引號(hào)封閉起來(lái)的一串字符,使用反斜杠來(lái)轉(zhuǎn)義,如:\\、\n等,JSON中字符串的概念與C/C++或者JAVA語(yǔ)言里的字符串概念差不多,如:”abc”。其結(jié)構(gòu)如圖三:
(圖三)
3、Object:對(duì)象,也可理解成一個(gè)結(jié)構(gòu)體,是以一對(duì)大括號(hào)封閉起來(lái)的無(wú)序的key-value鍵值對(duì)集合,例如:{name:"Susan", age:27, birthday:{year:1984, month:2, day:11}};也可以寫(xiě)成:{"name":"Susan", "age":27, "birthday":{"year":1984, "month":2, "day":11}};其結(jié)構(gòu)如圖四:
(圖四)
4、Array:數(shù)組,JSON的數(shù)組是一個(gè)以中括號(hào)封閉起來(lái)的value的集合,即數(shù)組內(nèi)的各個(gè)成員的數(shù)據(jù)類型可以不一樣,這一點(diǎn)就跟C/JAVA的數(shù)組概念不同了。每個(gè)value之間是由一個(gè)“,”(逗號(hào))隔開(kāi),例如:[123, abc, false, {name:mj}];其結(jié)構(gòu)如圖五:
(圖五)
關(guān)于JSON的詳細(xì)說(shuō)明與教程請(qǐng)自行到網(wǎng)絡(luò)上搜索,有很多。
下面我們就來(lái)動(dòng)手寫(xiě)一個(gè)例子:
{
  result:true,
 
  root:{
    version:"201007091640",
    channels:[
    {
      name:"新聞中心",
      subchnls:[
      {
        title:"焦點(diǎn)新聞",
        link:"http://jb51.net/news/channel/1/news.rss",
        desc:"家事、國(guó)事、天下事"
      },
      {
        title:"新聞?lì)l道",
        link:"http://jb51.net/news/channel/2/news.rss",
        desc:"讓您實(shí)時(shí)掌握國(guó)際動(dòng)態(tài)"
      },
      {
        title:"軍事頻道",
        link:"http://jb51.net/news/channel/3/news.rss",
        desc:"軍事"
      }
      ]
    },
    
    {
      name:"體育新聞",
      subchnls:[
      {
        title:"體育要聞匯總",
        link:"http://jb51.net/news/channel/4/news.rss",
        desc:"erewr"
      },
      {
        title:"國(guó)際足壇",
        link:"http://jb51.net/news/channel/5/news.rss",
        desc:"werewr"
      }
      ]
    }
    
    ]
  }
}
這段JSON描述了一個(gè)對(duì)象(最外層大括號(hào)包圍的部分),為了方便區(qū)分,我們就將其稱為對(duì)象A吧。對(duì)象A有兩個(gè)Item(即key-value鍵值對(duì)),一個(gè)是result,其值為true;一個(gè)是root,其值為一個(gè)對(duì)象,稱為對(duì)象B。對(duì)象B也有兩個(gè)Item,一個(gè)是version,其值為一個(gè)字串” 201007091640”;一個(gè)是channels,其值是一個(gè)數(shù)組,而數(shù)組的成員都是一個(gè)對(duì)象,每個(gè)對(duì)象又包含兩個(gè)Item,一個(gè)是name,值分別為字串"新聞中心"和"體育新聞";一個(gè)是subchnls,值都是數(shù)組,每個(gè)數(shù)組又分別有若干個(gè)成員,每個(gè)subchnls成員也都是一個(gè)對(duì)象,每個(gè)對(duì)象都有三個(gè)Item:title、link和desc。也許你看到這,已經(jīng)是一頭大汗了,不過(guò)沒(méi)關(guān)系,我們來(lái)帖張這段JSON文本對(duì)應(yīng)的結(jié)構(gòu)圖,有圖就有真相,請(qǐng)看圖六:
(圖六:黑色實(shí)線為對(duì)象,虛線為值,橙色實(shí)線為數(shù)組)
在RssReader中使用cJSON:
在RssReader中使用了開(kāi)源庫(kù)cJSON來(lái)解析JSON,所以在此就介紹下cJSON的使用:
在CJSON中,一個(gè)key-value鍵值對(duì)被解析并存放在一個(gè)cJSON結(jié)構(gòu)體變量中,其value取值集為:FALSE,TRUE,NULL,NUMBER,STRING,OBJECT,ARRAY。它們分別被存放在CJSON對(duì)象的child、valuestring、valueint、valuedouble變量中,而用于判斷某個(gè)CJSON對(duì)象value的數(shù)據(jù)類型則是CJSON對(duì)象的type變量,其取值范圍與CJSON對(duì)象的value集是一一對(duì)應(yīng)的,如:cJSON_False對(duì)應(yīng)FALSE。
cJSON Types:
#define cJSON_False 0 #define cJSON_True 1 #define cJSON_NULL 2 #define cJSON_Number 3 #define cJSON_String 4 #define cJSON_Array 5 #define cJSON_Object 6
cJSON 結(jié)構(gòu)體:
typedef struct cJSON
{
  struct cJSON *next,*prev;  //指向上一項(xiàng)/下一項(xiàng)
  struct cJSON *child;  //指向下一級(jí),也就是當(dāng)type為cJSON_Object或cJSON_Array時(shí),此指針不為空。
  int type;          
  char *valuestring; // 當(dāng)type為cJSON_String時(shí)
  int valueint;    // 當(dāng) type為cJSON_Number時(shí)
  double valuedouble; //當(dāng)type為cJSON_Number時(shí)
 
  char *string;    // 當(dāng)前項(xiàng)的名稱,也就是key-value鍵值對(duì)的key
} cJSON;
在解析JSON過(guò)程中,從JSON格式描述的value數(shù)據(jù)到CJSON對(duì)象中存放的變量的一個(gè)映射關(guān)系如圖七:
(圖七)
對(duì)CJSON格式的解析是使用cJSON_Parse()方法,其傳入的參數(shù)是一個(gè)CJSON的Object/Array結(jié)構(gòu)的字串,解析成功則返回一個(gè)cJSON結(jié)構(gòu)體變量的指針,在使用完成后需要調(diào)用cJSON_Delete()將該指針?shù)N毀。CJSON是以樹(shù)狀結(jié)構(gòu)來(lái)組織內(nèi)部的各個(gè)cJSON結(jié)構(gòu)體變量的,一般地,要使用某個(gè)cJSON結(jié)構(gòu)體變量,需要調(diào)用cJSON_GetObjectItem()方法并根據(jù)其父節(jié)點(diǎn)的cJSON結(jié)構(gòu)體變量指針與該項(xiàng)的名稱來(lái)獲取其指針,舉個(gè)例子:
bool bResult;
char jsonString[] = “{result:true}”;
//獲取result的值true
cJSON* pItem = NULL;
cJSON* pRoot = cJSON_Parse ( jsonString );
if ( pRoot )
{
  pItem = cJSON_GetObjectItem ( pRoot, “result” );
  if ( pItem )
  {
    bResult = pItem->valueint;  //由于result的值type為cJSON_False或cJSON_True,所以其值被存放在valueint變量中
  }
  cJSON_Delete ( pRoot );
}
在上例中,不管result的值type為何類型,都是通過(guò)調(diào)用cJSON_GetObjectItem()方法獲取其對(duì)應(yīng)的cJSON結(jié)構(gòu)體變量的指針,只是在處理其對(duì)應(yīng)的值時(shí)會(huì)有所不同。如果result的值type為cJSON_Object,則需要通過(guò)調(diào)用cJSON_GetObjectItem( pItem, “subItemKey”)來(lái)獲取其子Item的指針。在處理值type為cJSON_Array的Item時(shí),就需要再用到另外兩個(gè)API:cJSON_GetArraySize ()和cJSON_GetArrayItem()。我們舉個(gè)獲取一個(gè)數(shù)組成員值的例子:
char* out;
char jsonString[] = “{colors:[\“red\”, \“green\”,\ “blue\”, \“yellow\”, \“white\”]}”;
cJSON* pArray = NULL;
cJSON* pRoot = cJSON_Parse ( jsonString );
if ( pRoot )
{
  pArray = cJSON_GetObjectItem ( pRoot, “colors” );
  if ( pArray )
  {
    cJSON* pArrayItem = NULL;
    int nCount = cJSON_GetArraySize ( pArray ); //獲取pArray數(shù)組的大小
    for( int i = 0; i < nCount; i++)
    {
      pArrayItem = cJSON_GetArrayItem(pArray, i);
      out = cJSON_Print( pArrayItem );  //將pArrayItem的值以字串的形式打印到char型buffer上,cJSON_Print()會(huì)自動(dòng)分配內(nèi)存空間,用完需要釋放內(nèi)存。
      SS_printf( “array item %d: %s\n”, i, out);
      Free( out );
    }
  }
  cJSON_Delete ( pRoot );
}
在提取一個(gè)復(fù)雜的JSON格式的數(shù)據(jù)時(shí),也只是將以上兩個(gè)例子使用到的方法進(jìn)行組合調(diào)用罷了。所以對(duì)CJSON提供的API的使用是很簡(jiǎn)單有效的。有了以上知識(shí)的了解,我們就可以編寫(xiě)一些代碼將例一中的JSON解析并提取其中的數(shù)據(jù),還是貼點(diǎn)代碼才是硬道理,代碼如下:
TChannelsData.h:
/** 子頻道信息結(jié)構(gòu)體
* 
*/
struct SUBCHNL_DATA
{
  SUBCHNL_DATA();
  void clear();
 
  TUChar * m_title;
  char * m_link;
  TUChar * m_desc;
};
 
/** 大頻道信息結(jié)構(gòu)體
* 
*/
struct CHANNEL_DATA
{
  CHANNEL_DATA();
 
  TUChar* m_pszTitle;
  vector m_aSubChnlList;
};
 
//………….
// TChannelsData 類成員變量:RSSReaderConfig 版本號(hào)
char m_pszVersion[32];
// TChannelsData 類成員變量:頻道信息列表
vector m_aChnlsList;
//………….
TChannelsData.cpp: 
 
/** 解析JSON格式的內(nèi)容
* 
* \param pszJsonText 解析的JSON格式內(nèi)容字串
* 
* \return true:有更新數(shù)據(jù); false:沒(méi)有更新數(shù)據(jù)
*/
Boolean TChannelsData::ParseJson(char* pszJsonText)
{
  //char* out;
  cJSON* objJson;
 
  objJson= cJSON_Parse(pszJsonText);
 
  if (objJson)
  {
    //out=cJSON_Print(objJson);
    cJSON* objRootItem = NULL;
 
    //判斷是否需要更新
    objRootItem = cJSON_GetObjectItem(objJson, "result");
    if (objRootItem)
    {
      if (!objRootItem->valueint)
      {
        return FALSE;
      }
    }
    else
    {
      return FALSE;
    }
 
    //獲取更新數(shù)據(jù),根節(jié)點(diǎn)root
    objRootItem = cJSON_GetObjectItem(objJson, "root");
    if (objRootItem)
    {
      cJSON* objJsonItem = NULL;
 
      //獲取版本號(hào)
      objJsonItem = cJSON_GetObjectItem(objRootItem, "version");
      if (objJsonItem)
      {
        Int32 nLen = strlen(objJsonItem->valuestring);
        strncpy(m_pszVersion, objJsonItem->valuestring, (nLen < 32)? nLen : 31);
      }
 
      //解析出大頻道
      _ParseChannels(objRootItem);
    }
    
    //SS_printf("=======[parse json]%s\n",out);
    cJSON_Delete(objJson);
    //free(out);
  }
 
  return TRUE;
}
 
/** 解析出大頻道
* 
* \param pCJson cJSON對(duì)象指針
* 
* \return void
*/
void TChannelsData::_ParseChannels(cJSON* pCJson)
{
  cJSON* pJsonArray = NULL;
 
  if (!pCJson)
  {
    return;
  }
 
  pJsonArray = cJSON_GetObjectItem(pCJson, "channels");
  if(pJsonArray)
  {
    cJSON* pArrayItem = NULL;
    cJSON* pJsonTemp = NULL;
 
    Int32 nSize = cJSON_GetArraySize(pJsonArray);
    for (Int32 i = 0; i < nSize; i++)
    {
      pArrayItem = cJSON_GetArrayItem(pJsonArray, i);
      if (pArrayItem)
      {
        CHANNEL_DATA tChannelData;
        Int32 nLen = 0;
 
        //獲取大頻道名稱
        tChannelData.m_pszTitle = _JsonGetTUString(pArrayItem, "name");
        
        //解析出子頻道
        _ParseSubChnls(tChannelData.m_aSubChnlList, pArrayItem);
 
        //將大頻道信息對(duì)象壓入列表中
        m_aChnlsList.push_back(tChannelData);
      }
      else
      {
        continue;
      }
    }
  }
}
 
/** 解析子頻道
* 
* \param aSubChnlList 存放子頻道數(shù)據(jù)的vector對(duì)象
* \param pCJson cJSON對(duì)象指針
* 
* \return void
*/
void TChannelsData::_ParseSubChnls(vector& aSubChnlList, cJSON* pCJson)
{
  cJSON* pJsonArray = NULL;
 
  if (!pCJson)
  {
    return;
  }
 
  pJsonArray = cJSON_GetObjectItem(pCJson, "subchnls");
  if (pJsonArray)
  {
    cJSON* pArrayItem = NULL;
    //cJSON* pJsonTemp = NULL;
 
    Int32 nSize = cJSON_GetArraySize(pJsonArray);
    for (Int32 i = 0; i < nSize; i++)
    {
      pArrayItem = cJSON_GetArrayItem(pJsonArray, i);
      if (pArrayItem)
      {
        SUBCHNL_DATA tSubChnlData;
        Int32 nLen = 0;
 
        //get title
        tSubChnlData.m_title = _JsonGetTUString(pArrayItem, "title");
 
        //get link
        tSubChnlData.m_link = _JsonGetString(pArrayItem, "link");
 
        //get desc
        tSubChnlData.m_desc = _JsonGetTUString(pArrayItem, "desc");
 
        aSubChnlList.push_back(tSubChnlData);
      }
    }
  }
}
 
/** 獲取指定的cJSON對(duì)象的指定屬性值
* 
* \param pJsonItem cJSON對(duì)象指針
* \param pszKey cJSON對(duì)象屬性
* 
* \return 返回JSON對(duì)象的值,以TUChar字串形式返回
*/
TUChar* TChannelsData::_JsonGetTUString(cJSON* pJsonItem, char* pszKey)
{
  TUChar* pszValue = NULL;
  Int32 nLen;
  cJSON* pJsonTemp = NULL;
 
  pJsonTemp = cJSON_GetObjectItem(pJsonItem, pszKey);
  if (pJsonTemp)
  {
    nLen = strlen(pJsonTemp->valuestring) + 1;
    pszValue = new TUChar[nLen];
    if(pszValue)
    {
      MemSet(pszValue, 0, nLen * sizeof(TUChar));
      TUString::StrUtf8ToStrUnicode(pszValue, (const Char*)pJsonTemp->valuestring);
    }
  }
 
  return pszValue;
}
 
/** 獲取指定的cJSON對(duì)象的指定屬性值
* 
* \param pJsonItem cJSON對(duì)象指針
* \param pszKey cJSON對(duì)象屬性
* 
* \return 返回JSON對(duì)象的值,以char字串形式返回
*/
char* TChannelsData::_JsonGetString(cJSON* pJsonItem, char* pszKey)
{
  char* pszValue = NULL;
  Int32 nLen;
  cJSON* pJsonTemp = NULL;
 
  pJsonTemp = cJSON_GetObjectItem(pJsonItem, pszKey);
  if (pJsonTemp)
  {
    nLen = strlen(pJsonTemp->valuestring) + 1;
    pszValue = new char[nLen];
    if(pszValue)
    {
      MemSet(pszValue, 0, nLen);
      strncpy(pszValue, pJsonTemp->valuestring, nLen - 1);
    }
  }
 
  return pszValue;
}
 
/** 獲取指定的cJSON對(duì)象的指定屬性值
* 
* \param pJsonItem cJSON對(duì)象指針
* \param pszKey cJSON對(duì)象屬性
* 
* \return 返回JSON對(duì)象的值,以int32形式返回
*/
Int32 TChannelsData::_JsonGetInt(cJSON* pJsonItem, char* pszKey)
{
  Int32 nValue = 0;
  cJSON* pJsonTemp = NULL;
 
  pJsonTemp = cJSON_GetObjectItem(pJsonItem, pszKey);
  if (pJsonTemp)
  {
    nValue = pJsonTemp->valueint;
  }
 
  return nValue;
}
 
/** 獲取指定的cJSON對(duì)象的指定屬性值
* 
* \param pJsonItem cJSON對(duì)象指針
* \param pszKey cJSON對(duì)象屬性
* 
* \return 返回JSON對(duì)象的值,以Boolean形式返回
*/
Boolean TChannelsData::_JsonGetBoolean(cJSON* pJsonItem, char* pszKey)
{
  Boolean bValue = FALSE;
  cJSON* pJsonTemp = NULL;
 
  pJsonTemp = cJSON_GetObjectItem(pJsonItem, pszKey);
  if (pJsonTemp)
  {
    if(pJsonTemp->valueint)
    {
      bValue = TRUE;
    }
  }
 
  return bValue;
}
總結(jié):
JSON的結(jié)構(gòu)簡(jiǎn)約,所以使得JSON的文檔的數(shù)據(jù)量比較小,比較適合用于網(wǎng)絡(luò)數(shù)據(jù)的交換,而且對(duì)JSON文檔的解析和數(shù)據(jù)提取的方法也很簡(jiǎn)單,方便程序員的使用,當(dāng)然也正是因?yàn)镴SON的結(jié)構(gòu)簡(jiǎn)約,使得JSON的可讀性與可編輯性會(huì)稍差于XML,所以JSON比較適合在較少有人工閱讀和編輯的情況下使用期。
備注:經(jīng)驗(yàn)證名稱需加“ 比如char jsonString[] = "{\"result\":true}";
以上就是小編為大家?guī)?lái)的json格式解析和libjson的用法介紹(關(guān)于cjson的使用方法)全部?jī)?nèi)容了,希望大家多多支持我們~
上一篇:關(guān)于C++中構(gòu)造函數(shù)初始化成員列表的總結(jié)
欄 目:C語(yǔ)言
本文標(biāo)題:json格式解析和libjson的用法介紹(關(guān)于cjson的使用方法)
本文地址:http://www.jygsgssxh.com/a1/Cyuyan/1917.html
您可能感興趣的文章
- 01-10數(shù)據(jù)結(jié)構(gòu)課程設(shè)計(jì)- 解析最少換車次數(shù)的問(wèn)題詳解
 - 01-10深入解析最長(zhǎng)公共子串
 - 01-10深入解析Linux下\r\n的問(wèn)題
 - 01-10Linux線程管理必備:解析互斥量與條件變量的詳解
 - 01-10解析Linux下的時(shí)間函數(shù):設(shè)置以及獲取時(shí)間的方法
 - 01-10DHCP:解析開(kāi)發(fā)板上動(dòng)態(tài)獲取ip的2種實(shí)現(xiàn)方法詳解
 - 01-10節(jié)序問(wèn)題:解析大小的端判定
 - 01-10解析如何在C語(yǔ)言中調(diào)用shell命令的實(shí)現(xiàn)方法
 - 01-10解析c中stdout與stderr容易忽視的一些細(xì)節(jié)
 - 01-10深入解析C中的數(shù)值與真假
 


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
 - 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
 - 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
 - 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
 - 5c語(yǔ)言計(jì)算三角形面積代碼
 - 6什么是 WSH(腳本宿主)的詳細(xì)解釋
 - 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
 - 8正則表達(dá)式匹配各種特殊字符
 - 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
 - 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
 
本欄相關(guān)
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用
 - 04-02func函數(shù)+在C語(yǔ)言 func函數(shù)在c語(yǔ)言中
 - 04-02c語(yǔ)言的正則匹配函數(shù) c語(yǔ)言正則表達(dá)
 - 04-02c語(yǔ)言用函數(shù)寫(xiě)分段 用c語(yǔ)言表示分段
 - 04-02c語(yǔ)言中對(duì)數(shù)函數(shù)的表達(dá)式 c語(yǔ)言中對(duì)
 - 04-02c語(yǔ)言編寫(xiě)函數(shù)冒泡排序 c語(yǔ)言冒泡排
 - 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
 - 04-02c語(yǔ)言分段函數(shù)怎么求 用c語(yǔ)言求分段
 - 04-02C語(yǔ)言中怎么打出三角函數(shù) c語(yǔ)言中怎
 - 04-02c語(yǔ)言調(diào)用函數(shù)求fibo C語(yǔ)言調(diào)用函數(shù)求
 
隨機(jī)閱讀
- 01-11Mac OSX 打開(kāi)原生自帶讀寫(xiě)NTFS功能(圖文
 - 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
 - 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
 - 01-10C#中split用法實(shí)例總結(jié)
 - 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
 - 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置
 - 01-10delphi制作wav文件的方法
 - 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
 - 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
 - 04-02jquery與jsp,用jquery
 


