You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 

577 lines
12 KiB

  1. using System;
  2. using System.Collections;
  3. using System.Collections.Generic;
  4. using System.IO;
  5. using System.Text;
  6. namespace MiniJSONV
  7. {
  8. public static class Json
  9. {
  10. public static object Deserialize(string json)
  11. {
  12. if (json == null)
  13. {
  14. return null;
  15. }
  16. return Json.Parser.Parse(json);
  17. }
  18. public static string Serialize(object obj)
  19. {
  20. return Json.Serializer.Serialize(obj);
  21. }
  22. private sealed class Parser : IDisposable
  23. {
  24. private Parser(string jsonString)
  25. {
  26. this.json = new StringReader(jsonString);
  27. }
  28. public static bool IsWordBreak(char c)
  29. {
  30. return char.IsWhiteSpace(c) || "{}[],:\"".IndexOf(c) != -1;
  31. }
  32. public static object Parse(string jsonString)
  33. {
  34. object result;
  35. using (Json.Parser parser = new Json.Parser(jsonString))
  36. {
  37. result = parser.ParseValue();
  38. }
  39. return result;
  40. }
  41. public void Dispose()
  42. {
  43. this.json.Dispose();
  44. this.json = null;
  45. }
  46. private Dictionary<string, object> ParseObject()
  47. {
  48. Dictionary<string, object> dictionary = new Dictionary<string, object>();
  49. this.json.Read();
  50. for (;;)
  51. {
  52. Json.Parser.TOKEN nextToken = this.NextToken;
  53. switch (nextToken)
  54. {
  55. case Json.Parser.TOKEN.NONE:
  56. goto IL_37;
  57. default:
  58. if (nextToken != Json.Parser.TOKEN.COMMA)
  59. {
  60. string text = this.ParseString();
  61. if (text == null)
  62. {
  63. goto Block_2;
  64. }
  65. if (this.NextToken != Json.Parser.TOKEN.COLON)
  66. {
  67. goto Block_3;
  68. }
  69. this.json.Read();
  70. dictionary[text] = this.ParseValue();
  71. }
  72. break;
  73. case Json.Parser.TOKEN.CURLY_CLOSE:
  74. return dictionary;
  75. }
  76. }
  77. IL_37:
  78. return null;
  79. Block_2:
  80. return null;
  81. Block_3:
  82. return null;
  83. }
  84. private List<object> ParseArray()
  85. {
  86. List<object> list = new List<object>();
  87. this.json.Read();
  88. bool flag = true;
  89. while (flag)
  90. {
  91. Json.Parser.TOKEN nextToken = this.NextToken;
  92. switch (nextToken)
  93. {
  94. case Json.Parser.TOKEN.SQUARED_CLOSE:
  95. flag = false;
  96. break;
  97. default:
  98. {
  99. if (nextToken == Json.Parser.TOKEN.NONE)
  100. {
  101. return null;
  102. }
  103. object item = this.ParseByToken(nextToken);
  104. list.Add(item);
  105. break;
  106. }
  107. case Json.Parser.TOKEN.COMMA:
  108. break;
  109. }
  110. }
  111. return list;
  112. }
  113. private object ParseValue()
  114. {
  115. Json.Parser.TOKEN nextToken = this.NextToken;
  116. return this.ParseByToken(nextToken);
  117. }
  118. private object ParseByToken(Json.Parser.TOKEN token)
  119. {
  120. switch (token)
  121. {
  122. case Json.Parser.TOKEN.STRING:
  123. return this.ParseString();
  124. case Json.Parser.TOKEN.NUMBER:
  125. return this.ParseNumber();
  126. case Json.Parser.TOKEN.TRUE:
  127. return true;
  128. case Json.Parser.TOKEN.FALSE:
  129. return false;
  130. case Json.Parser.TOKEN.NULL:
  131. return null;
  132. default:
  133. switch (token)
  134. {
  135. case Json.Parser.TOKEN.CURLY_OPEN:
  136. return this.ParseObject();
  137. case Json.Parser.TOKEN.SQUARED_OPEN:
  138. return this.ParseArray();
  139. }
  140. return null;
  141. }
  142. }
  143. private string ParseString()
  144. {
  145. StringBuilder stringBuilder = new StringBuilder();
  146. this.json.Read();
  147. bool flag = true;
  148. while (flag)
  149. {
  150. if (this.json.Peek() == -1)
  151. {
  152. break;
  153. }
  154. char nextChar = this.NextChar;
  155. if (nextChar != '"')
  156. {
  157. if (nextChar != '\\')
  158. {
  159. stringBuilder.Append(nextChar);
  160. }
  161. else if (this.json.Peek() == -1)
  162. {
  163. flag = false;
  164. }
  165. else
  166. {
  167. nextChar = this.NextChar;
  168. switch (nextChar)
  169. {
  170. case 'r':
  171. stringBuilder.Append('\r');
  172. break;
  173. default:
  174. if (nextChar != '"' && nextChar != '/' && nextChar != '\\')
  175. {
  176. if (nextChar != 'b')
  177. {
  178. if (nextChar != 'f')
  179. {
  180. if (nextChar == 'n')
  181. {
  182. stringBuilder.Append('\n');
  183. }
  184. }
  185. else
  186. {
  187. stringBuilder.Append('\f');
  188. }
  189. }
  190. else
  191. {
  192. stringBuilder.Append('\b');
  193. }
  194. }
  195. else
  196. {
  197. stringBuilder.Append(nextChar);
  198. }
  199. break;
  200. case 't':
  201. stringBuilder.Append('\t');
  202. break;
  203. case 'u':
  204. {
  205. char[] array = new char[4];
  206. for (int i = 0; i < 4; i++)
  207. {
  208. array[i] = this.NextChar;
  209. }
  210. stringBuilder.Append((char)Convert.ToInt32(new string(array), 16));
  211. break;
  212. }
  213. }
  214. }
  215. }
  216. else
  217. {
  218. flag = false;
  219. }
  220. }
  221. return stringBuilder.ToString();
  222. }
  223. private object ParseNumber()
  224. {
  225. string nextWord = this.NextWord;
  226. if (nextWord.IndexOf('.') == -1)
  227. {
  228. long num;
  229. long.TryParse(nextWord, out num);
  230. return num;
  231. }
  232. double num2;
  233. double.TryParse(nextWord, out num2);
  234. return num2;
  235. }
  236. private void EatWhitespace()
  237. {
  238. while (char.IsWhiteSpace(this.PeekChar))
  239. {
  240. this.json.Read();
  241. if (this.json.Peek() == -1)
  242. {
  243. break;
  244. }
  245. }
  246. }
  247. private char PeekChar
  248. {
  249. get
  250. {
  251. return Convert.ToChar(this.json.Peek());
  252. }
  253. }
  254. private char NextChar
  255. {
  256. get
  257. {
  258. return Convert.ToChar(this.json.Read());
  259. }
  260. }
  261. private string NextWord
  262. {
  263. get
  264. {
  265. StringBuilder stringBuilder = new StringBuilder();
  266. while (!Json.Parser.IsWordBreak(this.PeekChar))
  267. {
  268. stringBuilder.Append(this.NextChar);
  269. if (this.json.Peek() == -1)
  270. {
  271. break;
  272. }
  273. }
  274. return stringBuilder.ToString();
  275. }
  276. }
  277. private Json.Parser.TOKEN NextToken
  278. {
  279. get
  280. {
  281. this.EatWhitespace();
  282. if (this.json.Peek() == -1)
  283. {
  284. return Json.Parser.TOKEN.NONE;
  285. }
  286. char peekChar = this.PeekChar;
  287. switch (peekChar)
  288. {
  289. case ',':
  290. this.json.Read();
  291. return Json.Parser.TOKEN.COMMA;
  292. case '-':
  293. case '0':
  294. case '1':
  295. case '2':
  296. case '3':
  297. case '4':
  298. case '5':
  299. case '6':
  300. case '7':
  301. case '8':
  302. case '9':
  303. return Json.Parser.TOKEN.NUMBER;
  304. default:
  305. switch (peekChar)
  306. {
  307. case '[':
  308. return Json.Parser.TOKEN.SQUARED_OPEN;
  309. default:
  310. switch (peekChar)
  311. {
  312. case '{':
  313. return Json.Parser.TOKEN.CURLY_OPEN;
  314. default:
  315. if (peekChar != '"')
  316. {
  317. string nextWord = this.NextWord;
  318. if (nextWord != null)
  319. {
  320. if (nextWord == "false")
  321. {
  322. return Json.Parser.TOKEN.FALSE;
  323. }
  324. if (nextWord == "true")
  325. {
  326. return Json.Parser.TOKEN.TRUE;
  327. }
  328. if (nextWord == "null")
  329. {
  330. return Json.Parser.TOKEN.NULL;
  331. }
  332. }
  333. return Json.Parser.TOKEN.NONE;
  334. }
  335. return Json.Parser.TOKEN.STRING;
  336. case '}':
  337. this.json.Read();
  338. return Json.Parser.TOKEN.CURLY_CLOSE;
  339. }
  340. break;
  341. case ']':
  342. this.json.Read();
  343. return Json.Parser.TOKEN.SQUARED_CLOSE;
  344. }
  345. break;
  346. case ':':
  347. return Json.Parser.TOKEN.COLON;
  348. }
  349. }
  350. }
  351. private const string WORD_BREAK = "{}[],:\"";
  352. private StringReader json;
  353. private enum TOKEN
  354. {
  355. NONE,
  356. CURLY_OPEN,
  357. CURLY_CLOSE,
  358. SQUARED_OPEN,
  359. SQUARED_CLOSE,
  360. COLON,
  361. COMMA,
  362. STRING,
  363. NUMBER,
  364. TRUE,
  365. FALSE,
  366. NULL
  367. }
  368. }
  369. private sealed class Serializer
  370. {
  371. private Serializer()
  372. {
  373. this.builder = new StringBuilder();
  374. }
  375. public static string Serialize(object obj)
  376. {
  377. Json.Serializer serializer = new Json.Serializer();
  378. serializer.SerializeValue(obj);
  379. return serializer.builder.ToString();
  380. }
  381. private void SerializeValue(object value)
  382. {
  383. string str;
  384. IList anArray;
  385. IDictionary obj;
  386. if (value == null)
  387. {
  388. this.builder.Append("null");
  389. }
  390. else if ((str = (value as string)) != null)
  391. {
  392. this.SerializeString(str);
  393. }
  394. else if (value is bool)
  395. {
  396. this.builder.Append((!(bool)value) ? "false" : "true");
  397. }
  398. else if ((anArray = (value as IList)) != null)
  399. {
  400. this.SerializeArray(anArray);
  401. }
  402. else if ((obj = (value as IDictionary)) != null)
  403. {
  404. this.SerializeObject(obj);
  405. }
  406. else if (value is char)
  407. {
  408. this.SerializeString(new string((char)value, 1));
  409. }
  410. else
  411. {
  412. this.SerializeOther(value);
  413. }
  414. }
  415. private void SerializeObject(IDictionary obj)
  416. {
  417. bool flag = true;
  418. this.builder.Append('{');
  419. IEnumerator enumerator = obj.Keys.GetEnumerator();
  420. try
  421. {
  422. while (enumerator.MoveNext())
  423. {
  424. object obj2 = enumerator.Current;
  425. if (!flag)
  426. {
  427. this.builder.Append(',');
  428. }
  429. this.SerializeString(obj2.ToString());
  430. this.builder.Append(':');
  431. this.SerializeValue(obj[obj2]);
  432. flag = false;
  433. }
  434. }
  435. finally
  436. {
  437. IDisposable disposable;
  438. if ((disposable = (enumerator as IDisposable)) != null)
  439. {
  440. disposable.Dispose();
  441. }
  442. }
  443. this.builder.Append('}');
  444. }
  445. private void SerializeArray(IList anArray)
  446. {
  447. this.builder.Append('[');
  448. bool flag = true;
  449. IEnumerator enumerator = anArray.GetEnumerator();
  450. try
  451. {
  452. while (enumerator.MoveNext())
  453. {
  454. object value = enumerator.Current;
  455. if (!flag)
  456. {
  457. this.builder.Append(',');
  458. }
  459. this.SerializeValue(value);
  460. flag = false;
  461. }
  462. }
  463. finally
  464. {
  465. IDisposable disposable;
  466. if ((disposable = (enumerator as IDisposable)) != null)
  467. {
  468. disposable.Dispose();
  469. }
  470. }
  471. this.builder.Append(']');
  472. }
  473. private void SerializeString(string str)
  474. {
  475. this.builder.Append('"');
  476. char[] array = str.ToCharArray();
  477. foreach (char c in array)
  478. {
  479. switch (c)
  480. {
  481. case '\b':
  482. this.builder.Append("\\b");
  483. break;
  484. case '\t':
  485. this.builder.Append("\\t");
  486. break;
  487. case '\n':
  488. this.builder.Append("\\n");
  489. break;
  490. default:
  491. if (c != '"')
  492. {
  493. if (c != '\\')
  494. {
  495. int num = Convert.ToInt32(c);
  496. if (num >= 32 && num <= 126)
  497. {
  498. this.builder.Append(c);
  499. }
  500. else
  501. {
  502. this.builder.Append("\\u");
  503. this.builder.Append(num.ToString("x4"));
  504. }
  505. }
  506. else
  507. {
  508. this.builder.Append("\\\\");
  509. }
  510. }
  511. else
  512. {
  513. this.builder.Append("\\\"");
  514. }
  515. break;
  516. case '\f':
  517. this.builder.Append("\\f");
  518. break;
  519. case '\r':
  520. this.builder.Append("\\r");
  521. break;
  522. }
  523. }
  524. this.builder.Append('"');
  525. }
  526. private void SerializeOther(object value)
  527. {
  528. if (value is float)
  529. {
  530. this.builder.Append(((float)value).ToString("R"));
  531. }
  532. else if (value is int || value is uint || value is long || value is sbyte || value is byte || value is short || value is ushort || value is ulong)
  533. {
  534. this.builder.Append(value);
  535. }
  536. else if (value is double || value is decimal)
  537. {
  538. this.builder.Append(Convert.ToDouble(value).ToString("R"));
  539. }
  540. else
  541. {
  542. this.SerializeString(value.ToString());
  543. }
  544. }
  545. private StringBuilder builder;
  546. }
  547. }
  548. }