From 2c30505365488efa1305e3e6193290acb8fc4a40 Mon Sep 17 00:00:00 2001 From: Simon Binder Date: Sat, 23 Sep 2023 12:50:30 +0200 Subject: [PATCH] Optimize json encoding and decoding (#130) --- lib/src/binary_codec.dart | 16 ++++++++++------ 1 file changed, 10 insertions(+), 6 deletions(-) diff --git a/lib/src/binary_codec.dart b/lib/src/binary_codec.dart index 0f610c27..c511968e 100644 --- a/lib/src/binary_codec.dart +++ b/lib/src/binary_codec.dart @@ -31,6 +31,10 @@ final _numericRegExp = RegExp(r'^(\d*)(\.\d*)?$'); final _leadingZerosRegExp = RegExp('^0+'); final _trailingZerosRegExp = RegExp(r'0+$'); +// The Dart SDK provides an optimized implementation for JSON from and to UTF-8 +// that doesn't allocate intermediate strings. +final _jsonUtf8 = json.fuse(utf8); + class PostgresBinaryEncoder extends Converter { final PgDataType _dataType; @@ -181,7 +185,7 @@ class PostgresBinaryEncoder case PgDataType.jsonb: { - final jsonBytes = utf8.encode(json.encode(input)); + final jsonBytes = _jsonUtf8.encode(input); final writer = ByteDataWriter(bufferLength: jsonBytes.length + 1); writer.writeUint8(1); writer.write(jsonBytes); @@ -189,7 +193,7 @@ class PostgresBinaryEncoder } case PgDataType.json: - return castBytes(utf8.encode(json.encode(input))); + return castBytes(_jsonUtf8.encode(input)); case PgDataType.byteArray: { @@ -324,7 +328,7 @@ class PostgresBinaryEncoder case PgDataType.jsonbArray: { if (input is List) { - final objectsArray = input.map((v) => utf8.encode(json.encode(v))); + final objectsArray = input.map(_jsonUtf8.encode); return writeListBytes>( objectsArray, 3802, (item) => item.length + 1, (writer, item) { writer.writeUint8(1); @@ -493,11 +497,11 @@ class PostgresBinaryDecoder extends Converter { // Removes version which is first character and currently always '1' final bytes = input.buffer .asUint8List(input.offsetInBytes + 1, input.lengthInBytes - 1); - return json.decode(utf8.decode(bytes)) as T; + return _jsonUtf8.decode(bytes) as T; } case PostgreSQLDataType.json: - return json.decode(utf8.decode(input)) as T; + return _jsonUtf8.decode(input) as T; case PostgreSQLDataType.byteArray: return input as T; @@ -556,7 +560,7 @@ class PostgresBinaryDecoder extends Converter { return readListBytes(input, (reader, length) { reader.read(1); final bytes = reader.read(length - 1); - return json.decode(utf8.decode(bytes)); + return _jsonUtf8.decode(bytes); }) as T; case PostgreSQLDataType.unknownType: