using Newtonsoft.Json.Linq;
using System;
using System.Collections.Generic;
using System.Drawing;
using System.IO;
using System.Runtime.Serialization.Formatters.Binary;
using System.Security.Cryptography;
using System.Text;
namespace TFT_MelsecMcNet
{
///
/// 一个软件基础类,提供常用的一些静态方法 ->
/// A software-based class that provides some common static methods
///
public class SoftBasic
{
#region MD5 Calculate
///
/// 获取文件的md5码 -> Get the MD5 code of the file
///
/// 文件的路径,既可以是完整的路径,也可以是相对的路径 -> The path to the file
/// Md5字符串
public static string CalculateFileMD5(string filePath)
{
string str_md5 = string.Empty;
using (FileStream fs = new FileStream(filePath, FileMode.Open, FileAccess.Read))
{
str_md5 = CalculateStreamMD5(fs);
}
return str_md5;
}
///
/// 获取数据流的md5码 -> Get the MD5 code for the data stream
///
/// 数据流,可以是内存流,也可以是文件流
/// Md5字符串
public static string CalculateStreamMD5(Stream stream)
{
byte[] bytes_md5 = null;
using (MD5 md5 = new MD5CryptoServiceProvider())
{
bytes_md5 = md5.ComputeHash(stream);
}
return BitConverter.ToString(bytes_md5).Replace("-", "");
}
///
/// 获取文本字符串信息的Md5码,编码为UTF8
///
/// 文本数据信息
/// Md5字符串
public static string CalculateStreamMD5(string data)
{
return CalculateStreamMD5(data, Encoding.UTF8);
}
///
/// 获取文本字符串信息的Md5码,使用指定的编码
///
/// 文本数据信息
/// 编码信息
/// Md5字符串
public static string CalculateStreamMD5(string data, Encoding encode)
{
string str_md5 = string.Empty;
using (MD5 md5 = new MD5CryptoServiceProvider())
{
byte[] bytes_md5 = md5.ComputeHash(encode.GetBytes(data));
str_md5 = BitConverter.ToString(bytes_md5).Replace("-", "");
}
return str_md5;
}
#if !NETSTANDARD2_0
///
/// 获取内存图片的md5码 -> Get the MD5 code of the memory picture
///
/// 内存图片
/// Md5字符串
public static string CalculateStreamMD5(Bitmap bitmap)
{
MemoryStream ms = new MemoryStream();
bitmap.Save(ms, bitmap.RawFormat);
MD5 md5 = new MD5CryptoServiceProvider();
byte[] bytes_md5 = md5.ComputeHash(ms);
ms.Dispose();
return BitConverter.ToString(bytes_md5).Replace("-", "");
}
#endif
#endregion
#region DataSize Format
///
/// 从一个字节大小返回带单位的描述
///
/// 实际的大小值
/// 最终的字符串值
public static string GetSizeDescription(long size)
{
if (size < 1000)
{
return size + " B";
}
else if (size < 1000 * 1000)
{
float data = (float)size / 1024;
return data.ToString("F2") + " Kb";
}
else if (size < 1000 * 1000 * 1000)
{
float data = (float)size / 1024 / 1024;
return data.ToString("F2") + " Mb";
}
else
{
float data = (float)size / 1024 / 1024 / 1024;
return data.ToString("F2") + " Gb";
}
}
#endregion
#region TimeSpan Format
///
/// 从一个时间差返回带单位的描述
///
/// 实际的时间差
/// 最终的字符串值
public static string GetTimeSpanDescription(TimeSpan ts)
{
if (ts.TotalSeconds <= 60)
{
return (int)ts.TotalSeconds + " 秒";
}
else if (ts.TotalMinutes <= 60)
{
return ts.TotalMinutes.ToString("F1") + " 分钟";
}
else if (ts.TotalHours <= 24)
{
return ts.TotalHours.ToString("F1") + " 小时";
}
else
{
return ts.TotalDays.ToString("F1") + " 天";
}
}
#endregion
#region Array Expaned
///
/// 一个通用的数组新增个数方法,会自动判断越界情况,越界的情况下,会自动的截断或是填充 ->
/// A common array of new methods, will automatically determine the cross-border situation, in the case of cross-border, will be automatically truncated or filled
///
/// 数据类型
/// 原数据
/// 等待新增的数据
/// 原数据的最大值
public static void AddArrayData(ref T[] array, T[] data, int max)
{
if (data == null) return; // 数据为空
if (data.Length == 0) return; // 数据长度为空
if (array.Length == max)
{
Array.Copy(array, data.Length, array, 0, array.Length - data.Length);
Array.Copy(data, 0, array, array.Length - data.Length, data.Length);
}
else
{
if ((array.Length + data.Length) > max)
{
T[] tmp = new T[max];
for (int i = 0; i < (max - data.Length); i++)
{
tmp[i] = array[i + (array.Length - max + data.Length)];
}
for (int i = 0; i < data.Length; i++)
{
tmp[tmp.Length - data.Length + i] = data[i];
}
// 更新数据
array = tmp;
}
else
{
T[] tmp = new T[array.Length + data.Length];
for (int i = 0; i < array.Length; i++)
{
tmp[i] = array[i];
}
for (int i = 0; i < data.Length; i++)
{
tmp[tmp.Length - data.Length + i] = data[i];
}
array = tmp;
}
}
}
///
/// 将一个数组进行扩充到指定长度,或是缩短到指定长度 ->
/// Extend an array to a specified length, or shorten to a specified length or fill
///
/// 数组的类型
/// 原先数据的数据
/// 新数组的长度
/// 新数组长度信息
public static T[] ArrayExpandToLength(T[] data, int length)
{
if (data == null) return new T[length];
if (data.Length == length) return data;
T[] buffer = new T[length];
Array.Copy(data, buffer, Math.Min(data.Length, buffer.Length));
return buffer;
}
///
/// 将一个数组进行扩充到偶数长度 ->
/// Extend an array to even lengths
///
/// 数组的类型
/// 原先数据的数据
/// 新数组长度信息
public static T[] ArrayExpandToLengthEven(T[] data)
{
if (data == null) return new T[0];
if (data.Length % 2 == 1)
{
return ArrayExpandToLength(data, data.Length + 1);
}
else
{
return data;
}
}
///
/// 将指定的数据按照指定长度进行分割,例如int[10],指定长度4,就分割成int[4],int[4],int[2],然后拼接list
///
/// 数组的类型
/// 等待分割的数组
/// 指定的长度信息
/// 分割后结果内容
public static List ArraySplitByLength(T[] array, int length)
{
if (array == null) return new List();
List result = new List();
int index = 0;
while (index < array.Length)
{
if (index + length < array.Length)
{
T[] tmp = new T[length];
Array.Copy(array, index, tmp, 0, length);
index += length;
result.Add(tmp);
}
else
{
T[] tmp = new T[array.Length - index];
Array.Copy(array, index, tmp, 0, tmp.Length);
index += length;
result.Add(tmp);
}
}
return result;
}
///
/// 将整数进行有效的拆分成数组
///
/// 整数信息
/// 单个的数组长度
/// 拆分后的数组长度
public static int[] SplitIntegerToArray(int integer, int everyLength)
{
int[] result = new int[(integer / everyLength) + ((integer % everyLength) == 0 ? 0 : 1)];
for (int i = 0; i < result.Length; i++)
{
if (i == result.Length - 1)
{
result[i] = (integer % everyLength) == 0 ? everyLength : (integer % everyLength);
}
else
{
result[i] = everyLength;
}
}
return result;
}
#endregion
#region Byte Array compare
///
/// 判断两个字节的指定部分是否相同 ->
/// Determines whether the specified portion of a two-byte is the same
///
/// 第一个字节
/// 第一个字节的起始位置
/// 第二个字节
/// 第二个字节的起始位置
/// 校验的长度
/// 返回是否相等
///
public static bool IsTwoBytesEquel(byte[] b1, int start1, byte[] b2, int start2, int length)
{
if (b1 == null || b2 == null) return false;
for (int i = 0; i < length; i++)
{
if (b1[i + start1] != b2[i + start2])
{
return false;
}
}
return true;
}
///
/// 判断两个字节的指定部分是否相同 ->
/// Determines whether the specified portion of a two-byte is the same
///
/// 第一个字节
/// 第二个字节
/// 返回是否相等
public static bool IsTwoBytesEquel(byte[] b1, byte[] b2)
{
if (b1 == null || b2 == null) return false;
if (b1.Length != b2.Length) return false;
return IsTwoBytesEquel(b1, 0, b2, 0, b1.Length);
}
///
/// 判断两个数据的令牌是否相等 ->
/// Determines whether the tokens of two data are equal
///
/// 字节数据
/// GUID数据
/// 返回是否相等
public static bool IsByteTokenEquel(byte[] head, Guid token)
{
return IsTwoBytesEquel(head, 12, token.ToByteArray(), 0, 16);
}
///
/// 判断两个数据的令牌是否相等 ->
/// Determines whether the tokens of two data are equal
///
/// 第一个令牌
/// 第二个令牌
/// 返回是否相等
public static bool IsTwoTokenEquel(Guid token1, Guid token2)
{
return IsTwoBytesEquel(token1.ToByteArray(), 0, token2.ToByteArray(), 0, 16);
}
#endregion
#region Enum About
///
/// 获取一个枚举类型的所有枚举值,可直接应用于组合框数据 ->
/// Gets all the enumeration values of an enumeration type that can be applied directly to the combo box data
///
/// 枚举的类型值
/// 枚举值数组
public static TEnum[] GetEnumValues() where TEnum : struct
{
return (TEnum[])Enum.GetValues(typeof(TEnum));
}
///
/// 从字符串的枚举值数据转换成真实的枚举值数据 ->
/// Convert enumeration value data from strings to real enumeration value data
///
/// 枚举的类型值
/// 枚举的字符串的数据值
/// 真实的枚举值
public static TEnum GetEnumFromString(string value) where TEnum : struct
{
return (TEnum)Enum.Parse(typeof(TEnum), value);
}
#endregion
#region JSON Data Get
///
/// 一个泛型方法,提供json对象的数据读取 ->
/// A generic method that provides data read for a JSON object
///
/// 读取的泛型
/// json对象
/// 值名称
/// 默认值
/// 值对象
public static T GetValueFromJsonObject(JObject json, string value_name, T default_value)
{
if (json.Property(value_name) != null)
{
return json.Property(value_name).Value.Value();
}
else
{
return default_value;
}
}
///
/// 一个泛型方法,提供json对象的数据写入 ->
/// A generic method that provides data writing to a JSON object
///
/// 写入的泛型
/// json对象
/// 值名称
/// 值数据
public static void JsonSetValue(JObject json, string property, T value)
{
if (json.Property(property) != null)
{
json.Property(property).Value = new JValue(value);
}
else
{
json.Add(property, new JValue(value));
}
}
#endregion
#region Exception Message Format
#if !NETSTANDARD2_0
///
/// 显示一个完整的错误信息 ->
/// Displays a complete error message
///
/// 异常对象
/// 调用本方法可以显示一个异常的详细信息
///
public static void ShowExceptionMessage(Exception ex)
{
//MessageBox.Show(GetExceptionMessage(ex));
}
///
/// 显示一个完整的错误信息,和额外的字符串描述信息 ->
/// Displays a complete error message, and additional string description information
///
/// 额外的描述信息
/// 调用本方法可以显示一个异常的详细信息
/// 异常对象
///
public static void ShowExceptionMessage(string extraMsg, Exception ex)
{
//MessageBox.Show(GetExceptionMessage(extraMsg, ex));
}
#endif
///
/// 获取一个异常的完整错误信息 ->
/// Gets the complete error message for an exception
///
/// 异常对象
/// 完整的字符串数据
/// 获取异常的完整信息
/// ex不能为空
public static string GetExceptionMessage(Exception ex)
{
return StringResources.Language.ExceptionMessage + ex.Message + Environment.NewLine +
StringResources.Language.ExceptionStackTrace + ex.StackTrace + Environment.NewLine +
StringResources.Language.ExceptopnTargetSite + ex.TargetSite;
}
///
/// 获取一个异常的完整错误信息,和额外的字符串描述信息 ->
/// Gets the complete error message for an exception, and additional string description information
///
/// 额外的信息
/// 异常对象
/// 完整的字符串数据
///
public static string GetExceptionMessage(string extraMsg, Exception ex)
{
if (string.IsNullOrEmpty(extraMsg))
{
return GetExceptionMessage(ex);
}
else
{
return extraMsg + Environment.NewLine + GetExceptionMessage(ex);
}
}
#endregion
#region Hex string and Byte[] transform
///
/// 字节数据转化成16进制表示的字符串 ->
/// Byte data into a string of 16 binary representations
///
/// 字节数组
/// 返回的字符串
///
public static string ByteToHexString(byte[] InBytes)
{
return ByteToHexString(InBytes, (char)0);
}
///
/// 字节数据转化成16进制表示的字符串 ->
/// Byte data into a string of 16 binary representations
///
/// 字节数组
/// 分割符
/// 返回的字符串
///
public static string ByteToHexString(byte[] InBytes, char segment)
{
StringBuilder sb = new StringBuilder();
foreach (byte InByte in InBytes)
{
if (segment == 0) sb.Append(string.Format("{0:X2}", InByte));
else sb.Append(string.Format("{0:X2}{1}", InByte, segment));
}
if (segment != 0 && sb.Length > 1 && sb[sb.Length - 1] == segment)
{
sb.Remove(sb.Length - 1, 1);
}
return sb.ToString();
}
///
/// 字符串数据转化成16进制表示的字符串 ->
/// String data into a string of 16 binary representations
///
/// 输入的字符串数据
/// 返回的字符串
///
public static string ByteToHexString(string InString)
{
return ByteToHexString(Encoding.Unicode.GetBytes(InString));
}
private static List hexCharList = new List()
{
'0','1','2','3','4','5','6','7','8','9','A','B','C','D','E','F'
};
///
/// 将16进制的字符串转化成Byte数据,将检测每2个字符转化,也就是说,中间可以是任意字符 ->
/// Converts a 16-character string into byte data, which will detect every 2 characters converted, that is, the middle can be any character
///
/// 十六进制的字符串,中间可以是任意的分隔符
/// 转换后的字节数组
/// 参数举例:AA 01 34 A8
public static byte[] HexStringToBytes(string hex)
{
hex = hex.ToUpper();
MemoryStream ms = new MemoryStream();
for (int i = 0; i < hex.Length; i++)
{
if ((i + 1) < hex.Length)
{
if (hexCharList.Contains(hex[i]) && hexCharList.Contains(hex[i + 1]))
{
// 这是一个合格的字节数据
ms.WriteByte((byte)(hexCharList.IndexOf(hex[i]) * 16 + hexCharList.IndexOf(hex[i + 1])));
i++;
}
}
}
byte[] result = ms.ToArray();
ms.Dispose();
return result;
}
#endregion
#region Byte Reverse By Word
///
/// 将byte数组按照双字节进行反转,如果为单数的情况,则自动补齐 ->
/// Reverses the byte array by double byte, or if the singular is the case, automatically
///
/// 输入的字节信息
/// 反转后的数据
public static byte[] BytesReverseByWord(byte[] inBytes)
{
if (inBytes == null) return null;
byte[] buffer = ArrayExpandToLengthEven(inBytes);
for (int i = 0; i < buffer.Length / 2; i++)
{
byte tmp = buffer[i * 2 + 0];
buffer[i * 2 + 0] = buffer[i * 2 + 1];
buffer[i * 2 + 1] = tmp;
}
return buffer;
}
#endregion
#region Byte[] and AsciiByte[] transform
///
/// 将原始的byte数组转换成ascii格式的byte数组 ->
/// Converts the original byte array to an ASCII-formatted byte array
///
/// 等待转换的byte数组
/// 转换后的数组
public static byte[] BytesToAsciiBytes(byte[] inBytes)
{
return Encoding.ASCII.GetBytes(ByteToHexString(inBytes));
}
///
/// 将ascii格式的byte数组转换成原始的byte数组 ->
/// Converts an ASCII-formatted byte array to the original byte array
///
/// 等待转换的byte数组
/// 转换后的数组
public static byte[] AsciiBytesToBytes(byte[] inBytes)
{
return HexStringToBytes(Encoding.ASCII.GetString(inBytes));
}
///
/// 从字节构建一个ASCII格式的数据内容
///
/// 数据
/// ASCII格式的字节数组
public static byte[] BuildAsciiBytesFrom(byte value)
{
return Encoding.ASCII.GetBytes(value.ToString("X2"));
}
///
/// 从short构建一个ASCII格式的数据内容
///
/// 数据
/// ASCII格式的字节数组
public static byte[] BuildAsciiBytesFrom(short value)
{
return Encoding.ASCII.GetBytes(value.ToString("X4"));
}
///
/// 从ushort构建一个ASCII格式的数据内容
///
/// 数据
/// ASCII格式的字节数组
public static byte[] BuildAsciiBytesFrom(ushort value)
{
return Encoding.ASCII.GetBytes(value.ToString("X4"));
}
///
/// 从字节数组构建一个ASCII格式的数据内容
///
/// 字节信息
/// ASCII格式的地址
public static byte[] BuildAsciiBytesFrom(byte[] value)
{
byte[] buffer = new byte[value.Length * 2];
for (int i = 0; i < value.Length; i++)
{
SoftBasic.BuildAsciiBytesFrom(value[i]).CopyTo(buffer, 2 * i);
}
return buffer;
}
#endregion
#region Bool[] and byte[] transform
///
/// 将bool数组转换到byte数组 ->
/// Converting a bool array to a byte array
///
/// bool数组
/// 转换后的字节数组
public static byte[] BoolArrayToByte(bool[] array)
{
if (array == null) return null;
int length = array.Length % 8 == 0 ? array.Length / 8 : array.Length / 8 + 1;
byte[] buffer = new byte[length];
for (int i = 0; i < array.Length; i++)
{
int index = i / 8;
int offect = i % 8;
byte temp = 0;
switch (offect)
{
case 0: temp = 0x01; break;
case 1: temp = 0x02; break;
case 2: temp = 0x04; break;
case 3: temp = 0x08; break;
case 4: temp = 0x10; break;
case 5: temp = 0x20; break;
case 6: temp = 0x40; break;
case 7: temp = 0x80; break;
default: break;
}
if (array[i]) buffer[index] += temp;
}
return buffer;
}
///
/// 从Byte数组中提取位数组,length代表位数 ->
/// Extracts a bit array from a byte array, length represents the number of digits
///
/// 原先的字节数组
/// 想要转换的长度,如果超出自动会缩小到数组最大长度
/// 转换后的bool数组
public static bool[] ByteToBoolArray(byte[] InBytes, int length)
{
if (InBytes == null) return null;
if (length > InBytes.Length * 8) length = InBytes.Length * 8;
bool[] buffer = new bool[length];
for (int i = 0; i < length; i++)
{
int index = i / 8;
int offect = i % 8;
byte temp = 0;
switch (offect)
{
case 0: temp = 0x01; break;
case 1: temp = 0x02; break;
case 2: temp = 0x04; break;
case 3: temp = 0x08; break;
case 4: temp = 0x10; break;
case 5: temp = 0x20; break;
case 6: temp = 0x40; break;
case 7: temp = 0x80; break;
default: break;
}
if ((InBytes[index] & temp) == temp)
{
buffer[i] = true;
}
}
return buffer;
}
///
/// 从Byte数组中提取所有的位数组 ->
/// Extracts a bit array from a byte array, length represents the number of digits
///
/// 原先的字节数组
/// 转换后的bool数组
public static bool[] ByteToBoolArray(byte[] InBytes)
{
if (InBytes == null) return null;
return ByteToBoolArray(InBytes, InBytes.Length * 8);
}
#endregion
#region Byte[] Splice
///
/// 拼接2个字节数组成一个数组 ->
/// Splicing 2 bytes to to an array
///
/// 数组一
/// 数组二
/// 拼接后的数组
public static byte[] SpliceTwoByteArray(byte[] bytes1, byte[] bytes2)
{
if (bytes1 == null && bytes2 == null) return null;
if (bytes1 == null) return bytes2;
if (bytes2 == null) return bytes1;
byte[] buffer = new byte[bytes1.Length + bytes2.Length];
bytes1.CopyTo(buffer, 0);
bytes2.CopyTo(buffer, bytes1.Length);
return buffer;
}
///
/// 选择一个byte数组的前面的几个byte数据信息
///
/// 原始的数据信息
/// 数据的长度
/// 选择的前面的几个数据信息
public static byte[] BytesArraySelectBegin(byte[] value, int length)
{
byte[] buffer = new byte[Math.Min(value.Length, length)];
Array.Copy(value, 0, buffer, 0, buffer.Length);
return buffer;
}
///
/// 将一个byte数组的前面指定位数移除,返回新的一个数组 ->
/// Removes the preceding specified number of bits in a byte array, returning a new array
///
/// 字节数组
/// 等待移除的长度
/// 新的数据
public static byte[] BytesArrayRemoveBegin(byte[] value, int length)
{
return BytesArrayRemoveDouble(value, length, 0);
}
///
/// 将一个byte数组的后面指定位数移除,返回新的一个数组 ->
/// Removes the specified number of digits after a byte array, returning a new array
///
/// 字节数组
/// 等待移除的长度
/// 新的数据
public static byte[] BytesArrayRemoveLast(byte[] value, int length)
{
return BytesArrayRemoveDouble(value, 0, length);
}
///
/// 将一个byte数组的前后移除指定位数,返回新的一个数组 ->
/// Removes a byte array before and after the specified number of bits, returning a new array
///
/// 字节数组
/// 前面的位数
/// 后面的位数
/// 新的数据
public static byte[] BytesArrayRemoveDouble(byte[] value, int leftLength, int rightLength)
{
if (value == null) return null;
if (value.Length <= (leftLength + rightLength)) return new byte[0];
byte[] buffer = new byte[value.Length - leftLength - rightLength];
Array.Copy(value, leftLength, buffer, 0, buffer.Length);
return buffer;
}
#endregion
#region Basic Framework
///
/// 设置或获取系统框架的版本号 ->
/// Set or get the version number of the system framework
///
///
/// 当你要显示本组件框架的版本号的时候,就可以用这个属性来显示
///
//public static SystemVersion FrameworkVersion { get; set; } = new SystemVersion("7.0.0");
#endregion
#region Deep Clone
///
/// 使用序列化反序列化深度克隆一个对象,该对象需要支持序列化特性 ->
/// Cloning an object with serialization deserialization depth that requires support for serialization attributes
///
/// 源对象,支持序列化
/// 新的一个实例化的对象
///
///
///
///
/// 参数必须实现序列化的特性
///
///
public static object DeepClone(object oringinal)
{
using (MemoryStream stream = new MemoryStream())
{
BinaryFormatter formatter = new BinaryFormatter()
{
Context = new System.Runtime.Serialization.StreamingContext(System.Runtime.Serialization.StreamingContextStates.Clone)
};
formatter.Serialize(stream, oringinal);
stream.Position = 0;
return formatter.Deserialize(stream);
}
}
#endregion
#region Unique String Get
///
/// 获取一串唯一的随机字符串,长度为20,由Guid码和4位数的随机数组成,保证字符串的唯一性 ->
/// Gets a string of unique random strings with a length of 20, consisting of a GUID code and a 4-digit random number to guarantee the uniqueness of the string
///
/// 随机字符串数据
public static string GetUniqueStringByGuidAndRandom()
{
Random random = new Random();
return Guid.NewGuid().ToString("N") + random.Next(1000, 10000);
}
#endregion
}
}