public class ByteArrayBuilder : IDisposable
{
#region Constants
/// <summary>
/// True in a byte form of the Line
/// </summary>
const byte streamTrue = (byte)1;
/// <summary>
/// False in the byte form of a line
/// </summary>
const byte streamFalse = (byte)0;
#endregion #region Fields
#region Internal
/// <summary>
/// Holds the actual bytes.
/// </summary>
MemoryStream store = new MemoryStream(); /// <summary>
/// Is Little Endian
/// True - little endian
/// False - big endian
/// </summary>
bool isLittleEndian;
#endregion #region Property bases
#endregion
#endregion #region Properties
/// <summary>
/// Bytes in the store.
/// </summary>
public int Length
{
get { return (int)store.Length; }
}
#endregion #region Regular Expressions
#endregion #region Enums
#endregion #region Constructors
/// <summary>
/// Create a new, empty builder ready to be filled.
/// </summary>
public ByteArrayBuilder(bool isLittleEndian = true)
{
this.isLittleEndian = isLittleEndian;
}
/// <summary>
/// Create a new builder from a set of data
/// </summary>
/// <param name="data">Data to preset the builder from</param>
public ByteArrayBuilder(byte[] data, bool isLittleEndian = true)
{
store.Close();
store.Dispose();
store = new MemoryStream(data);
this.isLittleEndian = isLittleEndian;
}
/// <summary>
/// Create a new builder from the Base64 string representation of an
/// existing instance.
/// The Base64 representation can be retrieved using the ToString override
/// </summary>
/// <param name="base64">Base64 string representation of an
/// existing instance.</param>
public ByteArrayBuilder(string base64, bool isLittleEndian = true)
{
store.Close();
store.Dispose();
store = new MemoryStream(Convert.FromBase64String(base64));
this.isLittleEndian = isLittleEndian;
}
#endregion #region Events
#region Event Constructors
#endregion #region Event Handlers
#endregion
#endregion #region Public Methods
#region Append overloads
/// <summary>
/// Adds a bool to an array
/// </summary>
/// <param name="b">Value to append to existing builder data</param>
public void Append(bool b)
{
store.WriteByte(b ? streamTrue : streamFalse);
}
/// <summary>
/// Adds a byte to an array
/// </summary>
/// <param name="b">Value to append to existing builder data</param>
public void Append(byte b)
{
store.WriteByte(b);
}
/// <summary>
/// Adds an array of bytes to an array
/// </summary>
/// <param name="b">Value to append to existing builder data</param>
/// <param name="addLength">
/// If true, the length is added before the value.
/// This allows extraction of individual elements back to the original input form.
/// </param>
public void Append(byte[] b, bool addLength = false)
{
if (b != null)
{
if (addLength) Append(b.Length);
AddBytes(b);
}
}
/// <summary>
/// Adds a char to an array
/// </summary>
/// <param name="c">Value to append to existing builder data</param>
public void Append(char c)
{
store.WriteByte((byte)c);
}
/// <summary>
/// Adds an array of characters to an array
/// </summary>
/// <param name="c">Value to append to existing builder data</param>
/// <param name="addLength">
/// If true, the length is added before the value.
/// This allows extraction of individual elements back to the original input form.
/// </param>
public void Append(char[] c, bool addLength = false)
{
if (c != null)
{
if (addLength) Append(c.Length);
Append(System.Text.Encoding.Unicode.GetBytes(c));
}
}
/// <summary>
/// Adds a DateTime to an array
/// </summary>
/// <param name="dt">Value to append to existing builder data</param>
public void Append(DateTime dt)
{
Append(dt.Ticks);
}
/// <summary>
/// Adds a decimal value to an array
/// </summary>
/// <param name="d">Value to append to existing builder data</param>
public void Append(decimal d)
{
// GetBits always returns four ints.
// We store them in a specific order so that they can be recovered later.
int[] bits = decimal.GetBits(d); if (isLittleEndian)
{
Append(bits[0]);
Append(bits[1]);
Append(bits[2]);
Append(bits[3]);
}
else
{
Append(bits[3]);
Append(bits[2]);
Append(bits[1]);
Append(bits[0]);
}
}
/// <summary>
/// Adds a double to an array
/// </summary>
/// <param name="d">Value to append to existing builder data</param>
public void Append(double d)
{
byte[] data = BitConverter.GetBytes(d);
if (!isLittleEndian)
{
Array.Reverse(data);
} AddBytes(data);
}
/// <summary>
/// Adds a float to an array
/// </summary>
/// <param name="f">Value to append to existing builder data</param>
public void Append(float f)
{
byte[] data = BitConverter.GetBytes(f);
if (!isLittleEndian)
{
Array.Reverse(data);
} AddBytes(data);
}
/// <summary>
/// Adds a Guid to an array
/// </summary>
/// <param name="g">Value to append to existing builder data</param>
public void Append(Guid g)
{
Append(g.ToByteArray());
}
/// <summary>
/// Adds an integer to an array
/// </summary>
/// <param name="i">Value to append to existing builder data</param>
public void Append(int i)
{
byte[] data = BitConverter.GetBytes(i);
if (!isLittleEndian)
{
Array.Reverse(data);
} AddBytes(data);
}
/// <summary>
/// Adds a long integer to an array
/// </summary>
/// <param name="l">Value to append to existing builder data</param>
public void Append(long l)
{
byte[] data = BitConverter.GetBytes(l);
if (!isLittleEndian)
{
Array.Reverse(data);
} AddBytes(data);
}
/// <summary>
/// Adds a short integer to an array
/// </summary>
/// <param name="i">Value to append to existing builder data</param>
public void Append(short s)
{
byte[] data = BitConverter.GetBytes(s);
if (!isLittleEndian)
{
Array.Reverse(data);
} AddBytes(data);
}
/// <summary>
/// Adds a string to an array
/// </summary>
/// <param name="s">Value to append to existing builder data</param>
/// <param name="addLength">
/// If true, the length is added before the value.
/// This allows extraction of individual elements back to the original input form.
/// </param>
public void Append(string s, bool addLength = false)
{
if (!string.IsNullOrEmpty(s))
{
byte[] data = System.Text.Encoding.Unicode.GetBytes(s);
if (addLength) Append(data.Length);
AddBytes(data);
}
}
/// <summary>
/// Adds an unsigned integer to an array
/// </summary>
/// <param name="ui">Value to append to existing builder data</param>
public void Append(uint ui)
{
byte[] data = BitConverter.GetBytes(ui);
if (!isLittleEndian)
{
Array.Reverse(data);
} AddBytes(data);
}
/// <summary>
/// Adds a unsigned long integer to an array
/// </summary>
/// <param name="ul">Value to append to existing builder data</param>
public void Append(ulong ul)
{
byte[] data = BitConverter.GetBytes(ul);
if (!isLittleEndian)
{
Array.Reverse(data);
} AddBytes(data);
}
/// <summary>
/// Adds a unsigned short integer to an array
/// </summary>
/// <param name="us">Value to append to existing builder data</param>
public void Append(ushort us)
{
byte[] data = BitConverter.GetBytes(us);
if (!isLittleEndian)
{
Array.Reverse(data);
} AddBytes(data);
}
#endregion #region Extraction
/// <summary>
/// Gets a bool from an array
/// </summary>
/// <returns></returns>
public bool GetBool()
{
return store.ReadByte() == streamTrue;
}
/// <summary>
/// Gets a byte from an array
/// </summary>
/// <returns></returns>
public byte GetByte()
{
return (byte)store.ReadByte();
}
/// <summary>
/// Gets an array of bytes from an array
/// </summary>
/// <returns></returns>
public byte[] GetByteArray()
{
int length = GetInt();
return GetBytes(length);
}
/// <summary>
/// Gets a char from an array
/// </summary>
/// <returns></returns>
public char GetChar()
{
return (char)store.ReadByte();
}
/// <summary>
/// Gets an array of characters from an array
/// </summary>
/// <returns></returns>
public char[] GetCharArray()
{
int length = GetInt();
return System.Text.Encoding.Unicode.GetChars(GetBytes(length));
}
/// <summary>
/// Gets a DateTime value from an array
/// </summary>
/// <returns></returns>
public DateTime GetDateTime()
{
return new DateTime(GetLong());
}
/// <summary>
/// Gets a decimal value from an array
/// </summary>
/// <returns></returns>
public decimal GetDecimal()
{
// GetBits always returns four ints.
// We store them in a specific order so that they can be recovered later.
int[] bits = new int[] { GetInt(), GetInt(), GetInt(), GetInt() };
if(!isLittleEndian)
{
Array.Reverse(bits);
}
return new decimal(bits);
}
/// <summary>
/// Gets a double from an array
/// </summary>
/// <returns></returns>
public double GetDouble()
{
byte[] data = GetBytes(8);
if (!isLittleEndian)
{
Array.Reverse(data);
}
return BitConverter.ToDouble(data, 0);
}
/// <summary>
/// Gets a float from an array
/// </summary>
/// <returns></returns>
public float GetFloat()
{
byte[] data = GetBytes(4);
if (!isLittleEndian)
{
Array.Reverse(data);
}
return BitConverter.ToSingle(data, 0);
}
/// <summary>
/// Gets a Guid from an array
/// </summary>
/// <returns></returns>
public Guid GetGuid()
{
return new Guid(GetByteArray());
}
/// <summary>
/// Gets an integer from an array
/// </summary>
/// <returns></returns>
public int GetInt()
{
byte[] data = GetBytes(4);
if (!isLittleEndian)
{
Array.Reverse(data);
}
return BitConverter.ToInt32(data, 0);
}
/// <summary>
/// Gets a long integer from an array
/// </summary>
/// <returns></returns>
public long GetLong()
{
byte[] data = GetBytes(8);
if (!isLittleEndian)
{
Array.Reverse(data);
}
return BitConverter.ToInt64(data, 0);
}
/// <summary>
/// Gets a short integer from an array
/// </summary>
/// <returns></returns>
public short GetShort()
{
byte[] data = GetBytes(2);
if (!isLittleEndian)
{
Array.Reverse(data);
}
return BitConverter.ToInt16(data, 0);
}
/// <summary>
/// Gets a string from an array
/// </summary>
/// <returns></returns>
public string GetString()
{
int length = GetInt();
return System.Text.Encoding.Unicode.GetString(GetBytes(length));
}
/// <summary>
/// Gets an unsigned integer from an array
/// </summary>
/// <returns></returns>
public uint GetUint()
{
byte[] data = GetBytes(4);
if (!isLittleEndian)
{
Array.Reverse(data);
}
return BitConverter.ToUInt32(data, 0);
}
/// <summary>
/// Gets a unsigned long integer from an array
/// </summary>
/// <returns></returns>
public ulong GetUlong()
{
byte[] data = GetBytes(8);
if (!isLittleEndian)
{
Array.Reverse(data);
}
return BitConverter.ToUInt64(data, 0);
}
/// <summary>
/// Gets a unsigned short integer from an array
/// </summary>
/// <returns></returns>
public ushort GetUshort()
{
byte[] data = GetBytes(2);
if (!isLittleEndian)
{
Array.Reverse(data);
}
return BitConverter.ToUInt16(data, 0);
}
#endregion #region Interaction
/// <summary>
/// Clear all content from the builder
/// </summary>
public void Clear()
{
store.Close();
store.Dispose();
store = new MemoryStream();
}
/// <summary>
/// Rewind the builder ready to read data
/// </summary>
public void Rewind()
{
store.Seek(0, SeekOrigin.Begin);
}
/// <summary>
/// Set an absolute position in the builder.
/// **WARNING**
/// If you add any variable size objects to the builder, the results of
/// reading after a Seek to a non-zero value are unpredictable.
/// A builder does not store just objects - for some it stores additional
/// information as well.
/// </summary>
/// <param name="position"></param>
public void Seek(int position)
{
store.Seek((long)position, SeekOrigin.Begin);
}
/// <summary>
/// Returns the builder as an array of bytes
/// </summary>
/// <returns></returns>
public byte[] ToArray()
{
byte[] data = new byte[Length];
Array.Copy(store.GetBuffer(), data, Length);
return data;
}
#endregion
#endregion #region Overrides
/// <summary>
/// Returns a text based (Base64) string version of the current content
/// </summary>
/// <returns></returns>
public override string ToString()
{
return Convert.ToBase64String(ToArray());
}
#endregion #region Private Methods
/// <summary>
/// Add a string of raw bytes to the store
/// </summary>
/// <param name="b"></param>
private void AddBytes(byte[] b)
{
if (b != null)
{
store.Write(b, 0, b.Length);
}
}
/// <summary>
/// Reads a specific number of bytes from the store
/// </summary>
/// <param name="length"></param>
/// <returns></returns>
private byte[] GetBytes(int length)
{
byte[] data = new byte[length];
if (length > 0)
{
int read = store.Read(data, 0, length);
if (read != length)
{
throw new ApplicationException("Buffer did not contain " + length + " bytes");
}
}
return data;
}
#endregion #region IDisposable Implememntation
/// <summary>
/// Dispose of this builder and it's resources
/// </summary>
public void Dispose()
{
store.Close();
store.Dispose();
}
#endregion
}

  

