/* * Copyright (c) 2000, 2013, 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 test.java.lang.String; /** * @test * @bug 6840246 6559590 * @summary test String.split() * @key randomness */ import java.util.Arrays; import java.util.Random; import java.util.regex.*; import org.testng.annotations.Test; import org.testng.annotations.BeforeClass; // Android-changed: migrate to testng public class Split { private static final String SOURCE = "0123456789"; private static final String FAST_PATH_SOURCE = "0123456789abcdefgABCDEFG"; private static Random R; @BeforeClass public static void setUp() { R = new Random(); } @Test public void testSplit_byDigit() { for (int limit=-2; limit<3; limit++) { for (int x=0; x<10; x++) { String[] result = SOURCE.split(Integer.toString(x), limit); int expectedLength = limit < 1 ? 2 : limit; if ((limit == 0) && (x == 9)) { // expected dropping of "" if (result.length != 1) throw new RuntimeException("String.split failure 1"); if (!result[0].equals("012345678")) { throw new RuntimeException("String.split failure 2"); } } else { if (result.length != expectedLength) { throw new RuntimeException("String.split failure 3"); } if (!result[0].equals(SOURCE.substring(0,x))) { if (limit != 1) { throw new RuntimeException( "String.split failure 4"); } else { if (!result[0].equals(SOURCE.substring(0,10))) { throw new RuntimeException( "String.split failure 10"); } } } if (expectedLength > 1) { // Check segment 2 if (!result[1].equals(SOURCE.substring(x+1,10))) throw new RuntimeException("String.split failure 5"); } } } } } @Test public void testSplit_noMatch() { for (int limit=-2; limit<3; limit++) { String[] result = SOURCE.split("e", limit); if (result.length != 1) throw new RuntimeException("String.split failure 6"); if (!result[0].equals(SOURCE)) throw new RuntimeException("String.split failure 7"); } } // Check the case for limit == 0, source = ""; // split() now returns 0-length for empty source "" see #6559590 @Test public void testSplit_zeroLimit_emptySource() { String source = ""; String[] result = source.split("e", 0); if (result.length != 1) throw new RuntimeException("String.split failure 8"); if (!result[0].equals(source)) throw new RuntimeException("String.split failure 9"); } /** * Tests fastpath of {@link java.lang.String#split(String, int)}. * * This test is disabled and just provides explanation of testSplitFastPath_shard* * test functions family. Originally, this test was running around 50 seconds and * was doing the following: * *
* for (boolean doEscape: new boolean[] {false, true}) {
* for (int cp = 0; cp < 0x11000; cp++) {
* ...
* }
* }
*
*
* To make it faster it was sharded into five shards:
*
* 1) First shard tests {@code doEscape = false} and {@code cp = [0, 0x11000)}, i.e. full range.
* It was accounted for ~10% of all time (~5s), so the whole range goes in single shard.
* 2-5) These four shards test {@code doEscape = true} and {@code cp} split in four equal
* intervals.
*
* There is also {@link #getShardRange(int)} helper function which returns shard range.
*/
@Test(enabled = false)
public void testSplit_fastPath() {
}
@Test
public void testSplit_fastPath_shard1_noEscape() {
testSplit_fastPath_(FAST_PATH_SOURCE, R, false, 0, 0x11000);
}
@Test
public void testSplit_fastPath_shard2() {
int[] shardRange = getShardRange(0);
testSplit_fastPath_(FAST_PATH_SOURCE, R, true, shardRange[0], shardRange[1]);
}
@Test
public void testSplit_fastPath_shard3() {
int[] shardRange = getShardRange(1);
testSplit_fastPath_(FAST_PATH_SOURCE, R, true, shardRange[0], shardRange[1]);
}
@Test
public void testSplit_fastPath_shard4() {
int[] shardRange = getShardRange(2);
testSplit_fastPath_(FAST_PATH_SOURCE, R, true, shardRange[0], shardRange[1]);
}
@Test
public void testSplit_fastPath_shard5() {
int[] shardRange = getShardRange(3);
testSplit_fastPath_(FAST_PATH_SOURCE, R, true, shardRange[0], shardRange[1]);
}
/**
* Calculates shard range, i.e. [start, end) by shard index,
* assuming there are 4 shards in total; and the whole range spans from 0 to 0x11000.
*
* @param shardIndex index of the
* @return int[] of size 2: [start, end).
*/
private static int[] getShardRange(int shardIndex) {
final int SHARD_LENGTH = 0x11000;
final int SHARD_COUNT = 4;
int shardLength = SHARD_LENGTH / SHARD_COUNT;
int shardStart = shardIndex * shardLength;
int shardEnd = (shardIndex + 1) * shardLength - 1;
return new int[]{ shardStart, shardEnd };
}
private void testSplit_fastPath_(String source, Random r, boolean doEscape, int cpFrom, int cpTo) {
for (int cp = cpFrom; cp < cpTo; cp++) {
Pattern p = null;
String regex = new String(Character.toChars(cp));
if (doEscape)
regex = "\\" + regex;
try {
p = Pattern.compile(regex);
} catch (PatternSyntaxException pse) {
// illegal syntax
try {
"abc".split(regex);
} catch (PatternSyntaxException pse0) {
continue;
}
throw new RuntimeException("String.split failure 11");
}
int off = r.nextInt(source.length());
String[] srcStrs = new String[] {
"",
source,
regex + source,
source + regex,
source.substring(0, 3)
+ regex + source.substring(3, 9)
+ regex + source.substring(9, 15)
+ regex + source.substring(15),
source.substring(0, off) + regex + source.substring(off)
};
for (String src: srcStrs) {
for (int limit=-2; limit<3; limit++) {
if (!Arrays.equals(src.split(regex, limit),
p.split(src, limit)))
throw new RuntimeException("String.split failure 12");
}
}
}
}
}