Move Hex dump related util from ByteBufUtil to inner class
Motivation: Initialisation of the ByteBufUtil class, a class frequently used is delayed because a significant number of String operations is performed to fill a HEXDUMP_ROWPREFIXES array. This array also sticks to the Strings forever. It is quite likely that applications never use the hexdump facility. Modification: Moved the static initialisation and references to a static inner class. This delays initialisation (and memory usage) until actually needed. The API is kept as is. Result: Faster startup time, less memory usage for most netty using applications.
This commit is contained in:
parent
fae6d16467
commit
cb446d8e98
@ -19,7 +19,6 @@ import io.netty.util.CharsetUtil;
|
|||||||
import io.netty.util.Recycler;
|
import io.netty.util.Recycler;
|
||||||
import io.netty.util.Recycler.Handle;
|
import io.netty.util.Recycler.Handle;
|
||||||
import io.netty.util.concurrent.FastThreadLocal;
|
import io.netty.util.concurrent.FastThreadLocal;
|
||||||
import io.netty.util.internal.ObjectUtil;
|
|
||||||
import io.netty.util.internal.PlatformDependent;
|
import io.netty.util.internal.PlatformDependent;
|
||||||
import io.netty.util.internal.StringUtil;
|
import io.netty.util.internal.StringUtil;
|
||||||
import io.netty.util.internal.SystemPropertyUtil;
|
import io.netty.util.internal.SystemPropertyUtil;
|
||||||
@ -36,6 +35,10 @@ import java.nio.charset.CharsetEncoder;
|
|||||||
import java.nio.charset.CoderResult;
|
import java.nio.charset.CoderResult;
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
|
|
||||||
|
import static io.netty.util.internal.StringUtil.NEWLINE;
|
||||||
|
import static io.netty.util.internal.ObjectUtil.checkNotNull;
|
||||||
|
import static io.netty.util.internal.MathUtil.isOutOfBounds;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A collection of utility methods that is related with handling {@link ByteBuf},
|
* A collection of utility methods that is related with handling {@link ByteBuf},
|
||||||
* such as the generation of hex dump and swapping an integer's byte order.
|
* such as the generation of hex dump and swapping an integer's byte order.
|
||||||
@ -51,71 +54,11 @@ public final class ByteBufUtil {
|
|||||||
};
|
};
|
||||||
|
|
||||||
private static final int MAX_CHAR_BUFFER_SIZE;
|
private static final int MAX_CHAR_BUFFER_SIZE;
|
||||||
private static final char[] HEXDUMP_TABLE = new char[256 * 4];
|
private static final int THREAD_LOCAL_BUFFER_SIZE;
|
||||||
private static final String NEWLINE = StringUtil.NEWLINE;
|
|
||||||
private static final String[] BYTE2HEX = new String[256];
|
|
||||||
private static final String[] HEXPADDING = new String[16];
|
|
||||||
private static final String[] BYTEPADDING = new String[16];
|
|
||||||
private static final char[] BYTE2CHAR = new char[256];
|
|
||||||
private static final String[] HEXDUMP_ROWPREFIXES = new String[65536 >>> 4];
|
|
||||||
|
|
||||||
static final ByteBufAllocator DEFAULT_ALLOCATOR;
|
static final ByteBufAllocator DEFAULT_ALLOCATOR;
|
||||||
|
|
||||||
private static final int THREAD_LOCAL_BUFFER_SIZE;
|
|
||||||
|
|
||||||
static {
|
static {
|
||||||
final char[] DIGITS = "0123456789abcdef".toCharArray();
|
|
||||||
for (int i = 0; i < 256; i ++) {
|
|
||||||
HEXDUMP_TABLE[ i << 1 ] = DIGITS[i >>> 4 & 0x0F];
|
|
||||||
HEXDUMP_TABLE[(i << 1) + 1] = DIGITS[i & 0x0F];
|
|
||||||
}
|
|
||||||
|
|
||||||
int i;
|
|
||||||
|
|
||||||
// Generate the lookup table for byte-to-hex-dump conversion
|
|
||||||
for (i = 0; i < BYTE2HEX.length; i ++) {
|
|
||||||
BYTE2HEX[i] = ' ' + StringUtil.byteToHexStringPadded(i);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Generate the lookup table for hex dump paddings
|
|
||||||
for (i = 0; i < HEXPADDING.length; i ++) {
|
|
||||||
int padding = HEXPADDING.length - i;
|
|
||||||
StringBuilder buf = new StringBuilder(padding * 3);
|
|
||||||
for (int j = 0; j < padding; j ++) {
|
|
||||||
buf.append(" ");
|
|
||||||
}
|
|
||||||
HEXPADDING[i] = buf.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Generate the lookup table for byte dump paddings
|
|
||||||
for (i = 0; i < BYTEPADDING.length; i ++) {
|
|
||||||
int padding = BYTEPADDING.length - i;
|
|
||||||
StringBuilder buf = new StringBuilder(padding);
|
|
||||||
for (int j = 0; j < padding; j ++) {
|
|
||||||
buf.append(' ');
|
|
||||||
}
|
|
||||||
BYTEPADDING[i] = buf.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Generate the lookup table for byte-to-char conversion
|
|
||||||
for (i = 0; i < BYTE2CHAR.length; i ++) {
|
|
||||||
if (i <= 0x1f || i >= 0x7f) {
|
|
||||||
BYTE2CHAR[i] = '.';
|
|
||||||
} else {
|
|
||||||
BYTE2CHAR[i] = (char) i;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Generate the lookup table for the start-offset header in each row (up to 64KiB).
|
|
||||||
for (i = 0; i < HEXDUMP_ROWPREFIXES.length; i ++) {
|
|
||||||
StringBuilder buf = new StringBuilder(12);
|
|
||||||
buf.append(NEWLINE);
|
|
||||||
buf.append(Long.toHexString(i << 4 & 0xFFFFFFFFL | 0x100000000L));
|
|
||||||
buf.setCharAt(buf.length() - 9, '|');
|
|
||||||
buf.append('|');
|
|
||||||
HEXDUMP_ROWPREFIXES[i] = buf.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
String allocType = SystemPropertyUtil.get("io.netty.allocator.type", "unpooled").toLowerCase(Locale.US).trim();
|
String allocType = SystemPropertyUtil.get("io.netty.allocator.type", "unpooled").toLowerCase(Locale.US).trim();
|
||||||
ByteBufAllocator alloc;
|
ByteBufAllocator alloc;
|
||||||
if ("unpooled".equals(allocType)) {
|
if ("unpooled".equals(allocType)) {
|
||||||
@ -151,28 +94,29 @@ public final class ByteBufUtil {
|
|||||||
* of the specified buffer's sub-region.
|
* of the specified buffer's sub-region.
|
||||||
*/
|
*/
|
||||||
public static String hexDump(ByteBuf buffer, int fromIndex, int length) {
|
public static String hexDump(ByteBuf buffer, int fromIndex, int length) {
|
||||||
if (length < 0) {
|
return HexUtil.hexDump(buffer, fromIndex, length);
|
||||||
throw new IllegalArgumentException("length: " + length);
|
|
||||||
}
|
|
||||||
if (length == 0) {
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
int endIndex = fromIndex + length;
|
|
||||||
char[] buf = new char[length << 1];
|
|
||||||
|
|
||||||
int srcIdx = fromIndex;
|
|
||||||
int dstIdx = 0;
|
|
||||||
for (; srcIdx < endIndex; srcIdx ++, dstIdx += 2) {
|
|
||||||
System.arraycopy(
|
|
||||||
HEXDUMP_TABLE, buffer.getUnsignedByte(srcIdx) << 1,
|
|
||||||
buf, dstIdx, 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
return new String(buf);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
<<<<<<< HEAD
|
||||||
|
=======
|
||||||
|
* Returns a <a href="http://en.wikipedia.org/wiki/Hex_dump">hex dump</a>
|
||||||
|
* of the specified byte array.
|
||||||
|
*/
|
||||||
|
public static String hexDump(byte[] array) {
|
||||||
|
return hexDump(array, 0, array.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a <a href="http://en.wikipedia.org/wiki/Hex_dump">hex dump</a>
|
||||||
|
* of the specified byte array's sub-region.
|
||||||
|
*/
|
||||||
|
public static String hexDump(byte[] array, int fromIndex, int length) {
|
||||||
|
return HexUtil.hexDump(array, fromIndex, length);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
>>>>>>> e5386b0... Move Hex dump related util from ByteBufUtil to inner class
|
||||||
* Calculates the hash code of the specified buffer. This method is
|
* Calculates the hash code of the specified buffer. This method is
|
||||||
* useful when implementing a new buffer type.
|
* useful when implementing a new buffer type.
|
||||||
*/
|
*/
|
||||||
@ -591,14 +535,7 @@ public final class ByteBufUtil {
|
|||||||
* starting at the given {@code offset} using the given {@code length}.
|
* starting at the given {@code offset} using the given {@code length}.
|
||||||
*/
|
*/
|
||||||
public static String prettyHexDump(ByteBuf buffer, int offset, int length) {
|
public static String prettyHexDump(ByteBuf buffer, int offset, int length) {
|
||||||
if (length == 0) {
|
return HexUtil.prettyHexDump(buffer, offset, length);
|
||||||
return StringUtil.EMPTY_STRING;
|
|
||||||
} else {
|
|
||||||
int rows = length / 16 + (length % 15 == 0? 0 : 1) + 4;
|
|
||||||
StringBuilder buf = new StringBuilder(rows * 80);
|
|
||||||
appendPrettyHexDump(buf, buffer, offset, length);
|
|
||||||
return buf.toString();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -615,7 +552,130 @@ public final class ByteBufUtil {
|
|||||||
* the given {@code length}.
|
* the given {@code length}.
|
||||||
*/
|
*/
|
||||||
public static void appendPrettyHexDump(StringBuilder dump, ByteBuf buf, int offset, int length) {
|
public static void appendPrettyHexDump(StringBuilder dump, ByteBuf buf, int offset, int length) {
|
||||||
if (offset < 0 || length > ObjectUtil.checkNotNull(buf, "buf").capacity() - offset) {
|
HexUtil.appendPrettyHexDump(dump, buf, offset, length);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Separate class so that the expensive static initialization is only done when needed */
|
||||||
|
private static final class HexUtil {
|
||||||
|
|
||||||
|
private static final char[] BYTE2CHAR = new char[256];
|
||||||
|
private static final char[] HEXDUMP_TABLE = new char[256 * 4];
|
||||||
|
private static final String[] HEXPADDING = new String[16];
|
||||||
|
private static final String[] HEXDUMP_ROWPREFIXES = new String[65536 >>> 4];
|
||||||
|
private static final String[] BYTE2HEX = new String[256];
|
||||||
|
private static final String[] BYTEPADDING = new String[16];
|
||||||
|
|
||||||
|
static {
|
||||||
|
final char[] DIGITS = "0123456789abcdef".toCharArray();
|
||||||
|
for (int i = 0; i < 256; i ++) {
|
||||||
|
HEXDUMP_TABLE[ i << 1 ] = DIGITS[i >>> 4 & 0x0F];
|
||||||
|
HEXDUMP_TABLE[(i << 1) + 1] = DIGITS[i & 0x0F];
|
||||||
|
}
|
||||||
|
|
||||||
|
int i;
|
||||||
|
|
||||||
|
// Generate the lookup table for hex dump paddings
|
||||||
|
for (i = 0; i < HEXPADDING.length; i ++) {
|
||||||
|
int padding = HEXPADDING.length - i;
|
||||||
|
StringBuilder buf = new StringBuilder(padding * 3);
|
||||||
|
for (int j = 0; j < padding; j ++) {
|
||||||
|
buf.append(" ");
|
||||||
|
}
|
||||||
|
HEXPADDING[i] = buf.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generate the lookup table for the start-offset header in each row (up to 64KiB).
|
||||||
|
for (i = 0; i < HEXDUMP_ROWPREFIXES.length; i ++) {
|
||||||
|
StringBuilder buf = new StringBuilder(12);
|
||||||
|
buf.append(NEWLINE);
|
||||||
|
buf.append(Long.toHexString(i << 4 & 0xFFFFFFFFL | 0x100000000L));
|
||||||
|
buf.setCharAt(buf.length() - 9, '|');
|
||||||
|
buf.append('|');
|
||||||
|
HEXDUMP_ROWPREFIXES[i] = buf.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generate the lookup table for byte-to-hex-dump conversion
|
||||||
|
for (i = 0; i < BYTE2HEX.length; i ++) {
|
||||||
|
BYTE2HEX[i] = ' ' + StringUtil.byteToHexStringPadded(i);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generate the lookup table for byte dump paddings
|
||||||
|
for (i = 0; i < BYTEPADDING.length; i ++) {
|
||||||
|
int padding = BYTEPADDING.length - i;
|
||||||
|
StringBuilder buf = new StringBuilder(padding);
|
||||||
|
for (int j = 0; j < padding; j ++) {
|
||||||
|
buf.append(' ');
|
||||||
|
}
|
||||||
|
BYTEPADDING[i] = buf.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generate the lookup table for byte-to-char conversion
|
||||||
|
for (i = 0; i < BYTE2CHAR.length; i ++) {
|
||||||
|
if (i <= 0x1f || i >= 0x7f) {
|
||||||
|
BYTE2CHAR[i] = '.';
|
||||||
|
} else {
|
||||||
|
BYTE2CHAR[i] = (char) i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String hexDump(ByteBuf buffer, int fromIndex, int length) {
|
||||||
|
if (length < 0) {
|
||||||
|
throw new IllegalArgumentException("length: " + length);
|
||||||
|
}
|
||||||
|
if (length == 0) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
int endIndex = fromIndex + length;
|
||||||
|
char[] buf = new char[length << 1];
|
||||||
|
|
||||||
|
int srcIdx = fromIndex;
|
||||||
|
int dstIdx = 0;
|
||||||
|
for (; srcIdx < endIndex; srcIdx ++, dstIdx += 2) {
|
||||||
|
System.arraycopy(
|
||||||
|
HEXDUMP_TABLE, buffer.getUnsignedByte(srcIdx) << 1,
|
||||||
|
buf, dstIdx, 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
return new String(buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String hexDump(byte[] array, int fromIndex, int length) {
|
||||||
|
if (length < 0) {
|
||||||
|
throw new IllegalArgumentException("length: " + length);
|
||||||
|
}
|
||||||
|
if (length == 0) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
int endIndex = fromIndex + length;
|
||||||
|
char[] buf = new char[length << 1];
|
||||||
|
|
||||||
|
int srcIdx = fromIndex;
|
||||||
|
int dstIdx = 0;
|
||||||
|
for (; srcIdx < endIndex; srcIdx ++, dstIdx += 2) {
|
||||||
|
System.arraycopy(
|
||||||
|
HEXDUMP_TABLE, (array[srcIdx] & 0xFF) << 1,
|
||||||
|
buf, dstIdx, 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
return new String(buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String prettyHexDump(ByteBuf buffer, int offset, int length) {
|
||||||
|
if (length == 0) {
|
||||||
|
return StringUtil.EMPTY_STRING;
|
||||||
|
} else {
|
||||||
|
int rows = length / 16 + (length % 15 == 0? 0 : 1) + 4;
|
||||||
|
StringBuilder buf = new StringBuilder(rows * 80);
|
||||||
|
appendPrettyHexDump(buf, buffer, offset, length);
|
||||||
|
return buf.toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void appendPrettyHexDump(StringBuilder dump, ByteBuf buf, int offset, int length) {
|
||||||
|
if (isOutOfBounds(offset, length, buf.capacity())) {
|
||||||
throw new IndexOutOfBoundsException(
|
throw new IndexOutOfBoundsException(
|
||||||
"expected: " + "0 <= offset(" + offset + ") <= offset + length(" + length
|
"expected: " + "0 <= offset(" + offset + ") <= offset + length(" + length
|
||||||
+ ") <= " + "buf.capacity(" + buf.capacity() + ')');
|
+ ") <= " + "buf.capacity(" + buf.capacity() + ')');
|
||||||
@ -674,12 +734,10 @@ public final class ByteBufUtil {
|
|||||||
dump.append('|');
|
dump.append('|');
|
||||||
}
|
}
|
||||||
|
|
||||||
dump.append(NEWLINE + "+--------+-------------------------------------------------+----------------+");
|
dump.append(NEWLINE +
|
||||||
|
"+--------+-------------------------------------------------+----------------+");
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Appends the prefix of each hex dump row. Uses the look-up table for the buffer <= 64 KiB.
|
|
||||||
*/
|
|
||||||
private static void appendHexDumpRowPrefix(StringBuilder dump, int row, int rowStartIndex) {
|
private static void appendHexDumpRowPrefix(StringBuilder dump, int row, int rowStartIndex) {
|
||||||
if (row < HEXDUMP_ROWPREFIXES.length) {
|
if (row < HEXDUMP_ROWPREFIXES.length) {
|
||||||
dump.append(HEXDUMP_ROWPREFIXES[row]);
|
dump.append(HEXDUMP_ROWPREFIXES[row]);
|
||||||
@ -690,6 +748,7 @@ public final class ByteBufUtil {
|
|||||||
dump.append('|');
|
dump.append('|');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a cached thread-local direct buffer, if available.
|
* Returns a cached thread-local direct buffer, if available.
|
||||||
|
Loading…
x
Reference in New Issue
Block a user