JsonCharsWriter

This commit is contained in:
redkale
2024-09-16 22:30:25 +08:00
parent fe293c6f25
commit c8ba21dac2
7 changed files with 720 additions and 68 deletions

View File

@@ -348,7 +348,7 @@ public class JsonByteBufferWriter extends JsonWriter {
}
@Override
public void writeFieldShortValue(final byte[] fieldBytes, final short value) {
public void writeFieldShortValue(final byte[] fieldBytes, final char[] fieldChars, final short value) {
byte[] bs1 = fieldBytes;
byte[] bs2 = Utility.latin1ByteArray(String.valueOf(value));
int expandsize = expand(bs1.length + bs2.length);
@@ -374,7 +374,7 @@ public class JsonByteBufferWriter extends JsonWriter {
}
@Override
public void writeFieldIntValue(final byte[] fieldBytes, final int value) {
public void writeFieldIntValue(final byte[] fieldBytes, final char[] fieldChars, final int value) {
byte[] bs1 = fieldBytes;
byte[] bs2 = Utility.latin1ByteArray(String.valueOf(value));
int expandsize = expand(bs1.length + bs2.length);
@@ -400,7 +400,7 @@ public class JsonByteBufferWriter extends JsonWriter {
}
@Override
public void writeFieldLongValue(final byte[] fieldBytes, final long value) {
public void writeFieldLongValue(final byte[] fieldBytes, final char[] fieldChars, final long value) {
byte[] bs1 = fieldBytes;
byte[] bs2 = Utility.latin1ByteArray(String.valueOf(value));
int expandsize = expand(bs1.length + bs2.length);
@@ -426,7 +426,7 @@ public class JsonByteBufferWriter extends JsonWriter {
}
@Override
public void writeFieldLatin1Value(final byte[] fieldBytes, final String value) {
public void writeFieldLatin1Value(final byte[] fieldBytes, final char[] fieldChars, final String value) {
byte[] bs1 = fieldBytes;
byte[] bs2 = Utility.latin1ByteArray(value);
int expandsize = expand(bs1.length + bs2.length + 2);
@@ -466,7 +466,7 @@ public class JsonByteBufferWriter extends JsonWriter {
}
@Override
public void writeLastFieldShortValue(final byte[] fieldBytes, final short value) {
public void writeLastFieldShortValue(final byte[] fieldBytes, final char[] fieldChars, final short value) {
byte[] bs1 = fieldBytes;
byte[] bs2 = Utility.latin1ByteArray(String.valueOf(value));
int expandsize = expand(bs1.length + bs2.length + 1);
@@ -499,7 +499,7 @@ public class JsonByteBufferWriter extends JsonWriter {
}
@Override
public void writeLastFieldIntValue(final byte[] fieldBytes, final int value) {
public void writeLastFieldIntValue(final byte[] fieldBytes, final char[] fieldChars, final int value) {
byte[] bs1 = fieldBytes;
byte[] bs2 = Utility.latin1ByteArray(String.valueOf(value));
int expandsize = expand(bs1.length + bs2.length + 1);
@@ -532,7 +532,7 @@ public class JsonByteBufferWriter extends JsonWriter {
}
@Override
public void writeLastFieldLongValue(final byte[] fieldBytes, final long value) {
public void writeLastFieldLongValue(final byte[] fieldBytes, final char[] fieldChars, final long value) {
byte[] bs1 = fieldBytes;
byte[] bs2 = Utility.latin1ByteArray(String.valueOf(value));
int expandsize = expand(bs1.length + bs2.length + 1);
@@ -565,7 +565,7 @@ public class JsonByteBufferWriter extends JsonWriter {
}
@Override
public void writeLastFieldLatin1Value(final byte[] fieldBytes, final String value) {
public void writeLastFieldLatin1Value(final byte[] fieldBytes, final char[] fieldChars, final String value) {
if (value == null && nullable()) {
writeTo(fieldBytes);
writeNull();
@@ -623,7 +623,8 @@ public class JsonByteBufferWriter extends JsonWriter {
}
@Override // firstFieldBytes 必须带{开头
public void writeObjectByOnlyOneLatin1FieldValue(final byte[] firstFieldBytes, final String value) {
public void writeObjectByOnlyOneLatin1FieldValue(
final byte[] firstFieldBytes, final char[] firstFieldChars, final String value) {
if (value == null && nullable()) {
writeTo('{');
writeTo(firstFieldBytes);
@@ -692,7 +693,12 @@ public class JsonByteBufferWriter extends JsonWriter {
@Override // firstFieldBytes 必须带{开头, lastFieldBytes必须,开头
public void writeObjectByOnlyTwoIntFieldValue(
final byte[] firstFieldBytes, final int value1, final byte[] lastFieldBytes, final int value2) {
final byte[] firstFieldBytes,
final char[] firstFieldChars,
final int value1,
final byte[] lastFieldBytes,
final char[] lastFieldChars,
final int value2) {
byte[] bs1 = firstFieldBytes;
byte[] bs2 = Utility.latin1ByteArray(String.valueOf(value1));
byte[] bs3 = lastFieldBytes;

View File

@@ -31,17 +31,11 @@ public class JsonBytesWriter extends JsonWriter implements ByteTuple {
private static final int TENTHOUSAND_MAX = 10001;
private static final byte[][] TENTHOUSAND_BYTES = new byte[TENTHOUSAND_MAX][];
private static final byte[][] TENTHOUSAND_BYTES2 = new byte[TENTHOUSAND_MAX][];
static {
for (int i = 0; i < TENTHOUSAND_BYTES.length; i++) {
TENTHOUSAND_BYTES[i] = String.valueOf(i).getBytes();
}
}
private static final byte[][] TENTHOUSAND_BYTES2 = new byte[TENTHOUSAND_MAX][];
static {
for (int i = 0; i < TENTHOUSAND_BYTES2.length; i++) {
TENTHOUSAND_BYTES2[i] = String.valueOf(-i).getBytes();
}
}
@@ -191,7 +185,7 @@ public class JsonBytesWriter extends JsonWriter implements ByteTuple {
}
@Override
public void writeFieldShortValue(final byte[] fieldBytes, final short value) {
public void writeFieldShortValue(final byte[] fieldBytes, final char[] fieldChars, final short value) {
byte[] bs1 = fieldBytes;
byte[] bs2 = (value >= 0 && value < TENTHOUSAND_MAX)
? TENTHOUSAND_BYTES[value]
@@ -208,7 +202,7 @@ public class JsonBytesWriter extends JsonWriter implements ByteTuple {
}
@Override
public void writeFieldIntValue(final byte[] fieldBytes, final int value) {
public void writeFieldIntValue(final byte[] fieldBytes, final char[] fieldChars, final int value) {
byte[] bs1 = fieldBytes;
byte[] bs2 = (value >= 0 && value < TENTHOUSAND_MAX)
? TENTHOUSAND_BYTES[value]
@@ -225,7 +219,7 @@ public class JsonBytesWriter extends JsonWriter implements ByteTuple {
}
@Override
public void writeFieldLongValue(final byte[] fieldBytes, final long value) {
public void writeFieldLongValue(final byte[] fieldBytes, final char[] fieldChars, final long value) {
byte[] bs1 = fieldBytes;
byte[] bs2 = (value >= 0 && value < TENTHOUSAND_MAX)
? TENTHOUSAND_BYTES[(int) value]
@@ -242,7 +236,7 @@ public class JsonBytesWriter extends JsonWriter implements ByteTuple {
}
@Override
public void writeFieldLatin1Value(final byte[] fieldBytes, final String value) {
public void writeFieldLatin1Value(final byte[] fieldBytes, final char[] fieldChars, final String value) {
byte[] bs1 = fieldBytes;
byte[] bs2 = Utility.latin1ByteArray(value);
int len1 = bs1.length;
@@ -257,7 +251,7 @@ public class JsonBytesWriter extends JsonWriter implements ByteTuple {
}
@Override
public void writeLastFieldShortValue(final byte[] fieldBytes, final short value) {
public void writeLastFieldShortValue(final byte[] fieldBytes, final char[] fieldChars, final short value) {
byte[] bs1 = fieldBytes;
byte[] bs2 = (value >= 0 && value < TENTHOUSAND_MAX)
? TENTHOUSAND_BYTES[value]
@@ -275,7 +269,7 @@ public class JsonBytesWriter extends JsonWriter implements ByteTuple {
}
@Override
public void writeLastFieldIntValue(final byte[] fieldBytes, final int value) {
public void writeLastFieldIntValue(final byte[] fieldBytes, final char[] fieldChars, final int value) {
byte[] bs1 = fieldBytes;
byte[] bs2 = (value >= 0 && value < TENTHOUSAND_MAX)
? TENTHOUSAND_BYTES[value]
@@ -293,7 +287,7 @@ public class JsonBytesWriter extends JsonWriter implements ByteTuple {
}
@Override
public void writeLastFieldLongValue(final byte[] fieldBytes, final long value) {
public void writeLastFieldLongValue(final byte[] fieldBytes, final char[] fieldChars, final long value) {
byte[] bs1 = fieldBytes;
byte[] bs2 = (value >= 0 && value < TENTHOUSAND_MAX)
? TENTHOUSAND_BYTES[(int) value]
@@ -311,7 +305,7 @@ public class JsonBytesWriter extends JsonWriter implements ByteTuple {
}
@Override
public void writeLastFieldLatin1Value(final byte[] fieldBytes, final String value) {
public void writeLastFieldLatin1Value(final byte[] fieldBytes, final char[] fieldChars, final String value) {
if (value == null && nullable()) {
writeTo(fieldBytes);
writeNull();
@@ -340,7 +334,8 @@ public class JsonBytesWriter extends JsonWriter implements ByteTuple {
}
@Override // firstFieldBytes 必须带{开头
public void writeObjectByOnlyOneLatin1FieldValue(final byte[] firstFieldBytes, final String value) {
public void writeObjectByOnlyOneLatin1FieldValue(
final byte[] firstFieldBytes, final char[] firstFieldChars, final String value) {
if (value == null && nullable()) {
writeTo(BYTE_LBRACE);
writeTo(firstFieldBytes);
@@ -372,7 +367,12 @@ public class JsonBytesWriter extends JsonWriter implements ByteTuple {
@Override // firstFieldBytes 必须带{开头, lastFieldBytes必须,开头
public void writeObjectByOnlyTwoIntFieldValue(
final byte[] firstFieldBytes, final int value1, final byte[] lastFieldBytes, final int value2) {
final byte[] firstFieldBytes,
final char[] firstFieldChars,
final int value1,
final byte[] lastFieldBytes,
final char[] lastFieldChars,
final int value2) {
byte[] bs1 = firstFieldBytes;
byte[] bs2 = (value1 >= 0 && value1 < TENTHOUSAND_MAX)
? TENTHOUSAND_BYTES[value1]
@@ -466,6 +466,14 @@ public class JsonBytesWriter extends JsonWriter implements ByteTuple {
bytes[curr++] = '\\';
bytes[curr++] = 'r';
break;
case '\f':
bytes[curr++] = '\\';
bytes[curr++] = 'f';
break;
case '\b':
bytes[curr++] = '\\';
bytes[curr++] = 'b';
break;
case '\t':
bytes[curr++] = '\\';
bytes[curr++] = 't';
@@ -474,9 +482,9 @@ public class JsonBytesWriter extends JsonWriter implements ByteTuple {
bytes[curr++] = '\\';
bytes[curr++] = '\\';
break;
case '"':
case BYTE_DQUOTE:
bytes[curr++] = '\\';
bytes[curr++] = '"';
bytes[curr++] = BYTE_DQUOTE;
break;
default:
if (ch < 0x80) {
@@ -521,7 +529,7 @@ public class JsonBytesWriter extends JsonWriter implements ByteTuple {
if (b2 == 0 && b >= 0) {
if (b == BYTE_DQUOTE) {
bytes[curr++] = '\\';
bytes[curr++] = '"';
bytes[curr++] = BYTE_DQUOTE;
} else if (b == '\\') {
bytes[curr++] = '\\';
bytes[curr++] = '\\';
@@ -583,14 +591,14 @@ public class JsonBytesWriter extends JsonWriter implements ByteTuple {
count = curr;
}
private static final char MIN_LOW_SURROGATE = '\uDC00';
private static final char MAX_LOW_SURROGATE = '\uDFFF';
private static final char MIN_HIGH_SURROGATE = '\uD800';
private static final char MAX_HIGH_SURROGATE = '\uDBFF';
private static final int MIN_SUPPLEMENTARY_CODE_POINT = 0x010000;
private static final char MAX_LOW_SURROGATE_MORE = MAX_LOW_SURROGATE + 1;
private static final char MAX_HIGH_SURROGATE_MORE = MAX_HIGH_SURROGATE + 1;
private static final int MIN_SUPPLEMENTARY_CODE_POINT_MORE =
static final char MIN_LOW_SURROGATE = '\uDC00';
static final char MAX_LOW_SURROGATE = '\uDFFF';
static final char MIN_HIGH_SURROGATE = '\uD800';
static final char MAX_HIGH_SURROGATE = '\uDBFF';
static final int MIN_SUPPLEMENTARY_CODE_POINT = 0x010000;
static final char MAX_LOW_SURROGATE_MORE = MAX_LOW_SURROGATE + 1;
static final char MAX_HIGH_SURROGATE_MORE = MAX_HIGH_SURROGATE + 1;
static final int MIN_SUPPLEMENTARY_CODE_POINT_MORE =
(MIN_SUPPLEMENTARY_CODE_POINT - (MIN_HIGH_SURROGATE << 10) - MIN_LOW_SURROGATE);
private void writeEscapeLatinString(final boolean quote, byte[] value) {
@@ -602,7 +610,7 @@ public class JsonBytesWriter extends JsonWriter implements ByteTuple {
for (byte b : value) {
if (b == BYTE_DQUOTE) {
bytes[curr++] = '\\';
bytes[curr++] = '"';
bytes[curr++] = BYTE_DQUOTE;
} else if (b == '\\') {
bytes[curr++] = '\\';
bytes[curr++] = '\\';

View File

@@ -0,0 +1,554 @@
/*
* Copyright (c) 2016-2116 Redkale
* All rights reserved.
*/
package org.redkale.convert.json;
import static org.redkale.convert.json.JsonWriter.BYTE_DQUOTE;
import static org.redkale.convert.json.JsonWriter.BYTE_LBRACE;
import static org.redkale.convert.json.JsonWriter.BYTE_RBRACE;
import static org.redkale.convert.json.JsonWriter.DEFAULT_SIZE;
import static org.redkale.convert.json.JsonWriter.DigitOnes;
import static org.redkale.convert.json.JsonWriter.DigitTens;
import static org.redkale.convert.json.JsonWriter.digits;
import static org.redkale.convert.json.JsonWriter.sizeTable;
import org.redkale.util.StringWrapper;
import org.redkale.util.Utility;
/**
* writeTo系列的方法输出的字符不能含特殊字符
*
* <p>详情见: https://redkale.org
*
* @author zhangjx
* @since 2.3.0
*/
public class JsonCharsWriter extends JsonWriter {
private static final char[] CHARS_TUREVALUE = "true".toCharArray();
private static final char[] CHARS_FALSEVALUE = "false".toCharArray();
private static final int TENTHOUSAND_MAX = 10001;
private static final char[][] TENTHOUSAND_CHARS = new char[TENTHOUSAND_MAX][];
private static final char[][] TENTHOUSAND_CHARS2 = new char[TENTHOUSAND_MAX][];
static {
for (int i = 0; i < TENTHOUSAND_CHARS.length; i++) {
TENTHOUSAND_CHARS[i] = String.valueOf(i).toCharArray();
TENTHOUSAND_CHARS2[i] = String.valueOf(-i).toCharArray();
}
}
private int count;
private char[] content;
public JsonCharsWriter() {
this(DEFAULT_SIZE);
}
public JsonCharsWriter(int size) {
this.content = new char[size > 1024 ? size : 1024];
}
// -----------------------------------------------------------------------
/**
* 返回指定至少指定长度的缓冲区
*
* @param len
*
* @return
*/
private char[] expand(int len) {
int ncount = count + len;
if (ncount <= content.length) {
return content;
}
char[] newdata = new char[Math.max(content.length * 3 / 2, ncount)];
System.arraycopy(content, 0, newdata, 0, count);
this.content = newdata;
return newdata;
}
@Override
protected boolean recycle() {
super.recycle();
this.count = 0;
if (this.content != null && this.content.length > DEFAULT_SIZE * 100) {
this.content = new char[DEFAULT_SIZE];
}
return true;
}
@Override
public final boolean charsMode() {
return true;
}
@Override
public void writeTo(final char ch) { // 只能是 0 - 127 的字符
expand(1);
content[count++] = ch;
}
@Override
public void writeTo(final char[] chs, final int start, final int len) { // 只能是 0 - 127 的字符
expand(len);
System.arraycopy(chs, start, content, count, len);
count += len;
}
@Override
public void writeTo(final byte b) { // 只能是 0 - 127 的字符
expand(1);
content[count++] = (char) (b & 0xff);
}
@Override
public void writeTo(final byte[] bs, final int start, final int len) { // 只能是 0 - 127 的字符
char[] chars = expand(len);
for (int i = 0; i < len; i++) {
chars[count + i] = (char) (bs[start + i] & 0xff);
}
count += len;
}
/**
* <b>注意:</b> 该String值不能为null且不会进行转义 只用于不含需要转义字符的字符串例如enum、double、BigInteger、BigDecimal转换的String
*
* @param quote 是否加双引号
* @param value 非null且不含需要转义的字符的String值
*/
@Override
public void writeLatin1To(final boolean quote, final String value) {
int len = value.length();
if (quote) {
expand(len + 2);
content[count++] = BYTE_DQUOTE;
value.getChars(0, len, content, count);
count += len;
content[count++] = BYTE_DQUOTE;
} else {
expand(len);
value.getChars(0, len, content, count);
count += len;
}
}
@Override
public void writeFieldShortValue(final byte[] fieldBytes, final char[] fieldChars, final short value) {
char[] bs1 = fieldChars;
char[] bs2 = (value >= 0 && value < TENTHOUSAND_MAX)
? TENTHOUSAND_CHARS[value]
: ((value < 0 && value > -TENTHOUSAND_MAX)
? TENTHOUSAND_CHARS2[-value]
: String.valueOf(value).toCharArray());
int len1 = bs1.length;
int len2 = bs2.length;
char[] src = expand(len1 + len2);
System.arraycopy(bs1, 0, src, count, len1);
count += len1;
System.arraycopy(bs2, 0, src, count, len2);
count += len2;
}
@Override
public void writeFieldIntValue(final byte[] fieldBytes, final char[] fieldChars, final int value) {
char[] bs1 = fieldChars;
char[] bs2 = (value >= 0 && value < TENTHOUSAND_MAX)
? TENTHOUSAND_CHARS[value]
: ((value < 0 && value > -TENTHOUSAND_MAX)
? TENTHOUSAND_CHARS2[-value]
: String.valueOf(value).toCharArray());
int len1 = bs1.length;
int len2 = bs2.length;
char[] src = expand(len1 + len2);
System.arraycopy(bs1, 0, src, count, len1);
count += len1;
System.arraycopy(bs2, 0, src, count, len2);
count += len2;
}
@Override
public void writeFieldLongValue(final byte[] fieldBytes, final char[] fieldChars, final long value) {
char[] bs1 = fieldChars;
char[] bs2 = (value >= 0 && value < TENTHOUSAND_MAX)
? TENTHOUSAND_CHARS[(int) value]
: ((value < 0 && value > -TENTHOUSAND_MAX)
? TENTHOUSAND_CHARS2[(int) -value]
: String.valueOf(value).toCharArray());
int len1 = bs1.length;
int len2 = bs2.length;
char[] src = expand(len1 + len2);
System.arraycopy(bs1, 0, src, count, len1);
count += len1;
System.arraycopy(bs2, 0, src, count, len2);
count += len2;
}
@Override
public void writeFieldLatin1Value(final byte[] fieldBytes, final char[] fieldChars, final String value) {
char[] bs1 = fieldChars;
int len1 = bs1.length;
int len2 = value.length();
char[] src = expand(len1 + len2 + 2);
System.arraycopy(bs1, 0, src, count, len1);
count += len1;
src[count++] = BYTE_DQUOTE;
value.getChars(0, len2, content, count);
count += len2;
src[count++] = BYTE_DQUOTE;
}
@Override
public void writeLastFieldShortValue(final byte[] fieldBytes, final char[] fieldChars, final short value) {
char[] bs1 = fieldChars;
char[] bs2 = (value >= 0 && value < TENTHOUSAND_MAX)
? TENTHOUSAND_CHARS[value]
: ((value < 0 && value > -TENTHOUSAND_MAX)
? TENTHOUSAND_CHARS2[-value]
: String.valueOf(value).toCharArray());
int len1 = bs1.length;
int len2 = bs2.length;
char[] src = expand(len1 + len2 + 1);
System.arraycopy(bs1, 0, src, count, len1);
count += len1;
System.arraycopy(bs2, 0, src, count, len2);
count += len2;
src[count++] = BYTE_RBRACE;
}
@Override
public void writeLastFieldIntValue(final byte[] fieldBytes, final char[] fieldChars, final int value) {
char[] bs1 = fieldChars;
char[] bs2 = (value >= 0 && value < TENTHOUSAND_MAX)
? TENTHOUSAND_CHARS[value]
: ((value < 0 && value > -TENTHOUSAND_MAX)
? TENTHOUSAND_CHARS2[-value]
: String.valueOf(value).toCharArray());
int len1 = bs1.length;
int len2 = bs2.length;
char[] src = expand(len1 + len2 + 1);
System.arraycopy(bs1, 0, src, count, len1);
count += len1;
System.arraycopy(bs2, 0, src, count, len2);
count += len2;
src[count++] = BYTE_RBRACE;
}
@Override
public void writeLastFieldLongValue(final byte[] fieldBytes, final char[] fieldChars, final long value) {
char[] bs1 = fieldChars;
char[] bs2 = (value >= 0 && value < TENTHOUSAND_MAX)
? TENTHOUSAND_CHARS[(int) value]
: ((value < 0 && value > -TENTHOUSAND_MAX)
? TENTHOUSAND_CHARS2[(int) -value]
: String.valueOf(value).toCharArray());
int len1 = bs1.length;
int len2 = bs2.length;
char[] src = expand(len1 + len2 + 1);
System.arraycopy(bs1, 0, src, count, len1);
count += len1;
System.arraycopy(bs2, 0, src, count, len2);
count += len2;
src[count++] = BYTE_RBRACE;
}
@Override
public void writeLastFieldLatin1Value(final byte[] fieldBytes, final char[] fieldChars, final String value) {
if (value == null && nullable()) {
writeTo(fieldChars);
writeNull();
writeTo(BYTE_RBRACE);
return;
}
if (value == null || (tiny() && value.isEmpty())) {
expand(1);
content[count++] = BYTE_RBRACE;
} else {
int len1 = fieldChars.length;
int len2 = value.length();
char[] src = expand(len1 + len2 + 3);
System.arraycopy(fieldChars, 0, src, count, len1);
count += len1;
content[count++] = BYTE_DQUOTE;
value.getChars(0, len2, content, count);
count += len2;
content[count++] = BYTE_DQUOTE;
content[count++] = BYTE_RBRACE;
}
}
@Override // firstFieldBytes 必须带{开头
public void writeObjectByOnlyOneLatin1FieldValue(
byte[] firstFieldBytes, char[] firstFieldChars, final String value) {
if (value == null && nullable()) {
writeTo(BYTE_LBRACE);
writeTo(firstFieldChars);
writeNull();
writeTo(BYTE_RBRACE);
return;
}
if (value == null || (tiny() && value.isEmpty())) {
expand(2);
content[count++] = BYTE_LBRACE;
content[count++] = BYTE_RBRACE;
} else {
int len1 = firstFieldChars.length;
int len2 = value.length();
char[] src = expand(len1 + len2 + 3);
System.arraycopy(firstFieldChars, 0, src, count, len1);
count += len1;
content[count++] = BYTE_DQUOTE;
value.getChars(0, len2, content, count);
count += len2;
content[count++] = BYTE_DQUOTE;
content[count++] = BYTE_RBRACE;
}
}
@Override // firstFieldBytes 必须带{开头, lastFieldBytes必须,开头
public void writeObjectByOnlyTwoIntFieldValue(
final byte[] firstFieldBytes,
final char[] firstFieldChars,
final int value1,
final byte[] lastFieldBytes,
final char[] lastFieldChars,
final int value2) {
char[] bs1 = firstFieldChars;
char[] bs2 = (value1 >= 0 && value1 < TENTHOUSAND_MAX)
? TENTHOUSAND_CHARS[value1]
: ((value1 < 0 && value1 > -TENTHOUSAND_MAX)
? TENTHOUSAND_CHARS2[-value1]
: String.valueOf(value1).toCharArray());
char[] bs3 = lastFieldChars;
char[] bs4 = (value2 >= 0 && value2 < TENTHOUSAND_MAX)
? TENTHOUSAND_CHARS[value2]
: ((value2 < 0 && value2 > -TENTHOUSAND_MAX)
? TENTHOUSAND_CHARS2[-value2]
: String.valueOf(value2).toCharArray());
int len1 = bs1.length;
int len2 = bs2.length;
int len3 = bs3.length;
int len4 = bs4.length;
char[] src = expand(len1 + len2 + len3 + len4 + 1);
System.arraycopy(bs1, 0, src, count, len1);
count += len1;
System.arraycopy(bs2, 0, src, count, len2);
count += len2;
System.arraycopy(bs3, 0, src, count, len3);
count += len3;
System.arraycopy(bs4, 0, src, count, len4);
count += len4;
src[count++] = BYTE_RBRACE;
}
public byte[] toBytes() {
return Utility.encodeUTF8(content, 0, count);
}
public int count() {
return this.count;
}
@Override
public void writeString(String value) {
writeString(true, value);
}
@Override
public void writeString(final boolean quote, String value) {
if (value == null) {
writeNull();
return;
}
final String str = value;
final int len = str.length();
char[] chars = expand(len + 2);
int curr = count;
if (quote) {
chars[curr++] = BYTE_DQUOTE;
}
for (int i = 0; i < len; i++) {
char ch = str.charAt(i);
switch (ch) {
case '\n':
chars[curr++] = '\\';
chars[curr++] = 'n';
break;
case '\r':
chars[curr++] = '\\';
chars[curr++] = 'r';
break;
case '\f':
chars[curr++] = '\\';
chars[curr++] = 'f';
break;
case '\b':
chars[curr++] = '\\';
chars[curr++] = 'b';
break;
case '\t':
chars[curr++] = '\\';
chars[curr++] = 't';
break;
case '\\':
chars[curr++] = '\\';
chars[curr++] = ch;
break;
case '"':
chars[curr++] = '\\';
chars[curr++] = ch;
break;
default:
chars[curr++] = ch;
break;
}
}
if (quote) {
chars[curr++] = BYTE_DQUOTE;
}
count = curr;
}
@Override
public String toString() {
return new String(content, 0, count);
}
// ----------------------------------------------------------------------------------------------
@Override
public void writeBoolean(boolean value) {
writeTo(value ? CHARS_TUREVALUE : CHARS_FALSEVALUE);
}
@Override
public void writeInt(int value) {
if (value >= 0 && value < TENTHOUSAND_MAX) {
writeTo(TENTHOUSAND_CHARS[value]);
return;
}
if (value < 0 && value > -TENTHOUSAND_MAX) {
writeTo(TENTHOUSAND_CHARS2[-value]);
return;
}
final char sign = value >= 0 ? 0 : '-';
if (value < 0) value = -value;
int size;
for (int i = 0; ; i++) {
if (value <= sizeTable[i]) {
size = i + 1;
break;
}
}
if (sign != 0) size++; // 负数
expand(size);
int q, r;
int charPos = count + size;
// Generate two digits per iteration
while (value >= 65536) {
q = value / 100;
// really: r = i - (q * 100);
r = value - ((q << 6) + (q << 5) + (q << 2));
value = q;
content[--charPos] = DigitOnes[r];
content[--charPos] = DigitTens[r];
}
// Fall thru to fast mode for smaller numbers
// assert(i <= 65536, i);
for (; ; ) {
q = (value * 52429) >>> (16 + 3);
r = value - ((q << 3) + (q << 1)); // r = i-(q*10) ...
content[--charPos] = digits[r];
value = q;
if (value == 0) break;
}
if (sign != 0) content[--charPos] = sign;
count += size;
}
@Override
public void writeLong(long value) {
if (value >= 0 && value < TENTHOUSAND_MAX) {
writeTo(TENTHOUSAND_CHARS[(int) value]);
return;
}
if (value < 0 && value > -TENTHOUSAND_MAX) {
writeTo(TENTHOUSAND_CHARS2[(int) -value]);
return;
}
final char sign = value >= 0 ? 0 : '-';
if (value < 0) value = -value;
int size = 19;
long p = 10;
for (int i = 1; i < 19; i++) {
if (value < p) {
size = i;
break;
}
p = 10 * p;
}
if (sign != 0) size++; // 负数
expand(size);
long q;
int r;
int charPos = count + size;
// Get 2 digits/iteration using longs until quotient fits into an int
while (value > Integer.MAX_VALUE) {
q = value / 100;
// really: r = i - (q * 100);
r = (int) (value - ((q << 6) + (q << 5) + (q << 2)));
value = q;
content[--charPos] = DigitOnes[r];
content[--charPos] = DigitTens[r];
}
// Get 2 digits/iteration using ints
int q2;
int i2 = (int) value;
while (i2 >= 65536) {
q2 = i2 / 100;
// really: r = i2 - (q * 100);
r = i2 - ((q2 << 6) + (q2 << 5) + (q2 << 2));
i2 = q2;
content[--charPos] = DigitOnes[r];
content[--charPos] = DigitTens[r];
}
// Fall thru to fast mode for smaller numbers
// assert(i2 <= 65536, i2);
for (; ; ) {
q2 = (i2 * 52429) >>> (16 + 3);
r = i2 - ((q2 << 3) + (q2 << 1)); // r = i2-(q2*10) ...
content[--charPos] = digits[r];
i2 = q2;
if (i2 == 0) break;
}
if (sign != 0) content[--charPos] = sign;
count += size;
}
@Override
public void writeWrapper(StringWrapper wrapper) {
if (wrapper == null || wrapper.getValue() == null) {
writeNull();
return;
}
String value = wrapper.getValue();
if (Utility.isLatin1(value)) {
writeTo(Utility.latin1ByteArray(value));
return;
}
int len2 = value.length();
char[] chars = expand(len2);
value.getChars(0, len2, chars, count);
count += len2;
}
}

View File

@@ -31,6 +31,8 @@ public class JsonConvert extends TextConvert<JsonReader, JsonWriter> {
private final ThreadLocal<JsonBytesWriter> bytesWriterPool = Utility.withInitialThreadLocal(JsonBytesWriter::new);
private final ThreadLocal<JsonCharsWriter> charsWriterPool = Utility.withInitialThreadLocal(JsonCharsWriter::new);
private final Consumer<JsonBytesWriter> offerBytesConsumer = this::offerJsonBytesWriter;
private final ThreadLocal<JsonReader> readerPool = Utility.withInitialThreadLocal(JsonReader::new);
@@ -138,6 +140,22 @@ public class JsonConvert extends TextConvert<JsonReader, JsonWriter> {
}
}
private JsonCharsWriter pollJsonCharsWriter() {
JsonCharsWriter writer = charsWriterPool.get();
if (writer == null) {
writer = new JsonCharsWriter();
} else {
charsWriterPool.set(null);
}
return configWrite((JsonCharsWriter) writer.withFeatures(features));
}
private void offerJsonCharsWriter(final JsonCharsWriter writer) {
if (writer != null) {
writer.recycle();
charsWriterPool.set(writer);
}
}
// ------------------------------ convertFrom -----------------------------------------------------------
@Override
public <T> T convertFrom(final Type type, final byte[] bytes) {
@@ -305,7 +323,7 @@ public class JsonConvert extends TextConvert<JsonReader, JsonWriter> {
if (value == null) {
return "null";
}
JsonBytesWriter writer = pollJsonBytesWriter();
JsonCharsWriter writer = pollJsonCharsWriter();
final Type t = type == null ? value.getClass() : type;
Encodeable encoder = this.lastConvertEncodeable;
if (encoder == null || encoder.getType() != t) {
@@ -318,7 +336,7 @@ public class JsonConvert extends TextConvert<JsonReader, JsonWriter> {
encoder.convertTo(writer, value);
String result = writer.toString();
offerJsonBytesWriter(writer);
offerJsonCharsWriter(writer);
return result;
}

View File

@@ -511,6 +511,12 @@ public abstract class JsonDynEncoder<T> implements Encodeable<JsonWriter, T> {
fv.visitEnd();
fv = cw.visitField(ACC_PROTECTED + ACC_FINAL, fieldname + "FirstFieldBytes", "[B", null, null);
fv.visitEnd();
fv = cw.visitField(ACC_PROTECTED + ACC_FINAL, fieldname + "FieldChars", "[C", null, null);
fv.visitEnd();
fv = cw.visitField(ACC_PROTECTED + ACC_FINAL, fieldname + "CommaFieldChars", "[C", null, null);
fv.visitEnd();
fv = cw.visitField(ACC_PROTECTED + ACC_FINAL, fieldname + "FirstFieldChars", "[C", null, null);
fv.visitEnd();
final Class fieldType = readGetSetFieldType(element);
if (fieldType != String.class && !fieldType.isPrimitive()) {
fv = cw.visitField(ACC_PROTECTED, fieldname + "Encoder", encodeableDesc, null, null);
@@ -561,6 +567,21 @@ public abstract class JsonDynEncoder<T> implements Encodeable<JsonWriter, T> {
mv.visitLdcInsn("{\"" + fieldname + "\":");
mv.visitMethodInsn(INVOKEVIRTUAL, "java/lang/String", "getBytes", "()[B", false);
mv.visitFieldInsn(PUTFIELD, newDynName, fieldname + "FirstFieldBytes", "[B");
// xxxFieldChars
mv.visitVarInsn(ALOAD, 0);
mv.visitLdcInsn("\"" + fieldname + "\":");
mv.visitMethodInsn(INVOKEVIRTUAL, "java/lang/String", "toCharArray", "()[C", false);
mv.visitFieldInsn(PUTFIELD, newDynName, fieldname + "FieldChars", "[C");
// xxxCommaFieldChars
mv.visitVarInsn(ALOAD, 0);
mv.visitLdcInsn(",\"" + fieldname + "\":");
mv.visitMethodInsn(INVOKEVIRTUAL, "java/lang/String", "toCharArray", "()[C", false);
mv.visitFieldInsn(PUTFIELD, newDynName, fieldname + "CommaFieldChars", "[C");
// xxxFirstFieldChars
mv.visitVarInsn(ALOAD, 0);
mv.visitLdcInsn("{\"" + fieldname + "\":");
mv.visitMethodInsn(INVOKEVIRTUAL, "java/lang/String", "toCharArray", "()[C", false);
mv.visitFieldInsn(PUTFIELD, newDynName, fieldname + "FirstFieldChars", "[C");
}
mv.visitInsn(RETURN);
mv.visitMaxs(1 + members.size(), 1 + members.size());
@@ -614,13 +635,15 @@ public abstract class JsonDynEncoder<T> implements Encodeable<JsonWriter, T> {
elementIndex++;
AccessibleObject element = members.get(elementIndex);
ConvertColumnEntry ref1 = factory.findRef(clazz, element);
final String fieldname =
final String fieldName =
ref1 == null || ref1.name().isEmpty() ? readGetSetFieldName(element) : ref1.name();
final Class fieldtype = readGetSetFieldType(element);
mv.visitVarInsn(ALOAD, 1);
mv.visitVarInsn(ALOAD, 0);
mv.visitFieldInsn(GETFIELD, newDynName, fieldname + "FirstFieldBytes", "[B");
mv.visitFieldInsn(GETFIELD, newDynName, fieldName + "FirstFieldBytes", "[B");
mv.visitVarInsn(ALOAD, 0);
mv.visitFieldInsn(GETFIELD, newDynName, fieldName + "FirstFieldChars", "[C");
mv.visitVarInsn(ALOAD, 2); // String message = value.getMessage(); 加载 value
if (element instanceof Field) {
@@ -641,7 +664,7 @@ public abstract class JsonDynEncoder<T> implements Encodeable<JsonWriter, T> {
INVOKEVIRTUAL,
writerName,
"writeObjectByOnlyOneLatin1FieldValue",
"([BLjava/lang/String;)V",
"([B[CLjava/lang/String;)V",
false);
maxLocals++;
} else if (onlyTwoIntFieldObjectFlag) {
@@ -650,7 +673,7 @@ public abstract class JsonDynEncoder<T> implements Encodeable<JsonWriter, T> {
ConvertColumnEntry ref1 = factory.findRef(clazz, element1);
final String fieldName1 =
ref1 == null || ref1.name().isEmpty() ? readGetSetFieldName(element1) : ref1.name();
final Class fieldtype1 = readGetSetFieldType(element1);
final Class fieldType1 = readGetSetFieldType(element1);
elementIndex++;
AccessibleObject element2 = members.get(elementIndex);
@@ -663,6 +686,8 @@ public abstract class JsonDynEncoder<T> implements Encodeable<JsonWriter, T> {
mv.visitVarInsn(ALOAD, 0);
mv.visitFieldInsn(GETFIELD, newDynName, fieldName1 + "FirstFieldBytes", "[B");
mv.visitVarInsn(ALOAD, 0);
mv.visitFieldInsn(GETFIELD, newDynName, fieldName1 + "FirstFieldChars", "[C");
mv.visitVarInsn(ALOAD, 2); // String message = value.getMessage(); 加载 value
if (element1 instanceof Field) {
@@ -670,19 +695,21 @@ public abstract class JsonDynEncoder<T> implements Encodeable<JsonWriter, T> {
GETFIELD,
valtypeName,
((Field) element1).getName(),
org.redkale.asm.Type.getDescriptor(fieldtype1));
org.redkale.asm.Type.getDescriptor(fieldType1));
} else {
mv.visitMethodInsn(
INVOKEVIRTUAL,
valtypeName,
((Method) element1).getName(),
"()" + org.redkale.asm.Type.getDescriptor(fieldtype1),
"()" + org.redkale.asm.Type.getDescriptor(fieldType1),
false);
}
maxLocals++;
mv.visitVarInsn(ALOAD, 0);
mv.visitFieldInsn(GETFIELD, newDynName, fieldName2 + "CommaFieldBytes", "[B");
mv.visitVarInsn(ALOAD, 0);
mv.visitFieldInsn(GETFIELD, newDynName, fieldName2 + "CommaFieldChars", "[C");
mv.visitVarInsn(ALOAD, 2); // String message = value.getMessage(); 加载 value
if (element2 instanceof Field) {
@@ -701,7 +728,8 @@ public abstract class JsonDynEncoder<T> implements Encodeable<JsonWriter, T> {
}
maxLocals++;
mv.visitMethodInsn(INVOKEVIRTUAL, writerName, "writeObjectByOnlyTwoIntFieldValue", "([BI[BI)V", false);
mv.visitMethodInsn(
INVOKEVIRTUAL, writerName, "writeObjectByOnlyTwoIntFieldValue", "([B[CI[B[CI)V", false);
} else if (onlyShotIntLongLatin1MoreFieldObjectFlag && mustHadComma) {
for (AccessibleObject element : members) {
@@ -718,6 +746,12 @@ public abstract class JsonDynEncoder<T> implements Encodeable<JsonWriter, T> {
newDynName,
fieldname + (elementIndex == 0 ? "FirstFieldBytes" : "CommaFieldBytes"),
"[B");
mv.visitVarInsn(ALOAD, 0);
mv.visitFieldInsn(
GETFIELD,
newDynName,
fieldname + (elementIndex == 0 ? "FirstFieldChars" : "CommaFieldChars"),
"[C");
mv.visitVarInsn(ALOAD, 2); // String message = value.getMessage(); 加载 value
if (element instanceof Field) {
@@ -739,28 +773,28 @@ public abstract class JsonDynEncoder<T> implements Encodeable<JsonWriter, T> {
INVOKEVIRTUAL,
writerName,
elementIndex + 1 == membersSize ? "writeLastFieldShortValue" : "writeFieldShortValue",
"([BS)V",
"([B[CS)V",
false);
} else if (fieldtype == int.class) {
mv.visitMethodInsn(
INVOKEVIRTUAL,
writerName,
elementIndex + 1 == membersSize ? "writeLastFieldIntValue" : "writeFieldIntValue",
"([BI)V",
"([B[CI)V",
false);
} else if (fieldtype == long.class) {
mv.visitMethodInsn(
INVOKEVIRTUAL,
writerName,
elementIndex + 1 == membersSize ? "writeLastFieldLongValue" : "writeFieldLongValue",
"([BJ)V",
"([B[CJ)V",
false);
} else {
mv.visitMethodInsn(
INVOKEVIRTUAL,
writerName,
elementIndex + 1 == membersSize ? "writeLastFieldLatin1Value" : "writeFieldLatin1Value",
"([BLjava/lang/String;)V",
"([B[CLjava/lang/String;)V",
false);
}
@@ -864,13 +898,17 @@ public abstract class JsonDynEncoder<T> implements Encodeable<JsonWriter, T> {
mv.visitVarInsn(ALOAD, 1);
mv.visitVarInsn(ALOAD, 0);
mv.visitFieldInsn(GETFIELD, newDynName, fieldname + "FieldBytes", "[B");
mv.visitMethodInsn(INVOKEVIRTUAL, writerName, "writeTo", "([B)V", false);
mv.visitVarInsn(ALOAD, 0);
mv.visitFieldInsn(GETFIELD, newDynName, fieldname + "FieldChars", "[C");
mv.visitMethodInsn(INVOKEVIRTUAL, writerName, "writeField", "([B[C)V", false);
} else {
// out.writeTo(messageCommaFieldBytes);
mv.visitVarInsn(ALOAD, 1);
mv.visitVarInsn(ALOAD, 0);
mv.visitFieldInsn(GETFIELD, newDynName, fieldname + "CommaFieldBytes", "[B");
mv.visitMethodInsn(INVOKEVIRTUAL, writerName, "writeTo", "([B)V", false);
mv.visitVarInsn(ALOAD, 0);
mv.visitFieldInsn(GETFIELD, newDynName, fieldname + "CommaFieldChars", "[C");
mv.visitMethodInsn(INVOKEVIRTUAL, writerName, "writeField", "([B[C)V", false);
}
} else { // if(comma) {} else {} 代码块
// if (comma) { start
@@ -882,7 +920,9 @@ public abstract class JsonDynEncoder<T> implements Encodeable<JsonWriter, T> {
mv.visitVarInsn(ALOAD, 1);
mv.visitVarInsn(ALOAD, 0);
mv.visitFieldInsn(GETFIELD, newDynName, fieldname + "CommaFieldBytes", "[B");
mv.visitMethodInsn(INVOKEVIRTUAL, writerName, "writeTo", "([B)V", false);
mv.visitVarInsn(ALOAD, 0);
mv.visitFieldInsn(GETFIELD, newDynName, fieldname + "CommaFieldChars", "[C");
mv.visitMethodInsn(INVOKEVIRTUAL, writerName, "writeField", "([B[C)V", false);
Label commaelse = new Label();
mv.visitJumpInsn(GOTO, commaelse);
@@ -915,7 +955,9 @@ public abstract class JsonDynEncoder<T> implements Encodeable<JsonWriter, T> {
mv.visitVarInsn(ALOAD, 1);
mv.visitVarInsn(ALOAD, 0);
mv.visitFieldInsn(GETFIELD, newDynName, fieldname + "FieldBytes", "[B");
mv.visitMethodInsn(INVOKEVIRTUAL, writerName, "writeTo", "([B)V", false);
mv.visitVarInsn(ALOAD, 0);
mv.visitFieldInsn(GETFIELD, newDynName, fieldname + "FieldChars", "[C");
mv.visitMethodInsn(INVOKEVIRTUAL, writerName, "writeField", "([B[C)V", false);
// comma = true;
mv.visitInsn(ICONST_1);
mv.visitVarInsn(ISTORE, 3);

View File

@@ -52,6 +52,10 @@ public abstract class JsonWriter extends Writer {
return this.objExtFunc == null && this.objFieldFunc == null;
}
public boolean charsMode() {
return false;
}
// -----------------------------------------------------------------------
public abstract void writeTo(final char ch); // 只能是 0 - 127 的字符
@@ -71,37 +75,52 @@ public abstract class JsonWriter extends Writer {
public abstract void writeLatin1To(final boolean quote, final String value);
@ClassDepends
public abstract void writeFieldShortValue(final byte[] fieldBytes, final short value);
public void writeField(final byte[] fieldBytes, final char[] fieldChars) {
if (charsMode()) {
writeTo(fieldChars, 0, fieldChars.length);
} else {
writeTo(fieldBytes, 0, fieldBytes.length);
}
}
@ClassDepends
public abstract void writeFieldIntValue(final byte[] fieldBytes, final int value);
public abstract void writeFieldShortValue(final byte[] fieldBytes, final char[] fieldChars, final short value);
@ClassDepends
public abstract void writeFieldLongValue(final byte[] fieldBytes, final long value);
public abstract void writeFieldIntValue(final byte[] fieldBytes, final char[] fieldChars, final int value);
@ClassDepends
public abstract void writeFieldLatin1Value(final byte[] fieldBytes, final String value);
public abstract void writeFieldLongValue(final byte[] fieldBytes, final char[] fieldChars, final long value);
@ClassDepends
public abstract void writeLastFieldShortValue(final byte[] fieldBytes, final short value);
public abstract void writeFieldLatin1Value(final byte[] fieldBytes, final char[] fieldChars, final String value);
@ClassDepends
public abstract void writeLastFieldIntValue(final byte[] fieldBytes, final int value);
public abstract void writeLastFieldShortValue(final byte[] fieldBytes, final char[] fieldChars, final short value);
@ClassDepends
public abstract void writeLastFieldLongValue(final byte[] fieldBytes, final long value);
public abstract void writeLastFieldIntValue(final byte[] fieldBytes, final char[] fieldChars, final int value);
@ClassDepends
public abstract void writeLastFieldLatin1Value(final byte[] fieldBytes, final String value);
public abstract void writeLastFieldLongValue(final byte[] fieldBytes, final char[] fieldChars, final long value);
@ClassDepends
public abstract void writeLastFieldLatin1Value(final byte[] fieldBytes, final char[] fieldChars, String value);
// firstFieldBytes 必须带{开头
@ClassDepends
public abstract void writeObjectByOnlyOneLatin1FieldValue(final byte[] firstFieldBytes, final String value);
public abstract void writeObjectByOnlyOneLatin1FieldValue(
final byte[] firstFieldBytes, final char[] firstFieldChars, final String value);
// firstFieldBytes 必须带{开头, lastFieldBytes必须,开头
@ClassDepends
public abstract void writeObjectByOnlyTwoIntFieldValue(
final byte[] firstFieldBytes, final int value1, final byte[] lastFieldBytes, final int value2);
final byte[] firstFieldBytes,
final char[] firstFieldChars,
final int value1,
final byte[] lastFieldBytes,
final char[] lastFieldChars,
final int value2);
@Override
@ClassDepends
@@ -127,7 +146,12 @@ public abstract class JsonWriter extends Writer {
writeTo(BYTE_COMMA);
}
if (member != null) {
writeTo(member.getJsonFieldNameBytes());
if (charsMode()) {
char[] chs = member.getJsonFieldNameChars();
writeTo(chs, 0, chs.length);
} else {
writeTo(member.getJsonFieldNameBytes());
}
} else {
writeLatin1To(true, fieldName);
writeTo(BYTE_COLON);

View File

@@ -35,7 +35,7 @@ public class StringWrapperTest {
{
String emoji =
new String(new byte[] {(byte) 0xF0, (byte) 0x9F, (byte) 0x98, (byte) 0x81}, StandardCharsets.UTF_8);
String val = "{id:'带中文" + emoji + "'}";
String val = "{id:'z带中文" + emoji + "a'}";
StringWrapper wrapper = new StringWrapper(val);
if (!main) Assertions.assertEquals(val, convert.convertTo(wrapper));
if (!main)