C# 二进制替换第二弹 二进制替换深入研究---List 与byte[] 效率对比 byte[] 数组 替换

上一章节,中提供了简单可以替换的例子,在本节中,进行了bug修改并且对不同方式进行了效率测试


测试用的代码

#region 性能对比
        protected void  二进制数据替换()
        {
            string sourcestr = "68 30 55 08 67 30 55 16 33 16";
            //string oldstr = "30";
            //string newvalstr = "FA FA FA";

            string oldstr = "FA FA FA";
            string newvalstr = "30";

            byte[] source = sourcestr.Replace(" ", "").HexToByteArray();
            //"68 12 00 12 00 68 8D DE 00 00 00 00 0D 62 01 00 68 30 30 34 73 42 31 30 ED 16".Replace(" ", "").HexToByteArray();
            byte[] old = oldstr.Replace(" ", "").HexToByteArray();
            byte[] newval = newvalstr.Replace(" ", "").HexToByteArray();

            byte[] result = HexUtility.Replace(source, old, newval);
            string strs = result.ByteArray2HexString();
            var sb = strs.Replace(" ", string.Empty) == sourcestr.Replace(oldstr, newvalstr).Replace(" ", "");
            if (!sb)
            {
                var d = "替换结果不正确";
            }

            var listRep = strs == source.ReplaceA(old, newval).ByteArray2HexString();



            //全部替换
            result = HexUtility.Replace(source, new List<HexReplaceEntity>()
                                                   {
                                                       new HexReplaceEntity()
                                                           {
                                                               oldValue = "68".Replace(" ", "").HexToByteArray(),
                                                               newValue = "67 02".Replace(" ", "").HexToByteArray(),
                                                           }
                                                           ,
                                                          new HexReplaceEntity()
                                                           {
                                                               oldValue = "67".Replace(" ", "").HexToByteArray(),
                                                               newValue = "67 01".Replace(" ", "").HexToByteArray(),
                                                           },
                                                          new HexReplaceEntity()
                                                           {
                                                               oldValue = "16".Replace(" ", "").HexToByteArray(),
                                                               newValue = "67 03".Replace(" ", "").HexToByteArray(),
                                                           },
                                                   });
            string str2 = result.ByteArray2HexString();
            //字符串替换检查
            if (str2.Replace(" ", string.Empty) != sourcestr
                .Replace("67", "67 01")
                 .Replace("68", "67 02")
                 .Replace("16", "67 03")
                .Replace(" ", string.Empty)
                )
            {

                var d1 = "替换结果不正确";

            }


            string restx = HexUtility.Replace(result, "67 01".Replace(" ", "").HexToByteArray(), "67".Replace(" ", "").HexToByteArray()).ByteArray2HexString();

            result = HexUtility.Replace(result, new List<HexReplaceEntity>()
                                                   {
                                                       new HexReplaceEntity()
                                                           {
                                                               oldValue = "67 02".Replace(" ", "").HexToByteArray(),
                                                               newValue = "68".Replace(" ", "").HexToByteArray(),
                                                           }
                                                           ,
                                                          new HexReplaceEntity()
                                                           {
                                                               oldValue = "67 01".Replace(" ", "").HexToByteArray(),
                                                               newValue = "67".Replace(" ", "").HexToByteArray(),
                                                           },
                                                          new HexReplaceEntity()
                                                           {
                                                               oldValue = "67 03".Replace(" ", "").HexToByteArray(),
                                                               newValue = "16".Replace(" ", "").HexToByteArray(),
                                                           },
                                                   });

            string rel = result.ByteArray2HexString();
            //字符串替换检查
            if (rel.Replace(" ", string.Empty) != sourcestr
                .Replace("67 01", "67")
                 .Replace("67 02", "68")
                 .Replace("67 03", "16")
                .Replace(" ", string.Empty)
                )
            {

                var d2 = "替换结果不正确";

            }
        }
        protected void 二进制数据替换数组复制与List效率对比()
        {
            string sourcestr = "68 30 55 08 67 30 55 16 33 16 68 39 00 39 00 68 C2 DE 00 00 00 00 0C 6D 08 00 01 04 29 00 00 01 00 01 09 04 00 19 44 00 00 02 00 14 03 11 18 24 25 00 01 01 11 42 46 01 48 43 03 01 95 99 00 00 00 00 00 00 00 00 00 00 28 C7 16 68 39 00 39 00 68 C2 DE 00 00 00 00 0C 6D 08 00 01 04 29 00 00 01 00 01 09 04 00 19 44 00 00 02 00 14 03 11 18 24 25 00 01 01 11 42 46 01 48 43 03 01 95 99 00 00 00 00 00 00 00 00 00 00 28 C7 16 68 39 00 39 00 68 C2 DE 00 00 00 00 0C 6D 08 00 01 04 29 00 00 01 00 01 09 04 00 19 44 00 00 02 00 14 03 11 18 24 25 00 01 01 11 42 46 01 48 43 03 01 95 99 00 00 00 00 00 00 00 00 00 00 28 C7 16";
            string oldstr = "30";
            string newvalstr = "FA FA FA";

            //string oldstr = "FA FA FA";
            //string newvalstr = "30";

            byte[] source = sourcestr.Replace(" ", "").HexToByteArray();
            //"68 12 00 12 00 68 8D DE 00 00 00 00 0D 62 01 00 68 30 30 34 73 42 31 30 ED 16".Replace(" ", "").HexToByteArray();
            byte[] old = oldstr.Replace(" ", "").HexToByteArray();
            byte[] newval = newvalstr.Replace(" ", "").HexToByteArray();



            //100万次计算
            var arr = System.Linq.Enumerable.Range(1, 1000000);

            //方法1
            Action act = () => { foreach (var k in arr) { source.Replace(old, newval); } };

            //方法2
            Action act2 = () => { foreach (var k in arr) { source.ReplaceA(old, newval); } };

            //array 用时1828
            System.Diagnostics.Debug.WriteLine("array 用时" + Stopwatch(act));
            //list 用时2481
            System.Diagnostics.Debug.WriteLine("list 用时" + Stopwatch(act2));


        }

        /// <summary>
        /// 测试运行时间
        /// </summary>
        /// <param name="method">需要执行的方法</param>
        /// <param name="summary">对方法的说明</param>
        /// <returns>返回,这个方法执行用了多少毫秒</returns>
        public static long Stopwatch(Action method)
        {
            System.Diagnostics.Stopwatch watch = new System.Diagnostics.Stopwatch();
            watch.Restart();
            //运行方法
            method.Invoke();

            watch.Stop();
           
            return watch.ElapsedMilliseconds;
        }
        /// <summary>
        /// 测试运行时间
        /// </summary>
        /// <param name="method">需要执行的方法</param>
        /// <param name="obj">参数对像</param>
        /// <returns>返回,这个方法执行用了多少毫秒</returns>
        public static long Stopwatch<T>(Action<T> method, T obj)
        {
            return Stopwatch(new Action(() =>
            {
                method.Invoke(obj);
            }));
        }

        #endregion

