testMemory.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294
  1. #include "../../common/testCommon.h"
  2. static void *yyMalloc(void *ctx, size_t size)
  3. {
  4. (void)ctx;
  5. return unityTestMalloc(size);
  6. }
  7. static void *yyRealloc(void *ctx, void *ptr, size_t oldSize, size_t size)
  8. {
  9. (void)ctx;
  10. (void)oldSize;
  11. return unityTestRealloc(ptr, size);
  12. }
  13. static void yyFree(void *ctx, void *ptr)
  14. {
  15. (void)ctx;
  16. unityTestFree(ptr);
  17. }
  18. static void RyanJsonMemoryFootprint(const char *jsonstr, int32_t *footprint)
  19. {
  20. int32_t baselineUse = unityTestGetUse();
  21. RyanJsonInitHooks(unityTestMalloc, unityTestFree, unityTestRealloc);
  22. RyanJson_t json = RyanJsonParse(jsonstr);
  23. TEST_ASSERT_NOT_NULL_MESSAGE(json, "RyanJson 解析失败");
  24. *footprint = unityTestGetUse() - baselineUse;
  25. RyanJsonDelete(json);
  26. TEST_ASSERT_EQUAL_INT_MESSAGE(baselineUse, unityTestGetUse(), "RyanJson 释放后 TLSF used 未回到基线");
  27. }
  28. static void cJSONMemoryFootprint(const char *jsonstr, int32_t *footprint)
  29. {
  30. int32_t baselineUse = unityTestGetUse();
  31. cJSON_Hooks hooks = {.malloc_fn = unityTestMalloc, .free_fn = unityTestFree};
  32. cJSON_InitHooks(&hooks);
  33. cJSON *json = cJSON_Parse(jsonstr);
  34. TEST_ASSERT_NOT_NULL_MESSAGE(json, "cJSON 解析失败");
  35. *footprint = unityTestGetUse() - baselineUse;
  36. cJSON_Delete(json);
  37. TEST_ASSERT_EQUAL_INT_MESSAGE(baselineUse, unityTestGetUse(), "cJSON 释放后 TLSF used 未回到基线");
  38. }
  39. static void yyjsonMemoryFootprint(const char *jsonstr, int32_t *footprint)
  40. {
  41. int32_t baselineUse = unityTestGetUse();
  42. yyjson_alc yyalc = {yyMalloc, yyRealloc, yyFree, NULL};
  43. // 先解析成只读文档
  44. yyjson_doc *doc = yyjson_read_opts(jsonstr, strlen(jsonstr), YYJSON_READ_NOFLAG, &yyalc, NULL);
  45. TEST_ASSERT_NOT_NULL_MESSAGE(doc, "yyjson 解析只读文档失败");
  46. // 从只读文档拷贝为可变文档
  47. yyjson_mut_doc *mdoc = yyjson_doc_mut_copy(doc, &yyalc);
  48. yyjson_doc_free(doc);
  49. TEST_ASSERT_NOT_NULL_MESSAGE(mdoc, "yyjson 拷贝可变文档失败");
  50. *footprint = unityTestGetUse() - baselineUse;
  51. yyjson_mut_doc_free(mdoc);
  52. TEST_ASSERT_EQUAL_INT_MESSAGE(baselineUse, unityTestGetUse(), "yyjson 释放后 TLSF used 未回到基线");
  53. }
  54. static void printfJsonCompare(const char *title, const char *jsonstr)
  55. {
  56. int32_t ryanJsonCount = 0;
  57. int32_t cJSONCount = 0;
  58. int32_t yyjsonCount = 0;
  59. RyanJsonMemoryFootprint(jsonstr, &ryanJsonCount);
  60. cJSONMemoryFootprint(jsonstr, &cJSONCount);
  61. yyjsonMemoryFootprint(jsonstr, &yyjsonCount);
  62. (void)testLog("%s 原始长度: %lu, 模拟分配(header=%lu,align=%lu) 峰值占用 -> RyanJson:%ld, cJSON:%ld, yyjson:%ld\r\n", title,
  63. (unsigned long)strlen(jsonstr), (unsigned long)RyanJsonTestAllocHeaderSize, (unsigned long)RyanJsonTestAllocAlignSize,
  64. (long)ryanJsonCount, (long)cJSONCount, (long)yyjsonCount);
  65. TEST_ASSERT_TRUE_MESSAGE(ryanJsonCount > 0 && cJSONCount > 0 && yyjsonCount > 0, "内存统计值必须大于 0");
  66. TEST_ASSERT_TRUE_MESSAGE(ryanJsonCount < cJSONCount, "RyanJson 内存占用应低于 cJSON");
  67. if (ryanJsonCount >= yyjsonCount)
  68. {
  69. (void)testLog("提示: 当前模拟参数下 RyanJson(%ld) 未低于 yyjson(%ld)\r\n", (long)ryanJsonCount, (long)yyjsonCount);
  70. }
  71. if (cJSONCount > 0 && yyjsonCount > 0)
  72. {
  73. double saveVsCjson = 100.0 - ((double)ryanJsonCount * 100.0) / (double)cJSONCount;
  74. double saveVsYyjson = 100.0 - ((double)ryanJsonCount * 100.0) / (double)yyjsonCount;
  75. (void)testLog("内存优化程度 -> 比 cJSON 节省: \033[1;32m%.2f%%\033[0m, 比 yyjson 节省: \033[1;32m%.2f%%\033[0m\r\n",
  76. saveVsCjson, saveVsYyjson);
  77. }
  78. }
  79. static void testMixedJsonMemory(void)
  80. {
  81. static const char jsonstr[] =
  82. "{\"item1\":{\"inter\":16,\"double\":16.89,\"string\":\"hello\",\"boolTrue\":true,\"boolFalse\":false,\"null\":null,"
  83. "\"item\":{\"inter\":16,"
  84. "\"double\":16.89,\"string\":\"hello\","
  85. "\"boolTrue\":true,\"boolFalse\":false,\"null\":null},\"arrayInt\":[16,16,16,16,16],\"arrayDouble\":[16.89,16.89,16.89,"
  86. "16.89,16.89],"
  87. "\"arrayString\":[\"hello\",\"hello\",\"hello\","
  88. "\"hello\",\"hello\"],\"array\":[16,16.89,\"hello\",true,false,null],\"arrayItem\":[{\"inter\":16,\"double\":16.89,"
  89. "\"string\":\"hello\","
  90. "\"boolTrue\":true,\"boolFalse\":false,"
  91. "\"null\":null},{\"inter\":16,\"double\":16.89,\"string\":\"hello\",\"boolTrue\":true,\"boolFalse\":false,\"null\":null}]"
  92. "},\"item2\":{"
  93. "\"inter\":16,\"double\":16.89,\"string\":"
  94. "\"hello\",\"boolTrue\":true,\"boolFalse\":false,\"null\":null,\"item\":{\"inter\":16,\"double\":16.89,\"string\":"
  95. "\"hello\",\"boolTrue\":"
  96. "true,\"boolFalse\":false,\"null\":null},"
  97. "\"arrayInt\":[16,16,16,16,16],\"arrayDouble\":[16.89,16.89,16.89,16.89,16.89],\"arrayString\":[\"hello\",\"hello\","
  98. "\"hello\",\"hello\","
  99. "\"hello\"],\"array\":[16,16.89,\"hello\","
  100. "true,false,null],\"arrayItem\":[{\"inter\":16,\"double\":16.89,\"string\":\"hello\",\"boolTrue\":true,\"boolFalse\":"
  101. "false,\"null\":null},{"
  102. "\"inter\":16,\"double\":16.89,\"string\":"
  103. "\"hello\",\"boolTrue\":true,\"boolFalse\":false,\"null\":null}]},\"item3\":{\"inter\":16,\"double\":16.89,\"string\":"
  104. "\"hello\",\"boolTrue\":"
  105. "true,\"boolFalse\":false,\"null\":null,"
  106. "\"item\":{\"inter\":16,\"double\":16.89,\"string\":\"hello\",\"boolTrue\":true,\"boolFalse\":false,\"null\":null},"
  107. "\"arrayInt\":[16,16,16,16,"
  108. "16],\"arrayDouble\":[16.89,16.89,16.89,"
  109. "16.89,16.89],\"arrayString\":[\"hello\",\"hello\",\"hello\",\"hello\",\"hello\"],\"array\":[16,16.89,\"hello\",true,"
  110. "false,null],"
  111. "\"arrayItem\":[{\"inter\":16,\"double\":16.89,"
  112. "\"string\":\"hello\",\"boolTrue\":true,\"boolFalse\":false,\"null\":null},{\"inter\":16,\"double\":16.89,\"string\":"
  113. "\"hello\",\"boolTrue\":"
  114. "true,\"boolFalse\":false,\"null\":null}]}"
  115. ",\"item4\":{\"inter\":16,\"double\":16.89,\"string\":\"hello\",\"boolTrue\":true,\"boolFalse\":false,\"null\":null,"
  116. "\"item\":{\"inter\":16,"
  117. "\"double\":16.89,\"string\":\"hello\","
  118. "\"boolTrue\":true,\"boolFalse\":false,\"null\":null},\"arrayInt\":[16,16,16,16,16],\"arrayDouble\":[16.89,16.89,16.89,"
  119. "16.89,16.89],"
  120. "\"arrayString\":[\"hello\",\"hello\",\"hello\","
  121. "\"hello\",\"hello\"],\"array\":[16,16.89,\"hello\",true,false,null],\"arrayItem\":[{\"inter\":16,\"double\":16.89,"
  122. "\"string\":\"hello\","
  123. "\"boolTrue\":true,\"boolFalse\":false,"
  124. "\"null\":null},{\"inter\":16,\"double\":16.89,\"string\":\"hello\",\"boolTrue\":true,\"boolFalse\":false,\"null\":null}]"
  125. "}}";
  126. printfJsonCompare("混合对象", jsonstr);
  127. }
  128. static void testObjectJsonMemory(void)
  129. {
  130. static const char jsonstr[] =
  131. "{\"message\":\"success感谢又拍云(upyun.com)提供CDN赞助\",\"status\":200,\"date\":\"20230822\",\"time\":\"2023-08-22 "
  132. "09:44:54\",\"cityInfo\":{\"city\":\"郑州市\",\"citykey\":\"101180101\",\"parent\":\"河南\",\"updateTime\":\"07:46\"},"
  133. "\"data\":{\"shidu\":"
  134. "\"85%\",\"pm25\":20,\"pm10\":56,"
  135. "\"quality\":\"良\",\"wendu\":\"29\",\"ganmao\":\"极少数敏感人群应减少户外活动\",\"forecast\":[{\"date\":\"22\",\"high\":"
  136. "\"高温 "
  137. "35℃\",\"low\":\"低温 "
  138. "23℃\",\"ymd\":\"2023-08-22\",\"week\":\"星期二\",\"sunrise\":\"05:51\",\"sunset\":\"19:05\",\"aqi\":78,\"fx\":\"东南风\","
  139. "\"fl\":\"2级\","
  140. "\"type\":\"晴\",\"notice\":"
  141. "\"愿你拥有比阳光明媚的心情\"},{\"date\":\"23\",\"high\":\"高温 33℃\",\"low\":\"低温 "
  142. "23℃\",\"ymd\":\"2023-08-23\",\"week\":\"星期三\",\"sunrise\":\"05:52\",\"sunset\":\"19:04\",\"aqi\":71,\"fx\":\"南风\","
  143. "\"fl\":\"2级\","
  144. "\"type\":\"中雨\",\"notice\":"
  145. "\"记得随身携带雨伞哦\"},{\"date\":\"24\",\"high\":\"高温 31℃\",\"low\":\"低温 "
  146. "21℃\",\"ymd\":\"2023-08-24\",\"week\":\"星期四\",\"sunrise\":\"05:52\",\"sunset\":\"19:03\",\"aqi\":74,\"fx\":\"东风\","
  147. "\"fl\":\"2级\","
  148. "\"type\":\"晴\",\"notice\":"
  149. "\"愿你拥有比阳光明媚的心情\"},{\"date\":\"25\",\"high\":\"高温 30℃\",\"low\":\"低温 "
  150. "23℃\",\"ymd\":\"2023-08-25\",\"week\":\"星期五\",\"sunrise\":\"05:53\",\"sunset\":\"19:02\",\"aqi\":93,\"fx\":\"东风\","
  151. "\"fl\":\"1级\","
  152. "\"type\":\"小雨\",\"notice\":"
  153. "\"雨虽小,注意保暖别感冒\"},{\"date\":\"26\",\"high\":\"高温 25℃\",\"low\":\"低温 "
  154. "22℃\",\"ymd\":\"2023-08-26\",\"week\":\"星期六\",\"sunrise\":\"05:54\",\"sunset\":\"19:00\",\"aqi\":80,\"fx\":\"东北风\","
  155. "\"fl\":\"1级\","
  156. "\"type\":\"阴\",\"notice\":"
  157. "\"不要被阴云遮挡住好心情\"},{\"date\":\"27\",\"high\":\"高温 27℃\",\"low\":\"低温 "
  158. "20℃\",\"ymd\":\"2023-08-27\",\"week\":\"星期日\",\"sunrise\":\"05:55\",\"sunset\":\"18:59\",\"aqi\":74,\"fx\":\"西北风\","
  159. "\"fl\":\"1级\","
  160. "\"type\":\"阴\",\"notice\":"
  161. "\"不要被阴云遮挡住好心情\"},{\"date\":\"28\",\"high\":\"高温 30℃\",\"low\":\"低温 "
  162. "20℃\",\"ymd\":\"2023-08-28\",\"week\":\"星期 "
  163. "一\",\"sunrise\":\"05:55\",\"sunset\":\"18:58\",\"aqi\":80,\"fx\":\"东北风\",\"fl\":\"2级\",\"type\":\"多云\",\"notice\":"
  164. "\"阴晴之间,谨防紫外线侵扰\"},{\"date\":\"29\",\"high\":"
  165. "\"高温 30℃\",\"low\":\"低温 "
  166. "20℃\",\"ymd\":\"2023-08-29\",\"week\":\"星期二\",\"sunrise\":\"05:56\",\"sunset\":\"18:56\",\"aqi\":80,\"fx\":\"东北风\","
  167. "\"fl\":\"2级\","
  168. "\"type\":\"多云\",\"notice\":"
  169. "\"阴晴之间,谨防紫外线侵扰\"},{\"date\":\"30\",\"high\":\"高温 31℃\",\"low\":\"低温 "
  170. "20℃\",\"ymd\":\"2023-08-30\",\"week\":\"星期三\",\"sunrise\":\"05:57\",\"sunset\":\"18:55\",\"aqi\":92,\"fx\":\"南风\","
  171. "\"fl\":\"1级\","
  172. "\"type\":\"晴\",\"notice\":"
  173. "\"愿你拥有比阳光明媚的心情\"},{\"date\":\"31\",\"high\":\"高温 33℃\",\"low\":\" 低温 "
  174. "22℃\",\"ymd\":\"2023-08-31\",\"week\":\"星期四\",\"sunrise\":\"05:57\",\"sunset\":\"18:54\",\"aqi\":91,\"fx\":\"南风\","
  175. "\"fl\":\"1级\","
  176. "\"type\":\"晴\",\"notice\":"
  177. "\"愿你拥有比阳光明媚的心情\"},{\"date\":\"01\",\"high\":\"高温 34℃\",\"low\":\"低温 "
  178. "23℃\",\"ymd\":\"2023-09-01\",\"week\":\"星期五\",\"sunrise\":\"05:58\",\"sunset\":\"18:52\",\"aqi\":91,\"fx\":\"西风\","
  179. "\"fl\":\"1级\","
  180. "\"type\":\"晴\",\"notice\":"
  181. "\"愿你拥有比阳光明媚的心情\"},{\"date\":\"02\",\"high\":\"高温 36℃\",\"low\":\"低温 "
  182. "25℃\",\"ymd\":\"2023-09-02\",\"week\":\"星期六\",\"sunrise\":\"05:59\",\"sunset\":\"18:51\",\"aqi\":78,\"fx\":\"南风\","
  183. "\"fl\":\"1级\","
  184. "\"type\":\"阴\",\"notice\":"
  185. "\"不要被阴云遮挡住好心情\"},{\"date\":\"03\",\"high\":\"高温 35℃\",\"low\":\"低温 "
  186. "24℃\",\"ymd\":\"2023-09-03\",\"week\":\"星期日\",\"sunrise\":\"06:00\",\"sunset\":\"18:50\",\"aqi\":82,\"fx\":\"东北风\","
  187. "\"fl\":\"1级\","
  188. "\"type\":\"晴\",\"notice\":"
  189. "\"愿你拥有比阳光明媚的心情\"},{\"date\":\"04\",\"high\":\"高温 35℃\",\"low\":\"低温 "
  190. "25℃\",\"ymd\":\"2023-09-04\",\"week\":\"星期一\",\"sunrise\":\"06:00\",\"sunset\":\"18:48\",\"aqi\":88,\"fx\":\"南风\","
  191. "\"fl\":\"2级\","
  192. "\"type\":\"晴\",\"notice\":"
  193. "\"愿你拥有比阳光明媚的心情\"},{\"date\":\"05\",\"high\":\"高温 35℃\",\"low\":\"低温 "
  194. "25℃\",\"ymd\":\"2023-09-05\",\"week\":\"星期二\",\"sunrise\":\"06:01\",\"sunset\":\"18:47\",\"aqi\":58,\"fx\":\"南风\","
  195. "\"fl\":\"2级\","
  196. "\"type\":\"阴\",\"notice\":"
  197. "\"不要被阴云遮挡住好心情\"}],\"yesterday\":{\"date\":\"21\",\"high\":\"高温 34℃\",\"low\":\"低温 "
  198. "24℃\",\"ymd\":\"2023-08-21\",\"week\":\" "
  199. "星期一\",\"sunrise\":\"05:50\",\"sunset\":\"19:07\",\"aqi\":60,\"fx\":\"西风\",\"fl\":\"2级\",\"type\":\"小雨\","
  200. "\"notice\":"
  201. "\"雨虽小,注意保暖别感冒\"}}}";
  202. printfJsonCompare("经典天气对象", jsonstr);
  203. }
  204. static void testArrayJsonMemory(void)
  205. {
  206. static const char jsonstr[] =
  207. "{\"item1\":{\"arrayInt\":[16,16,16,16,16,16,16,16,16,16],\"arrayDouble\":[16.89,16.89,16.89,16.89,16.89,16.89,16.89,16."
  208. "89,16.89,16.89],"
  209. "\"arrayString\":[\"hello\",\"hello\","
  210. "\"hello\",\"hello\",\"hello\",\"hello\",\"hello\",\"hello\",\"hello\",\"hello\"],\"array\":[16,16.89,\"hello\",true,"
  211. "false,null,16,16.89,"
  212. "\"hello\",true,false,null]},\"item2\":{"
  213. "\"arrayInt\":[16,16,16,16,16,16,16,16,16,16],\"arrayDouble\":[16.89,16.89,16.89,16.89,16.89,16.89,16.89,16.89,16.89,16."
  214. "89],\"arrayString\":["
  215. "\"hello\",\"hello\",\"hello\",\"hello\","
  216. "\"hello\",\"hello\",\"hello\",\"hello\",\"hello\",\"hello\"],\"array\":[16,16.89,\"hello\",true,false,null,16,16.89,"
  217. "\"hello\",true,false,"
  218. "null]},\"item3\":{\"arrayInt\":[16,16,16,"
  219. "16,16,16,16,16,16,16],\"arrayDouble\":[16.89,16.89,16.89,16.89,16.89,16.89,16.89,16.89,16.89,16.89],\"arrayString\":["
  220. "\"hello\",\"hello\","
  221. "\"hello\",\"hello\",\"hello\",\"hello\","
  222. "\"hello\",\"hello\",\"hello\",\"hello\"],\"array\":[16,16.89,\"hello\",true,false,null,16,16.89,\"hello\",true,false,"
  223. "null]},\"item4\":{"
  224. "\"arrayInt\":[16,16,16,16,16,16,16,16,16,16],"
  225. "\"arrayDouble\":[16.89,16.89,16.89,16.89,16.89,16.89,16.89,16.89,16.89,16.89],\"arrayString\":[\"hello\",\"hello\","
  226. "\"hello\",\"hello\","
  227. "\"hello\",\"hello\",\"hello\",\"hello\","
  228. "\"hello\",\"hello\"],\"array\":[16,16.89,\"hello\",true,false,null,16,16.89,\"hello\",true,false,null]}}";
  229. printfJsonCompare("深度数组", jsonstr);
  230. }
  231. static void testSmallMixedJsonMemory(void)
  232. {
  233. static const char jsonstr[] =
  234. "{\"inter\":16,\"double\":16.89,\"string\":\"hello\",\"boolTrue\":true,\"boolFalse\":false,\"null\":null}";
  235. printfJsonCompare("小型混合对象", jsonstr);
  236. }
  237. static void testSmallStringJsonMemory(void)
  238. {
  239. static const char jsonstr[] =
  240. "{\"inter\":\"16\",\"double\":\"16.89\",\"string\":\"hello\",\"boolTrue\":\"true\",\"boolFalse\":\"false\",\"null\":"
  241. "\"null\"}";
  242. printfJsonCompare("小型字符串对象", jsonstr);
  243. }
  244. static void testCompressedBusinessJsonMemory(void)
  245. {
  246. static const char jsonstr[] =
  247. "{\"0\":\"0\",\"1\":\"189774523\",\"2\":{\"7\":\"3\",\"8\":\"103\",\"9\":\"37\",\"20\":\"0\",\"26\":\"37\",\"27\":"
  248. "\"367\",\"28\":\"367\",\"s\":\"0\",\"t\":\"0\",\"a\":\"24.98\",\"2a\":\"0\",\"1p\":\"23628\"},\"3\":\"0\",\"22\":"
  249. "\"epmgrow1105\",\"23\":\"0\",\"29\":\"0\",\"i\":\"4\",\"b\":\"900\",\"c\":\"1\",\"rsrp\":\"-111\",\"rsrq\":\"-4\","
  250. "\"sinr\":\"0\",\"soc\":\"XXXXXXX\",\"j\":\"0\",\"g\":\"898604asdf0210\",\"h\":\"866968798839\",\"d\":\"1.3.5."
  251. "00.20991231\",\"f\":\"0\",\"k\":\"1\",\"l\":\"20000\",\"m\":\"20000\",\"u\":\"0\",\"v\":\"0\",\"e\":\"1\",\"w\":\"0."
  252. "00\",\"n\":\"0\",\"2h\":\"0\",\"o\":\"30\",\"1v\":\"12000\",\"2c\":\"0\",\"p\":\"1\",\"q\":\"1\",\"x\":\"0\",\"y\":"
  253. "\"167\",\"r\":\"0\",\"1x\":\"0\",\"1w\":\"0\",\"1y\":\"100.00\",\"1u\":\"0\"}";
  254. printfJsonCompare("压缩业务对象", jsonstr);
  255. }
  256. void testMemoryRunner(void)
  257. {
  258. UnitySetTestFile(__FILE__);
  259. unityTestSetAllocSimulation(1U);
  260. RUN_TEST(testMixedJsonMemory);
  261. RUN_TEST(testObjectJsonMemory);
  262. RUN_TEST(testArrayJsonMemory);
  263. RUN_TEST(testSmallMixedJsonMemory);
  264. RUN_TEST(testSmallStringJsonMemory);
  265. RUN_TEST(testCompressedBusinessJsonMemory);
  266. unityTestSetAllocSimulation(0U);
  267. }