ByteArrayBuilder的更多相关文章

  1. Android WebKit 内核

    一.WebKit简介 WebKit是一个开源的浏览器网页排版引擎,包含WebCore排版引擎和JSCore引擎.WebCore和JSCore引擎来自于KDE项目的KHTML和KJS开源项目.Andro ...

  2. ObjectMapper对象的使用 Object2JSON

    // // Source code recreated from a .class file by IntelliJ IDEA // (powered by Fernflower decompiler ...

  3. webKit 内核浏览器 源码分析

    如需转载,请注明出处! WebSite: http://www.jjos.org/ 作者: 姜江 linuxemacs@gmail.com QQ: 457283 这是一篇自己写于一年前的工作文档,分享 ...

随机推荐

  1. Virtualbox中不能为虚拟机打开一个新任务的原因及解决方法

    VirtualBox新建虚拟机时报错,不能为虚拟机打开一个新任务的原因 解决办法如下 1.保证bios里的virtualization technology的选项开启,不同电脑BIOS设置可能会不一样 ...

  2. Openfire基础

    网上很多openfire相关资料,这里做下学习汇总 openfire官网:http://www.igniterealtime.org/ 可以下载openfire.spark.smack安装包及源码,安 ...

  3. VS.Net 2015 Update3 学习(1) 支持Webpack

    让vs.net 编译的时候自动执行webpack 首先 管理员模式下打开 “Developer Command Prompt for VS2015", 是管理员模式啊! 然后进入 cd c: ...

  4. iOS - AppStores App 上架

    前言 1.准备 开发者账号 完工的项目 2.上架步骤 1) 创建 App ID 2) 创建证书请求文件(CSR文件) 3) 创建发布证书(CER) 4) 创建 Provisioning Profile ...

  5. Linux下LVM

    http://www.cnblogs.com/xiaoluo501395377/archive/2013/05/22/3093405.html

  6. PHP 中 define() 和 const 定义常量时的区别

    自 PHP 5.3.0 起,有两种方式定义常量,使用 const 关键字或者 define() 函数:   1 2 const FOO = 'BAR'; define('FOO', 'BAR'); 这 ...

  7. spring boot给http添加正向代理

    http://blog.csdn.net/jaune161/article/details/44198599http://46aae4d1e2371e4aa769798941cef698.devpro ...

  8. android native crash 分析

    工具: addr2line arm-linux-androideabi-addr2line -aCfe libart.so 0x63006d 当libart.so包含符号表的情况下,可以查询到他的地址 ...

  9. Sprint2(12.6)

    Sprint1第二阶段 1.类名:软件工程-第二阶段 方案一:此方案操作界面只有前台.厨房 (1)前台:用户到前台点餐,服务员操作界面,勾选客人所在桌号(不可重复勾选),并输入所选菜品,可增.删.改所 ...

  10. 数字图像处理作业使用OpenCV - 块提取

    今天要记录的是树图第二次作业的第二题,Image Patch Extraction.这个概念真的不难懂,但是如果要我实际写的话,还真的不知道要怎么去遍历图像矩阵来提取块.在此要多谢邓大神的热心帮助,告 ...