hextobytearray() 是效字符串转换成为二进制数据


替换核心代码


  /// <summary>
    /// 二进制数据 操作,对单个数组替换时,如果新值长度小于旧值长度时出现的问题进行了修改
    /// </summary>
    public static class HexUtility
    {
        /// <summary>
        /// 二进制替换,如果没有替换则返回原数组对像的复本.比使用List效率高25%
        /// </summary>
        /// <param name="sourceByteArray">源数据</param>
        /// <param name="oldValue">需要替换的数据</param>
        /// <param name="newValue">将要替换成为的数据</param>
        public static byte[] Replace(this byte[] sourceByteArray, byte[] oldValue, byte[] newValue)
        {
            //创建新数据多出1字节
            int newArrayLen = (int)((newValue.Length / (double)oldValue.Length) * sourceByteArray.Length) + 1;
            //得到数组长度
            newArrayLen = Math.Max(newArrayLen, sourceByteArray.Length);
            //新的最后结果数组
            byte[] newByteArray = new byte[newArrayLen];
            //新数组的当前索引
            int curIndex = 0;
            //开始结束
            int start = -1;
            int end = -1;
            //当前查找到的索引
            int oldCurindex = 0;
            //替换数据替换
            for (int x = 0; x < sourceByteArray.Length; x++)
            {
                newByteArray[curIndex] = sourceByteArray[x];
                curIndex++;
                //查找要替换的数据
                if (sourceByteArray[x] == oldValue[oldCurindex])
                {
                    if (oldCurindex == 0)
                    {
                        start = x;
                    }
                    if (oldCurindex == oldValue.Length - 1)
                    {
                        end = x;
                        oldCurindex = 0;
                    }
                    else
                    {
                        oldCurindex++;
                    }
                }
                else
                {
                    oldCurindex = 0;
                }
                //数据查找完成
                if (start != -1 && end != -1)
                {
                    if (curIndex > 0)
                    {
                        curIndex -= 1;
                    }
                    curIndex -= end - start;

                    //复制替换数据
                    Buffer.BlockCopy(newValue, 0, newByteArray, curIndex, newValue.Length);
                    //计算新数组的偏移量
                    curIndex += newValue.Length;
                    //重新设置需要复制索引的索引
                    start = end = -1;
                }
            }

            //处理返回结果
            byte[] result = null;
            if (curIndex != 0)
            {
                result = new byte[curIndex];
                Buffer.BlockCopy(newByteArray, 0, result, 0, result.Length);
            }
            else
            {
                result = new byte[sourceByteArray.Length];
                Buffer.BlockCopy(sourceByteArray, 0, result, 0, result.Length);
            }
            return result;
        }

        /// <summary>
        /// 二进制替换,如果没有替换则返回原数组对像的复本.
        /// </summary>
        /// <param name="sourceByteArray">源数据</param>
        /// <param name="replaces">需要替换的数据集合</param>
        public static byte[] Replace(this byte[] sourceByteArray, List<HexReplaceEntity> replaces)
        {
            //创建新数据多出1字节
            int newArrayLen = (int)((replaces.Sum(p => p.newValue.Length) / (double)replaces.Sum(p => p.oldValue.Length)) * sourceByteArray.Length) + 1;
            //得到数组长度
            newArrayLen = Math.Max(newArrayLen, sourceByteArray.Length);
            //新的最后结果数组
            byte[] newByteArray = new byte[newArrayLen];
            //新数组的当前索引
            int curIndex = 0;
            bool find = false;
            //替换数据替换
            for (int x = 0; x < sourceByteArray.Length; x++)
            {

                foreach (HexReplaceEntity rep in replaces)
                {
                    //查找要替换的数据
                    if (sourceByteArray[x] == rep.oldValue[rep.oldCurindex])
                    {
                        if (rep.oldCurindex == 0)
                        {
                            rep.start = x;
                        }
                        if (rep.oldCurindex == rep.oldValue.Length - 1)
                        {
                            rep.end = x;
                            rep.oldCurindex = 0;
                        }
                        else
                        {
                            rep.oldCurindex++;
                        }
                    }
                    else
                    {
                        rep.oldCurindex = 0;
                        newByteArray[curIndex] = sourceByteArray[x];
                        find = false;
                    }
                    //数据查找完成
                    if (rep.start != -1 && rep.end != -1)
                    {
                        find = true;
                        if (rep.newValue.Length >= rep.oldValue.Length)
                        {
                            //复制替换数据
                            Buffer.BlockCopy(rep.newValue, 0, newByteArray, curIndex, rep.newValue.Length);
                            //计算新数组的偏移量
                            curIndex += rep.newValue.Length;
                        }
                        else
                        //由大字节替换为少字节时出现了问题
                        {
                            curIndex -= rep.end - rep.start;
                            //复制替换数据
                            Buffer.BlockCopy(rep.newValue, 0, newByteArray, curIndex, rep.newValue.Length);
                            //计算新数组的偏移量
                            curIndex += rep.newValue.Length;
                        }
                        //重新设置需要复制索引的索引
                        rep.start = rep.end = -1;
                        break;
                    }
                }
                if (!find)
                {
                    curIndex++;
                }
            }

            //处理返回结果
            byte[] result = null;
            if (curIndex != 0)
            {
                result = new byte[curIndex];
                Buffer.BlockCopy(newByteArray, 0, result, 0, result.Length);
            }
            else
            {
                result = new byte[sourceByteArray.Length];
                Buffer.BlockCopy(sourceByteArray, 0, result, 0, result.Length);
            }
            return result;
        }

        /// <summary>
        /// 二进制替换,如果没有替换则返回原数组对像的复本. 内部使用List,比直接用数组替换效率低25%
        /// </summary>
        /// <param name="sourceByteArray">源数据</param>
        /// 
        /// </summary>
        /// <param name="oldValue">需要替换的数据</param>
        /// <param name="newValue">将要替换成为的数据</param>
        public static byte[] ReplaceA(this byte[] sourceByteArray, byte[] oldValue, byte[] newValue)
        {
            //创建新数据多出1字节
            int newArrayLen = (int)((newValue.Length / (double)oldValue.Length) * sourceByteArray.Length) + 1;
            //得到数组长度
            newArrayLen = Math.Max(newArrayLen, sourceByteArray.Length);
            int oldCurindex = 0;
            List<byte> result = new List<byte>(newArrayLen);
            //替换数据替换
            for (int x = 0; x < sourceByteArray.Length; x++)
            {
                var b = sourceByteArray[x];
                result.Add(b);
                if (b == oldValue[oldCurindex])
                {
                    if (oldCurindex == oldValue.Length - 1)
                    {
                        oldCurindex = 0;
                        //称除现有数据
                        for (int k = 0; k < oldValue.Length; k++)
                        {
                            result.RemoveAt(result.Count - 1);
                        }
                        //添加新数据
                        result.AddRange(newValue);
                    }
                    else
                    {
                        oldCurindex++;
                    }
                }
            }
            byte[] resultarr = result.ToArray();
            result.Clear();
            result = null;
            return resultarr;

        }

    }

    /// <summary>
    /// 替换用的数据实体
    /// </summary>
    public class HexReplaceEntity
    {
        /// <summary>
        /// 需要替换的原始值
        /// </summary>
        public byte[] oldValue { get; set; }

        /// <summary>
        /// 新值
        /// </summary>
        public byte[] newValue { get; set; }

        /// <summary>
        /// 默认开始结束标记
        /// </summary>
        internal int start = -1;
        /// <summary>
        /// 默认开始结束标记
        /// </summary>
        internal int end = -1;

        //当前查找到的索引
        internal int oldCurindex = 0;

    }






  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 4
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值