c24c66db97
Reviewed-by: dfuchs, rriggs
208 lines
7.4 KiB
Java
208 lines
7.4 KiB
Java
/*
|
|
* Copyright (c) 2014, Oracle and/or its affiliates. All rights reserved.
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
*
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License version 2 only, as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
* accompanied this code).
|
|
*
|
|
* You should have received a copy of the GNU General Public License version
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
* questions.
|
|
*/
|
|
package org.openjdk.bench.java.net;
|
|
|
|
import org.openjdk.jmh.annotations.Benchmark;
|
|
import org.openjdk.jmh.annotations.BenchmarkMode;
|
|
import org.openjdk.jmh.annotations.Fork;
|
|
import org.openjdk.jmh.annotations.Measurement;
|
|
import org.openjdk.jmh.annotations.Mode;
|
|
import org.openjdk.jmh.annotations.OutputTimeUnit;
|
|
import org.openjdk.jmh.annotations.Param;
|
|
import org.openjdk.jmh.annotations.Scope;
|
|
import org.openjdk.jmh.annotations.Setup;
|
|
import org.openjdk.jmh.annotations.State;
|
|
import org.openjdk.jmh.annotations.Warmup;
|
|
import org.openjdk.jmh.infra.Blackhole;
|
|
|
|
import java.io.UnsupportedEncodingException;
|
|
import java.net.URLDecoder;
|
|
import java.nio.charset.StandardCharsets;
|
|
import java.util.Random;
|
|
import java.util.concurrent.TimeUnit;
|
|
|
|
/**
|
|
* Tests java.net.URLEncoder.encode and Decoder.decode.
|
|
*/
|
|
@BenchmarkMode(Mode.AverageTime)
|
|
@OutputTimeUnit(TimeUnit.MILLISECONDS)
|
|
@State(Scope.Thread)
|
|
@Warmup(iterations = 5, time = 1)
|
|
@Measurement(iterations = 5, time = 1)
|
|
@Fork(value = 3)
|
|
public class URLEncodeDecode {
|
|
|
|
private static final int COUNT = 1024;
|
|
|
|
@Param("1024")
|
|
public int maxLength;
|
|
|
|
/**
|
|
* Percentage of strings that will remain unchanged by an encoding/decoding (0-100)
|
|
*/
|
|
@Param({"0", "75", "100"})
|
|
public int unchanged;
|
|
|
|
/**
|
|
* Percentage of chars in changed strings that cause encoding/decoding to happen (0-100)
|
|
*/
|
|
@Param({"6"})
|
|
public int encodeChars;
|
|
|
|
public String[] testStringsEncode;
|
|
public String[] testStringsDecode;
|
|
public String[] toStrings;
|
|
|
|
@Setup()
|
|
public void setupStrings() {
|
|
char[] encodeTokens = new char[] { '[', '(', ' ', '\u00E4', '\u00E5', '\u00F6', ')', '='};
|
|
char[] tokens = new char[('Z' - 'A' + 1) + ('z' - 'a' + 1) + ('9' - '0' + 1) + 4];
|
|
int n = 0;
|
|
for (char c = '0'; c <= '9'; c++) {
|
|
tokens[n++] = c;
|
|
}
|
|
for (char c = 'A'; c <= 'Z'; c++) {
|
|
tokens[n++] = c;
|
|
}
|
|
for (char c = 'a'; c <= 'z'; c++) {
|
|
tokens[n++] = c;
|
|
}
|
|
tokens[n++] = '-';
|
|
tokens[n++] = '_';
|
|
tokens[n++] = '.';
|
|
tokens[n] = '*';
|
|
|
|
Random r = new Random(3);
|
|
testStringsEncode = new String[COUNT];
|
|
testStringsDecode = new String[COUNT];
|
|
toStrings = new String[COUNT];
|
|
for (int i = 0; i < COUNT; i++) {
|
|
int l = r.nextInt(maxLength);
|
|
boolean needEncoding = r.nextInt(100) >= unchanged;
|
|
StringBuilder sb = new StringBuilder();
|
|
boolean hasEncoded = false;
|
|
for (int j = 0; j < l; j++) {
|
|
if (needEncoding && r.nextInt(100) < encodeChars) {
|
|
addToken(encodeTokens, r, sb);
|
|
hasEncoded = true;
|
|
} else {
|
|
addToken(tokens, r, sb);
|
|
}
|
|
}
|
|
if (needEncoding && !hasEncoded) {
|
|
addToken(encodeTokens, r, sb);
|
|
}
|
|
testStringsEncode[i] = sb.toString();
|
|
}
|
|
int countUnchanged = 0;
|
|
for (String s : testStringsEncode) {
|
|
if (s.equals(java.net.URLEncoder.encode(s, StandardCharsets.UTF_8))) {
|
|
countUnchanged++;
|
|
} else {
|
|
if (unchanged == 100) {
|
|
System.out.println("Unexpectedly needs encoding action: ");
|
|
System.out.println("\t" + s);
|
|
System.out.println("\t" + java.net.URLEncoder.encode(s, StandardCharsets.UTF_8));
|
|
}
|
|
}
|
|
}
|
|
System.out.println();
|
|
System.out.println("Generated " + testStringsEncode.length + " encodable strings, " + countUnchanged + " of which does not need encoding action");
|
|
|
|
for (int i = 0; i < COUNT; i++) {
|
|
int l = r.nextInt(maxLength);
|
|
boolean needDecoding = r.nextInt(100) >= unchanged;
|
|
StringBuilder sb = new StringBuilder();
|
|
boolean hasDecoded = false;
|
|
for (int j = 0; j < l; j++) {
|
|
if (needDecoding && r.nextInt(100) < encodeChars) {
|
|
addDecodableChar(tokens, r, sb);
|
|
hasDecoded = true;
|
|
} else {
|
|
addToken(tokens, r, sb);
|
|
}
|
|
}
|
|
if (needDecoding && !hasDecoded) {
|
|
addDecodableChar(tokens, r, sb);
|
|
}
|
|
testStringsDecode[i] = sb.toString();
|
|
}
|
|
countUnchanged = 0;
|
|
for (String s : testStringsDecode) {
|
|
if (s.equals(java.net.URLDecoder.decode(s, StandardCharsets.UTF_8))) {
|
|
countUnchanged++;
|
|
} else {
|
|
if (unchanged == 100) {
|
|
System.out.println("Unexpectedly needs encoding action: ");
|
|
System.out.println("\t" + s);
|
|
System.out.println("\t" + java.net.URLDecoder.decode(s, StandardCharsets.UTF_8));
|
|
}
|
|
}
|
|
}
|
|
System.out.println("Generated " + testStringsDecode.length + " decodable strings, " + countUnchanged + " of which does not need decoding action");
|
|
}
|
|
|
|
private static void addToken(char[] tokens, Random r, StringBuilder sb) {
|
|
int c = r.nextInt(tokens.length);
|
|
sb.append(tokens[c]);
|
|
}
|
|
|
|
private static void addDecodableChar(char[] tokens, Random r, StringBuilder sb) {
|
|
if (r.nextInt(100) < 15) {
|
|
sb.append('+'); // exercise '+' -> ' ' decoding paths.
|
|
} else {
|
|
sb.append("%").append(tokens[r.nextInt(16)]).append(tokens[r.nextInt(16)]);
|
|
}
|
|
}
|
|
|
|
@Benchmark
|
|
public void testEncodeUTF8(Blackhole bh) throws UnsupportedEncodingException {
|
|
for (String s : testStringsEncode) {
|
|
bh.consume(java.net.URLEncoder.encode(s, StandardCharsets.UTF_8));
|
|
}
|
|
}
|
|
|
|
@Benchmark
|
|
public void testDecodeUTF8(Blackhole bh) throws UnsupportedEncodingException {
|
|
for (String s : testStringsDecode) {
|
|
bh.consume(URLDecoder.decode(s, StandardCharsets.UTF_8));
|
|
}
|
|
}
|
|
|
|
|
|
@Benchmark
|
|
public void testEncodeLatin1(Blackhole bh) throws UnsupportedEncodingException {
|
|
for (String s : testStringsEncode) {
|
|
bh.consume(java.net.URLEncoder.encode(s, StandardCharsets.ISO_8859_1));
|
|
}
|
|
}
|
|
|
|
@Benchmark
|
|
public void testDecodeLatin1(Blackhole bh) throws UnsupportedEncodingException {
|
|
for (String s : testStringsDecode) {
|
|
bh.consume(URLDecoder.decode(s, StandardCharsets.ISO_8859_1));
|
|
}
|
|
}
|
|
|
|
}
|