这是创建超低延迟的Chronicle FIX-Engine时使用的另一个好技巧。
在从字节流中读取数据时,如果可能的话,将数据存储在char
而不是将其读取到String
效率更高。 (至少您要避免创建String对象,尽管可以通过使用缓存或使用CharSequence
而不是String
来缓解这种情况,但这是另一篇文章的主题。)
使用JMH基准测试,我发现了这些时机:(我没有包括源代码,因为这将是另一篇文章的主题,在此我将更详细地描述不同的方法论)。
从一个字节流中读取2个ascii字符到:
String - 34.48ns
Pooled String - 28.57ns
StringBuilder - 21.27ns
char (using 2 chars method) - 6.75ns
关键是,将数据读入String
至少是char
3倍,甚至没有考虑所创建的垃圾。
因此,不用说,当您知道期望的数据始终是单个字符时,应该将其读取为char
,而不是将其读取为String
变量。
现在,如果您知道流中期望的数据不超过2个字符,该怎么办。 (例如在FIX 5.0标记35 msgType中找到这种情况)。 您是否必须使用字符串来容纳额外的字符? 乍一看似乎如此,毕竟一个char只能包含一个字符。
可以吗
一个Java char
由2个字节组成,而不是一个字节。 因此,如果您知道您的数据是由ascii字符组成的,那么您将知道将仅使用一个字节( char
中的2个字节)。 例如,“ A”是65,而“ z”是122。
您可以通过以下简单循环打印出适合单个字节的值:
for (int i = 0; i < 256; i++) {char c = (char)i;System.out.println(i+ ":" + c);
}
现在,您可以自由使用char的另一个字符来保留第二个ascii字符。
这是这样做的方法:
在此示例中,您已读取2个字节“ a”和“ b”,并希望将它们存储在单个char中。
byte a = (byte)'a';
byte b = (byte)'b';
//Now place a and b into a single char
char ab = (char)((a << 8) + b);//To retrieve the bytes individually see code below
System.out.println((char)(ab>>8) +""+ (char)(ab & 0xff));
为了更好地理解这一点,让我们看一下二进制文件:
byte a = (byte)'a' // 01100001byte b = (byte)'b' // 01100010As you can see below, when viewed as a char, the top 8 bits are not being usedchar ca = 'a' // 00000000 01100001char cb = 'b' // 00000000 01100010Combine the characters with a taking the top 8 bits and b the bottom 8 bits.char ab = (char)((a << 8) + b); // 01100001 01100010
摘要
将数据读入char而不是String效率更高。 如果您知道最多2个ascii字符,则可以将它们组合成一个Java char。 当然,如果您真的担心超低延迟,请仅使用此技术!
翻译自: https://www.javacodegeeks.com/2016/01/writing-2-characters-single-java-char.html