IO.cs 39 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using System.IO;
  5. using System.Reflection;
  6. using System.Xml;
  7. using CommonLang.Xml;
  8. using System.IO.Compression;
  9. namespace CommonLang.IO
  10. {
  11. public static class IOUtil
  12. {
  13. public static byte TryReadByte(Stream data)
  14. {
  15. int b = data.ReadByte();
  16. if (b < 0) { throw new IOException("EOF of stream"); }
  17. return (byte)b;
  18. }
  19. /// <summary>
  20. /// 从输入流复制到输出流
  21. /// </summary>
  22. /// <param name="input">输入流</param>
  23. /// <param name="output">输出流</param>
  24. /// <param name="total_bytes">总共复制多少字节</param>
  25. /// <param name="progress">进度回调,返回False表示终止进程</param>
  26. /// <param name="buffer_size">缓冲区大小</param>
  27. /// <returns></returns>
  28. public static bool ReadTo(Stream input, Stream output, long total_bytes, Predicate<int> progress, int buffer_size = 16384)
  29. {
  30. byte[] io_buffer = new byte[buffer_size];
  31. long total_readed = 0;
  32. while (total_readed < total_bytes)
  33. {
  34. if (!progress(0)) return false;
  35. int expect = (int)Math.Min(io_buffer.Length, total_bytes - total_readed);
  36. int readed = input.Read(io_buffer, 0, expect);
  37. total_readed += readed;
  38. if (!progress(readed)) return false;
  39. output.Write(io_buffer, 0, readed);
  40. }
  41. output.Flush();
  42. return true;
  43. }
  44. public static byte[] ReadExpect(Stream input, int count)
  45. {
  46. if (count == 0) return new byte[0];
  47. byte[] data = new byte[count];
  48. ReadToEnd(input, data, 0, count);
  49. return data;
  50. }
  51. public static void ReadToEnd(Stream input, byte[] data, int offset, int count)
  52. {
  53. if (count == 0) return;
  54. int readed = input.Read(data, offset, count);
  55. if (readed <= 0)
  56. throw new IOException("EOF of stream");
  57. while (readed < count)
  58. {
  59. int bytes = input.Read(data, offset + readed, count - readed);
  60. if (bytes <= 0)
  61. {
  62. throw new IOException("EOF of stream");
  63. }
  64. readed += bytes;
  65. }
  66. }
  67. public static void ReadToEnd(Stream src, Stream dst)
  68. {
  69. byte[] buffer = new byte[1024];
  70. while (true)
  71. {
  72. int readed = src.Read(buffer, 0, buffer.Length);
  73. if (readed > 0)
  74. {
  75. dst.Write(buffer, 0, readed);
  76. }
  77. else
  78. {
  79. break;
  80. }
  81. }
  82. }
  83. public static byte[] ReadToEnd(Stream src)
  84. {
  85. using (MemoryStream dst = new MemoryStream())
  86. {
  87. ReadToEnd(src, dst);
  88. return dst.ToArray();
  89. }
  90. }
  91. public static void WriteToEnd(Stream output, byte[] data, int offset, int count)
  92. {
  93. output.Write(data, offset, count);
  94. }
  95. public static T CloneIExternalizable<T>(IExternalizableFactory decode, T src) where T : IExternalizable, new()
  96. {
  97. if (src != null)
  98. {
  99. T ret = new T();
  100. {
  101. using (MemoryStream ms = new MemoryStream())
  102. {
  103. src.WriteExternal(new OutputStream(ms, decode));
  104. ms.Position = 0;
  105. ret.ReadExternal(new InputStream(ms, decode));
  106. }
  107. }
  108. return ret;
  109. }
  110. return default(T);
  111. }
  112. public static IExternalizable Clone(IExternalizableFactory decode, IExternalizable src)
  113. {
  114. IExternalizable ret = null;
  115. if (src != null)
  116. {
  117. using (MemoryStream ms = new MemoryStream())
  118. {
  119. new OutputStream(ms, decode).PutExt(src);
  120. ms.Position = 0;
  121. ret = new InputStream(ms, decode).GetExtAny();
  122. }
  123. }
  124. return ret;
  125. }
  126. public static byte[] LoadFromAssembly(Assembly assembly, string resource)
  127. {
  128. Stream stream = assembly.GetManifestResourceStream(resource);
  129. byte[] ret = new byte[stream.Length];
  130. /*stream.Read(ret, 0, ret.Length);*/
  131. IOUtil.ReadToEnd(stream, ret, 0, ret.Length);
  132. return ret;
  133. }
  134. public static byte[] ObjectToBin(IExternalizableFactory decode, IExternalizable ext)
  135. {
  136. using (MemoryStream ms = new MemoryStream())
  137. {
  138. OutputStream output = new OutputStream(ms, decode);
  139. output.PutExt(ext);
  140. int len = (int)ms.Position;
  141. byte[] buffer = ms.GetBuffer();
  142. Array.Resize(ref buffer, len);
  143. return buffer;
  144. }
  145. }
  146. public static T BinToObject<T>(IExternalizableFactory decode, byte[] data) where T : IExternalizable, new()
  147. {
  148. if (data == null || data.Length == 0) return default(T);
  149. using (MemoryStream ms = new MemoryStream(data))
  150. {
  151. var input = new InputStream(ms, decode);
  152. return input.GetExt<T>();
  153. }
  154. }
  155. public static IExternalizable BinToObjectAny(IExternalizableFactory decode, byte[] data)
  156. {
  157. if (data == null || data.Length == 0) return null;
  158. using (MemoryStream ms = new MemoryStream(data))
  159. {
  160. InputStream input = new InputStream(ms, decode);
  161. return input.GetExtAny();
  162. }
  163. }
  164. public static void Zip(Stream src, Stream dst)
  165. {
  166. GZipStream zipStream = null;
  167. try
  168. {
  169. zipStream = new GZipStream(dst, CompressionMode.Compress, true);
  170. byte[] buffer = new byte[Math.Min(1024, src.Length)];
  171. while (true)
  172. {
  173. int bytesRead = src.Read(buffer, 0, buffer.Length);
  174. if (bytesRead == 0)
  175. break;
  176. zipStream.Write(buffer, 0, bytesRead);
  177. }
  178. zipStream.Flush();
  179. dst.Flush();
  180. }
  181. finally
  182. {
  183. if (zipStream != null) zipStream.Close();
  184. }
  185. }
  186. public static void Unzip(Stream src, Stream dst)
  187. {
  188. GZipStream zipStream = null;
  189. try
  190. {
  191. // Create a compression stream pointing to the destiantion stream
  192. zipStream = new GZipStream(src, CompressionMode.Decompress, true);
  193. // Read the footer to determine the length of the destiantion file
  194. byte[] buffer = new byte[1024];
  195. // Read the compressed data into the buffer
  196. while (true)
  197. {
  198. int bytesRead = zipStream.Read(buffer, 0, buffer.Length);
  199. if (bytesRead == 0)
  200. break;
  201. dst.Write(buffer, 0, bytesRead);
  202. }
  203. dst.Flush();
  204. }
  205. finally
  206. {
  207. if (zipStream != null) zipStream.Close();
  208. }
  209. }
  210. public static byte[] Zip(byte[] data)
  211. {
  212. // Read in the compressed source stream
  213. MemoryStream src = new MemoryStream(data);
  214. MemoryStream dst = new MemoryStream(data.Length * 2);
  215. try
  216. {
  217. Zip(src, dst);
  218. byte[] ret = new byte[dst.Position];
  219. Array.Copy(dst.GetBuffer(), 0, ret, 0, ret.Length);
  220. return ret;
  221. }
  222. catch (Exception err)
  223. {
  224. Console.WriteLine(err.Message + "\r" + err.StackTrace);
  225. return null;
  226. }
  227. finally
  228. {
  229. if (src != null) src.Close();
  230. if (dst != null) dst.Close();
  231. }
  232. }
  233. public static byte[] Unzip(byte[] data)
  234. {
  235. // Read in the compressed source stream
  236. MemoryStream src = new MemoryStream(data);
  237. MemoryStream dst = new MemoryStream(data.Length);
  238. try
  239. {
  240. Unzip(src, dst);
  241. byte[] ret = new byte[dst.Position];
  242. Array.Copy(dst.GetBuffer(), 0, ret, 0, ret.Length);
  243. return ret;
  244. }
  245. catch (Exception err)
  246. {
  247. Console.WriteLine(err.Message + "\r" + err.StackTrace);
  248. return null;
  249. }
  250. finally
  251. {
  252. if (src != null) src.Close();
  253. if (dst != null) dst.Close();
  254. }
  255. }
  256. }
  257. public interface IExternalizable
  258. {
  259. void WriteExternal(IOutputStream output);
  260. void ReadExternal(IInputStream input);
  261. }
  262. public interface IExternalizableFactory
  263. {
  264. int GetTypeID(Type type);
  265. Type GetType(int id);
  266. }
  267. public enum DataType : byte
  268. {
  269. NA = 0,
  270. U8 = 1,
  271. S8 = 2,
  272. U16 = 3,
  273. S16 = 4,
  274. U32 = 5,
  275. S32 = 6,
  276. U64 = 7,
  277. S64 = 8,
  278. F32 = 9,
  279. F64 = 10,
  280. UTF = 11,
  281. EXT = 12,
  282. UC = 13,
  283. }
  284. public delegate T GetData<T>();
  285. public delegate void PutData<T>(T v);
  286. public abstract class IOStream
  287. {
  288. public static int DEFAULT_ARRAY_LIMIT = UInt16.MaxValue;
  289. public static int DEFAULT_BYTES_LIMIT = 100 * 1024 * 1024;
  290. public int ARRAY_LIMIT = DEFAULT_ARRAY_LIMIT;
  291. public int BYTES_LIMIT = DEFAULT_BYTES_LIMIT;
  292. }
  293. public abstract class IOutputStream : IOStream
  294. {
  295. protected static System.Text.UTF8Encoding utfenc = new System.Text.UTF8Encoding(false);
  296. public IOutputStream(IExternalizableFactory factory)
  297. {
  298. this.Factory = factory;
  299. }
  300. public IExternalizableFactory Factory { get; private set; }
  301. #region _Abstract_
  302. public abstract void PutBytes(byte[] bytes);
  303. public abstract void PutRawData(byte[] bytes, int offset, int count);
  304. public abstract void PutBool(bool value);
  305. public abstract void PutU8(byte value);
  306. public abstract void PutS8(sbyte value);
  307. public abstract void PutU16(ushort value);
  308. public abstract void PutS16(short value);
  309. public abstract void PutU32(uint value);
  310. public abstract void PutS32(int value);
  311. public abstract void PutU64(ulong value);
  312. public abstract void PutS64(long value);
  313. public abstract void PutF32(float value);
  314. public abstract void PutF64(double value);
  315. public abstract void PutUnicode(char value);
  316. public abstract void PutUTF(string str);
  317. public abstract long GetBuffPos();
  318. #endregion
  319. #region _基础类型_
  320. /// <summary>
  321. /// 写入可变长度32位
  322. /// </summary>
  323. public void PutVU32(uint value)
  324. {
  325. PutVU64(value);
  326. }
  327. /// <summary>
  328. /// 写入可变长度32位
  329. /// </summary>
  330. public void PutVS32(int value)
  331. {
  332. PutVS64(value);
  333. }
  334. /// <summary>
  335. /// 写入可变长度64位
  336. /// </summary>
  337. public void PutVS64(long value)
  338. {
  339. ulong m;
  340. if (value < 0)
  341. {
  342. m = (ulong)(((-value) << 1) | 1);
  343. }
  344. else
  345. {
  346. m = (ulong)(((value) << 1));
  347. }
  348. PutVU64(m);
  349. }
  350. /// <summary>
  351. /// 写入可变长度64位
  352. /// </summary>
  353. public void PutVU64(ulong value)
  354. {
  355. do
  356. {
  357. byte b = (byte)(value & 0x7F);
  358. value = (value >> 7);
  359. if (value != 0)
  360. {
  361. b = (byte)(b | 0x80);
  362. }
  363. this.PutU8(b);
  364. }
  365. while (value != 0);
  366. }
  367. public void PutEnum8(ValueType enum8)
  368. {
  369. byte u8 = (byte)enum8;
  370. PutU8(u8);
  371. }
  372. public void PutEnum32(ValueType enum32)
  373. {
  374. int s32 = (int)enum32;
  375. PutS32(s32);
  376. }
  377. public void PutEnum8<T>(T enum8)
  378. {
  379. byte u8 = Convert.ToByte(enum8);
  380. PutU8(u8);
  381. }
  382. public void PutEnum32<T>(T enum32)
  383. {
  384. int s32 = Convert.ToInt32(enum32);
  385. PutS32(s32);
  386. }
  387. #endregion
  388. #region _对象类型_
  389. public void PutExt(IExternalizable value)
  390. {
  391. if (value != null)
  392. {
  393. int typeID = Factory.GetTypeID(value.GetType());
  394. if (typeID != 0)
  395. {
  396. PutS32(typeID);
  397. value.WriteExternal(this);
  398. //if(typeID != 0x8601 && typeID != 0x8006 && typeID != 0x8100 && typeID != 0x4003 && typeID != 0x4004 && typeID != 0x000B0010 &
  399. // typeID != 0x4009 && typeID != 0x4104 && typeID != 0x4109 && typeID != 0x8407 && typeID != 0x9001012 && typeID != 0xF000024 &&
  400. // typeID != 0xF00001D && typeID != 0x400B && typeID != 0x4107 && typeID != 0x400C && typeID != 0xF000038 && typeID != 0x400C &&
  401. // typeID != 0x000B0005 && typeID != 0x900100A && typeID != 0x9001004 && typeID != 0x9001005 && typeID != 0x9001002 &&
  402. // typeID != 0x9001013 && typeID != 0x000B0001 && typeID != 0x8401 && typeID != 0x8116 && typeID != 0x9001010 &&
  403. // typeID != 0x9001016 && typeID != 0xF000008 && typeID != 0x9001003 && typeID != 0x9001006 && typeID != 0x4016 && typeID != 0x9005 &&
  404. // typeID != 0x9002 && typeID != 0xF00001E && typeID != 0xF000015 && typeID != 0x9006 && typeID != 0x9010)
  405. //{
  406. // Console.WriteLine(" -- putExt -- {0:x}", typeID);
  407. //}
  408. }
  409. else
  410. {
  411. PutS32(0);
  412. //log.Error("Can Not Resolve IExternalizable Message : " + value.GetType());
  413. throw new IOException("Can Not Resolve IExternalizable Message : " + value.GetType());
  414. }
  415. }
  416. else
  417. {
  418. PutS32(0);
  419. }
  420. }
  421. public int GetTypeID(IExternalizable value)
  422. {
  423. return Factory.GetTypeID(value.GetType());
  424. }
  425. public void PutObj2Xml(object obj)
  426. {
  427. if (obj != null)
  428. {
  429. XmlDocument x = XmlUtil.ObjectToXml(obj);
  430. string utf = XmlUtil.ToString(x);
  431. byte[] data = CUtils.UTF8.GetBytes(utf);
  432. PutBytes(data);
  433. }
  434. else
  435. {
  436. PutBytes(null);
  437. }
  438. }
  439. /// <summary>
  440. /// 自动检测类型
  441. /// </summary>
  442. /// <param name="value"></param>
  443. public void PutData(object value)
  444. {
  445. if (value is byte)
  446. {
  447. PutU8((byte)DataType.U8);
  448. PutU8((byte)value);
  449. }
  450. else if (value is sbyte)
  451. {
  452. PutU8((byte)DataType.S8);
  453. PutS8((sbyte)value);
  454. }
  455. else if (value is ushort)
  456. {
  457. PutU8((byte)DataType.U16);
  458. PutU16((ushort)value);
  459. }
  460. else if (value is short)
  461. {
  462. PutU8((byte)DataType.S16);
  463. PutS16((short)value);
  464. }
  465. else if (value is uint)
  466. {
  467. PutU8((byte)DataType.U32);
  468. PutU32((uint)value);
  469. }
  470. else if (value is int)
  471. {
  472. PutU8((byte)DataType.S32);
  473. PutS32((int)value);
  474. }
  475. else if (value is ulong)
  476. {
  477. PutU8((byte)DataType.U64);
  478. PutU64((ulong)value);
  479. }
  480. else if (value is long)
  481. {
  482. PutU8((byte)DataType.S64);
  483. PutS64((long)value);
  484. }
  485. else if (value is float)
  486. {
  487. PutU8((byte)DataType.F32);
  488. PutF32((float)value);
  489. }
  490. else if (value is double)
  491. {
  492. PutU8((byte)DataType.F64);
  493. PutF64((double)value);
  494. }
  495. else if (value is char)
  496. {
  497. PutU8((byte)DataType.UC);
  498. PutUnicode((char)value);
  499. }
  500. else if (value is string)
  501. {
  502. PutU8((byte)DataType.UTF);
  503. PutUTF((string)value);
  504. }
  505. else if (value is IExternalizable)
  506. {
  507. PutU8((byte)DataType.EXT);
  508. PutExt((IExternalizable)value);
  509. }
  510. else
  511. {
  512. PutU8((byte)DataType.NA);
  513. }
  514. }
  515. #endregion
  516. #region _集合类型_
  517. public void PutUTFArray(string[] array)
  518. {
  519. if (array == null)
  520. {
  521. PutS32(-1);
  522. }
  523. else
  524. {
  525. int len = array.Length;
  526. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  527. PutS32(len);
  528. for (int i = 0; i < len; i++)
  529. {
  530. this.PutUTF(array[i]);
  531. }
  532. }
  533. }
  534. public void PutUTFList(IList<string> list)
  535. {
  536. if (list == null)
  537. {
  538. PutS32(-1);
  539. }
  540. else
  541. {
  542. int len = list.Count;
  543. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  544. PutS32(len);
  545. for (int i = 0; i < len; i++)
  546. {
  547. this.PutUTF(list[i]);
  548. }
  549. }
  550. }
  551. public void PutArray<T>(T[] array, PutData<T> action)
  552. {
  553. if (array == null)
  554. {
  555. PutS32(-1);
  556. }
  557. else
  558. {
  559. int len = array.Length;
  560. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  561. PutS32(len);
  562. for (int i = 0; i < len; i++)
  563. {
  564. action.Invoke(array[i]);
  565. }
  566. }
  567. }
  568. public void PutList<T>(IList<T> list, PutData<T> action)
  569. {
  570. if (list == null)
  571. {
  572. PutS32(-1);
  573. }
  574. else
  575. {
  576. int len = list.Count;
  577. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  578. PutS32(len);
  579. for (int i = 0; i < len; i++)
  580. {
  581. action.Invoke(list[i]);
  582. }
  583. }
  584. }
  585. public void PutMap<K, V>(IDictionary<K, V> map, PutData<K> k_action, PutData<V> v_action)
  586. {
  587. if (map == null)
  588. {
  589. PutS32(-1);
  590. }
  591. else
  592. {
  593. int len = map.Count;
  594. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  595. this.PutS32(len);
  596. foreach (var kv in map)
  597. {
  598. k_action.Invoke(kv.Key);
  599. v_action.Invoke(kv.Value);
  600. }
  601. }
  602. }
  603. /// <summary>
  604. /// 不包括消息头,列表中元素必须保证类型一致
  605. /// </summary>
  606. public void PutStructArray<T>(T[] array) where T : IExternalizable
  607. {
  608. if (array == null)
  609. {
  610. PutS32(-1);
  611. }
  612. else
  613. {
  614. int len = array.Length;
  615. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  616. this.PutS32(array.Length);
  617. for (int i = 0; i < array.Length; i++)
  618. {
  619. array[i].WriteExternal(this);
  620. }
  621. }
  622. }
  623. /// <summary>
  624. /// 不包括消息头,列表中元素必须保证类型一致
  625. /// </summary>
  626. public void PutStructList<T>(IList<T> array) where T : IExternalizable
  627. {
  628. if (array == null)
  629. {
  630. PutS32(-1);
  631. }
  632. else
  633. {
  634. int len = array.Count;
  635. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  636. this.PutS32(array.Count);
  637. for (int i = 0; i < array.Count; i++)
  638. {
  639. array[i].WriteExternal(this);
  640. }
  641. }
  642. }
  643. /// <summary>
  644. /// 不包括消息头,列表中元素必须保证类型一致
  645. /// </summary>
  646. public void PutExtArray<T>(T[] array) where T : IExternalizable
  647. {
  648. if (array == null)
  649. {
  650. PutS32(-1);
  651. }
  652. else
  653. {
  654. int len = array.Length;
  655. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  656. this.PutS32(array.Length);
  657. for (int i = 0; i < array.Length; i++)
  658. {
  659. array[i].WriteExternal(this);
  660. }
  661. }
  662. }
  663. /// <summary>
  664. /// 不包括消息头,列表中元素必须保证类型一致
  665. /// </summary>
  666. public void PutExtList<T>(IList<T> array) where T : IExternalizable
  667. {
  668. if (array == null)
  669. {
  670. PutS32(-1);
  671. }
  672. else
  673. {
  674. int len = array.Count;
  675. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  676. this.PutS32(array.Count);
  677. for (int i = 0; i < array.Count; i++)
  678. {
  679. array[i].WriteExternal(this);
  680. }
  681. }
  682. }
  683. #endregion
  684. }
  685. public abstract class IInputStream : IOStream
  686. {
  687. protected static System.Text.UTF8Encoding utfenc = new System.Text.UTF8Encoding(false);
  688. public IInputStream(IExternalizableFactory factory)
  689. {
  690. this.Factory = factory;
  691. }
  692. public IExternalizableFactory Factory { get; private set; }
  693. #region _Abstract_
  694. public abstract byte[] GetBytes();
  695. public abstract void GetRawData(byte[] buff, int offset, int count);
  696. public abstract bool GetBool();
  697. public abstract byte GetU8();
  698. public abstract sbyte GetS8();
  699. public abstract ushort GetU16();
  700. public abstract short GetS16();
  701. public abstract uint GetU32();
  702. public abstract int GetS32();
  703. public abstract ulong GetU64();
  704. public abstract long GetS64();
  705. public abstract float GetF32();
  706. public abstract double GetF64();
  707. public abstract char GetUnicode();
  708. public abstract string GetUTF();
  709. #endregion
  710. #region _基础类型_
  711. /// <summary>
  712. /// 写入可变长度32位
  713. /// </summary>
  714. public UInt32 GetVU32()
  715. {
  716. return (UInt32)GetVU64();
  717. }
  718. /// <summary>
  719. /// 写入可变长度32位
  720. /// </summary>
  721. public Int32 GetVS32()
  722. {
  723. return (Int32)GetVS64();
  724. }
  725. /// <summary>
  726. /// 写入可变长度64位
  727. /// </summary>
  728. public Int64 GetVS64()
  729. {
  730. ulong m = GetVU64();
  731. long v = (long)(m >> 1);
  732. if (m % 2 == 1)
  733. {
  734. return -v;
  735. }
  736. else
  737. {
  738. return v;
  739. }
  740. }
  741. /// <summary>
  742. /// 写入可变长度64位
  743. /// </summary>
  744. public UInt64 GetVU64()
  745. {
  746. UInt64 value = 0;
  747. UInt64 b = 0;
  748. for (int i = 0; i <= 70; i += 7)
  749. {
  750. b = GetU8();
  751. value |= ((b & 0x7F) << i);
  752. if ((b & 0x80) == 0)
  753. break;
  754. }
  755. return value;
  756. }
  757. public T GetEnum8<T>()
  758. {
  759. byte u8 = GetU8();
  760. if (Enum.IsDefined(typeof(T), u8))
  761. {
  762. return (T)Enum.ToObject(typeof(T), u8);
  763. }
  764. return default(T);
  765. }
  766. public T GetEnum32<T>()
  767. {
  768. int s32 = GetS32();
  769. if (Enum.IsDefined(typeof(T), s32))
  770. {
  771. return (T)Enum.ToObject(typeof(T), s32);
  772. }
  773. return default(T);
  774. }
  775. #endregion
  776. #region _对象类型_
  777. public T GetExt<T>() where T : IExternalizable, new()
  778. {
  779. int typeID = GetS32();
  780. if (typeID != 0)
  781. {
  782. Type type = Factory.GetType(typeID);
  783. if (type != null)
  784. {
  785. T any = (T)Activator.CreateInstance(type);
  786. any.ReadExternal(this);
  787. return any;
  788. }
  789. else
  790. {
  791. throw new IOException("Can Not Resolve IExternalizable Message : 0x" + typeID.ToString("X"));
  792. }
  793. }
  794. return default(T);
  795. }
  796. public T GetExt<T>(T ret) where T : IExternalizable
  797. {
  798. int typeID = GetS32();
  799. if (typeID != 0)
  800. {
  801. Type type = Factory.GetType(typeID);
  802. if (type == null)
  803. {
  804. throw new IOException("Can Not Resolve IExternalizable Message : 0x" + typeID.ToString("X"));
  805. }
  806. if (!type.IsInstanceOfType(ret))
  807. {
  808. throw new IOException("type can't match:" + type + "->" + ret.GetType());
  809. }
  810. else
  811. {
  812. ret.ReadExternal(this);
  813. return ret;
  814. }
  815. }
  816. return default(T);
  817. }
  818. public IExternalizable GetExtAny()
  819. {
  820. int typeID = GetS32();
  821. if (typeID != 0)
  822. {
  823. Type type = Factory.GetType(typeID);
  824. if (type != null)
  825. {
  826. IExternalizable any = (IExternalizable)Activator.CreateInstance(type);
  827. any.ReadExternal(this);
  828. return any;
  829. }
  830. else
  831. {
  832. throw new IOException("Can Not Resolve IExternalizable Message : 0x" + typeID.ToString("X"));
  833. }
  834. }
  835. return null;
  836. }
  837. public T GetXml2Obj<T>(Action<Exception> error = null)
  838. {
  839. byte[] data = GetBytes();
  840. if (data != null && data.Length > 0)
  841. {
  842. string xml = CUtils.UTF8.GetString(data);
  843. XmlDocument x = XmlUtil.FromString(xml);
  844. T ret = (T)XmlUtil.XmlToObject(x, error);
  845. return ret;
  846. }
  847. return default(T);
  848. }
  849. /// <summary>
  850. /// 自动检测类型
  851. /// </summary>
  852. /// <param name="type"></param>
  853. public object GetData(out DataType type)
  854. {
  855. byte dt = GetU8();
  856. if (Enum.IsDefined(typeof(DataType), dt))
  857. {
  858. type = (DataType)dt;
  859. switch (type)
  860. {
  861. case DataType.U8: return GetU8();
  862. case DataType.S8: return GetS8();
  863. case DataType.U16: return GetU16();
  864. case DataType.S16: return GetS16();
  865. case DataType.U32: return GetU32();
  866. case DataType.S32: return GetS32();
  867. case DataType.U64: return GetU64();
  868. case DataType.S64: return GetS64();
  869. case DataType.F32: return GetF32();
  870. case DataType.F64: return GetF64();
  871. case DataType.UC: return GetUnicode();
  872. case DataType.UTF: return GetUTF();
  873. case DataType.EXT: return GetExtAny();
  874. }
  875. }
  876. type = DataType.NA;
  877. return null;
  878. }
  879. public object GetData()
  880. {
  881. byte dt = GetU8();
  882. if (Enum.IsDefined(typeof(DataType), dt))
  883. {
  884. DataType type = (DataType)dt;
  885. switch (type)
  886. {
  887. case DataType.U8: return GetU8();
  888. case DataType.S8: return GetS8();
  889. case DataType.U16: return GetU16();
  890. case DataType.S16: return GetS16();
  891. case DataType.U32: return GetU32();
  892. case DataType.S32: return GetS32();
  893. case DataType.U64: return GetU64();
  894. case DataType.S64: return GetS64();
  895. case DataType.F32: return GetF32();
  896. case DataType.F64: return GetF64();
  897. case DataType.UC: return GetUnicode();
  898. case DataType.UTF: return GetUTF();
  899. case DataType.EXT: return GetExtAny();
  900. }
  901. }
  902. return null;
  903. }
  904. #endregion
  905. #region _集合类型_
  906. public string[] GetUTFArray()
  907. {
  908. int len = GetS32();
  909. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  910. if (len < 0) return null;
  911. string[] ret = new string[len];
  912. for (int i = 0; i < len; i++)
  913. {
  914. ret[i] = GetUTF();
  915. }
  916. return ret;
  917. }
  918. public T[] GetArray<T>(GetData<T> action)
  919. {
  920. int len = GetS32();
  921. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  922. if (len < 0) return null;
  923. T[] ret = new T[len];
  924. for (int i = 0; i < len; i++)
  925. {
  926. T d = action.Invoke();
  927. ret[i] = d;
  928. }
  929. return ret;
  930. }
  931. /// <summary>
  932. /// 不包括消息头,列表中元素必须保证类型一致
  933. /// </summary>
  934. public T[] GetStructArray<T>() where T : IExternalizable, new()
  935. {
  936. int len = GetS32();
  937. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  938. if (len < 0) return null;
  939. T[] array = new T[len];
  940. for (int i = 0; i < len; i++)
  941. {
  942. array[i].ReadExternal(this);
  943. }
  944. return array;
  945. }
  946. /// <summary>
  947. /// 不包括消息头,列表中元素必须保证类型一致
  948. /// </summary>
  949. public T[] GetExtArray<T>() where T : IExternalizable, new()
  950. {
  951. int len = GetS32();
  952. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  953. if (len < 0) return null;
  954. T[] array = new T[len];
  955. for (int i = 0; i < len; i++)
  956. {
  957. array[i] = new T();
  958. array[i].ReadExternal(this);
  959. }
  960. return array;
  961. }
  962. public List<string> GetUTFList()
  963. {
  964. int len = GetS32();
  965. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  966. if (len < 0) return null;
  967. List<string> ret = new List<string>(len);
  968. for (int i = 0; i < len; i++)
  969. {
  970. string d = GetUTF();
  971. ret.Add(d);
  972. }
  973. return ret;
  974. }
  975. public List<T> GetList<T>(GetData<T> action)
  976. {
  977. int len = GetS32();
  978. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  979. if (len < 0) return null;
  980. List<T> ret = new List<T>(len);
  981. for (int i = 0; i < len; i++)
  982. {
  983. T d = action.Invoke();
  984. ret.Add(d);
  985. }
  986. return ret;
  987. }
  988. public List<T> GetListAny<T>() where T : IExternalizable
  989. {
  990. int len = GetS32();
  991. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  992. if (len < 0) return null;
  993. List<T> ret = new List<T>(len);
  994. for (int i = 0; i < len; i++)
  995. {
  996. IExternalizable ext = GetExtAny();
  997. if (ext != null)
  998. {
  999. ret.Add((T)ext);
  1000. }
  1001. }
  1002. return ret;
  1003. }
  1004. /// <summary>
  1005. /// 不包括消息头,列表中元素必须保证类型一致
  1006. /// </summary>
  1007. public List<T> GetStructList<T>() where T : IExternalizable, new()
  1008. {
  1009. int len = GetS32();
  1010. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  1011. if (len < 0) return null;
  1012. List<T> list = new List<T>();
  1013. for (int i = 0; i < len; i++)
  1014. {
  1015. T item = new T();
  1016. item.ReadExternal(this);
  1017. list.Add(item);
  1018. }
  1019. return list;
  1020. }
  1021. /// <summary>
  1022. /// 不包括消息头,列表中元素必须保证类型一致
  1023. /// </summary>
  1024. public List<T> GetExtList<T>() where T : IExternalizable, new()
  1025. {
  1026. int len = GetS32();
  1027. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  1028. if (len < 0) return null;
  1029. List<T> list = new List<T>();
  1030. for (int i = 0; i < len; i++)
  1031. {
  1032. T item = new T();
  1033. item.ReadExternal(this);
  1034. list.Add(item);
  1035. }
  1036. return list;
  1037. }
  1038. //--------------------------------------------------------------------------------------------
  1039. public L GetGenericUTFList<L>() where L : class, IList<string>, new()
  1040. {
  1041. int len = GetS32();
  1042. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  1043. if (len < 0) return null;
  1044. var ret = new L();
  1045. for (int i = 0; i < len; i++)
  1046. {
  1047. string d = GetUTF();
  1048. ret.Add(d);
  1049. }
  1050. return ret;
  1051. }
  1052. public L GetGenericList<L, T>(GetData<T> action) where L : class, IList<T>, new()
  1053. {
  1054. int len = GetS32();
  1055. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  1056. if (len < 0) return null;
  1057. L ret = new L();
  1058. for (int i = 0; i < len; i++)
  1059. {
  1060. T d = action.Invoke();
  1061. ret.Add(d);
  1062. }
  1063. return ret;
  1064. }
  1065. public L GetGenericListAny<L, T>()
  1066. where T : IExternalizable
  1067. where L : class, IList<T>, new()
  1068. {
  1069. int len = GetS32();
  1070. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  1071. if (len < 0) return null;
  1072. var ret = new L();
  1073. for (int i = 0; i < len; i++)
  1074. {
  1075. IExternalizable ext = GetExtAny();
  1076. if (ext != null)
  1077. {
  1078. ret.Add((T)ext);
  1079. }
  1080. }
  1081. return ret;
  1082. }
  1083. /// <summary>
  1084. /// 不包括消息头,列表中元素必须保证类型一致
  1085. /// </summary>
  1086. public L GetGenericStructList<L, T>()
  1087. where T : struct, IExternalizable
  1088. where L : class, IList<T>, new()
  1089. {
  1090. int len = GetS32();
  1091. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  1092. if (len < 0) return null;
  1093. var list = new L();
  1094. for (int i = 0; i < len; i++)
  1095. {
  1096. T item = new T();
  1097. item.ReadExternal(this);
  1098. list.Add(item);
  1099. }
  1100. return list;
  1101. }
  1102. /// <summary>
  1103. /// 不包括消息头,列表中元素必须保证类型一致
  1104. /// </summary>
  1105. public L GetGenericExtList<L, T>()
  1106. where T : IExternalizable, new()
  1107. where L : class, IList<T>, new()
  1108. {
  1109. int len = GetS32();
  1110. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  1111. if (len < 0) return null;
  1112. var list = new L();
  1113. for (int i = 0; i < len; i++)
  1114. {
  1115. T item = new T();
  1116. item.ReadExternal(this);
  1117. list.Add(item);
  1118. }
  1119. return list;
  1120. }
  1121. //--------------------------------------------------------------------------------------------
  1122. public HashMap<K, V> GetMap<K, V>(GetData<K> k_action, GetData<V> v_action)
  1123. {
  1124. int len = GetS32();
  1125. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  1126. if (len < 0) return null;
  1127. HashMap<K, V> ret = new HashMap<K, V>(len);
  1128. for (int i = 0; i < len; i++)
  1129. {
  1130. K k = k_action.Invoke();
  1131. V v = v_action.Invoke();
  1132. ret[k] = v;
  1133. }
  1134. return ret;
  1135. }
  1136. public M GetGenericMap<M, K, V>(GetData<K> k_action, GetData<V> v_action) where M : class, IDictionary<K, V>, new()
  1137. {
  1138. int len = GetS32();
  1139. if (len > ARRAY_LIMIT) { throw new IOException("Collection overflow : " + len + " > " + ARRAY_LIMIT); }
  1140. if (len < 0) return null;
  1141. M ret = new M();
  1142. for (int i = 0; i < len; i++)
  1143. {
  1144. K k = k_action.Invoke();
  1145. V v = v_action.Invoke();
  1146. ret[k] = v;
  1147. }
  1148. return ret;
  1149. }
  1150. //--------------------------------------------------------------------------------------------
  1151. #endregion
  1152. }
  1153. }