Java 如何高效打印ByteArray的内容

在使用TCP协议作为传输协议时,很多时候都需要输出原始报文用来查看传输数据是否正确。特别是在物联网中的硬件服务器,上下行的数据量非常大,任何关于报文的处理的开销都会因为消息的增加而成倍放大。

不管使用ByteArray或者ByteBuffer当做数据容器,输出日志时,都需要进行两步,第一步:把字节数组的字节转成字符;第二步:在拼接字符形成字符串。

一般的处理方式

迭代拼接字符串

创建StringBuilder,迭代ByteArray,格式化每个byte后,追加到StringBuilder中,最后使用StringBuilder.toString()方法输出完整字符串。

    /**
     * 根据字节数组,输出对应的格式化字符串
     * @param bytes 字节数组
     * @return 字节数组字符串
     */
    public static String printBytesByStringBuilder(byte[] bytes){
        StringBuilder stringBuilder = new StringBuilder();

        for (byte aByte : bytes) {
            stringBuilder.append(byte2String(aByte));
        }

        return stringBuilder.toString();

    }

    public static String byte2String(byte b){

        return String.format("%02x ",b);
    }

特定优化处理

预分配字符数组,使用常量池来实现字节到字符的转换。填充提前预分配好的字符数组。输出字符串!
第一步:穷举所有byte对应的字符数组


穷举所有的byte对应的两位字符

第二步:使用索引的方式进行字符串拼接


    /**
     * 使用字符数组进行byte字节信息的输出 如果默认进制标识的话,不打印'0x',一个byte只需要两个char 例如: 0x9  = '0' + '9' 0xAF = 'A' + 'F'
     * 0x9 0xAF = > '0'+'9'+' '+'A'+'F'
     * <p>
     * 一个byte需要2个字符标识,外加一个空格字符
     *
     * @param bytes 需要格式化的byte
     * @return 字节数组 字符串
     */
    public static String printBytesByCharPool(byte[] bytes) {
        int byteLength = bytes.length;
        int charLength = byteLength * 3;


        char[] content = new char[charLength];
        int unsignedByte = 0;
        int startIndex = 0;
        for (int i = 0; i < byteLength; i++) {
            // 使byte变为无符号的byte
            // b2u
            unsignedByte = ((int) bytes[i]) & 0xFF;

            char[] chars = BYTE_CHARS[unsignedByte];
            startIndex = i * 3;
            // byte的第一位字符
            content[startIndex] = chars[0];
            // byte格式化的第二位字符
            content[startIndex + 1] = chars[1];
            // 尾随的空格字符
            content[startIndex + 2] = WHITE_CHAR;
        }

        return new String(content);
    }

分别使用1024长度的byteArray进行循环1200次打印结果:

# 使用stringBuilder
printBytesByStringBuilder useTime=1070
# 使用字符池
printBytesByCharPool useTime=4

可以看出来速度差别在两个数量级

其他开销分析

分析这两种方法存在的性能开销点有:String.format的性能,StringBuilder的性能

String.format

String.format方法用来格式化代码,进入源码可以看到,每次都会创建一个模板对象,不能复用,这个必然存在一定的性能开销

 public static String format(String format, Object... args) {
        return new Formatter().format(format, args).toString();
    }

StringBuilder的性能

虽然StringBuilder 已经是最常用的字符串拼接方法,如果频繁调用也会发现内部其实也有动态扩容的机制,类似于hasmap的扩容。当容量没有预估时,会发生多次动态扩容。

使用场景

在用于做Tcp服务器时,客户端与服务端使用Tcp长连接进行通讯,需要输出原始报文进行信息查看和追踪,这个时候输出报文的开销就伴随这设备增多会成倍放大,关于输出Tcp中报文字节的开销就不能小视。
如果是次场景优化,其他可以采取的方法有:

  1. 尽量使用定长协议,可以提前分割定长协议的格式化所需要内存空间
  2. 如果在输出字节数组时,需要增加前缀或者后缀,也可以在申请字节数组对应的字符数组时提前把前后缀的内容对应的空间一起创建出来
  3. 如果消息量特别大时也可以考虑使用队列进行延迟或异步打印,不影响网络吞吐
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,772评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,458评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,610评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,640评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,657评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,590评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,962评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,631评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,870评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,611评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,704评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,386评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,969评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,944评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,179评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,742评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,440评论 2 342

推荐阅读更多精彩内容

  • 1.常量&变量 1.1.直接赋值常量值,禁止声明新对象 直接赋值常量值,只是创建了一个对象引用,而这个对象引用指向...
    非著名程序员i阅读 214评论 0 0
  • Java继承关系初始化顺序 父类的静态变量-->父类的静态代码块-->子类的静态变量-->子类的静态代码快-->父...
    第六象限阅读 2,142评论 0 9
  • 表情是什么,我认为表情就是表现出来的情绪。表情可以传达很多信息。高兴了当然就笑了,难过就哭了。两者是相互影响密不可...
    Persistenc_6aea阅读 124,015评论 2 7
  • 16宿命:用概率思维提高你的胜算 以前的我是风险厌恶者,不喜欢去冒险,但是人生放弃了冒险,也就放弃了无数的可能。 ...
    yichen大刀阅读 6,030评论 0 4