2021-06-07 07:01:30 +00:00
|
|
|
/*
|
2022-05-16 07:49:26 +00:00
|
|
|
* Copyright (c) 2017, 2022, Oracle and/or its affiliates. All rights reserved.
|
2021-06-07 07:01:30 +00:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @test
|
2021-06-15 09:04:35 +00:00
|
|
|
* @bug 8262891 8268663
|
2021-06-07 07:01:30 +00:00
|
|
|
* @summary Check guards implementation.
|
|
|
|
* @compile --enable-preview -source ${jdk.version} Guards.java
|
|
|
|
* @run main/othervm --enable-preview Guards
|
|
|
|
*/
|
|
|
|
|
|
|
|
import java.util.Objects;
|
|
|
|
import java.util.function.Function;
|
|
|
|
|
|
|
|
public class Guards {
|
|
|
|
public static void main(String... args) {
|
|
|
|
new Guards().run();
|
|
|
|
}
|
|
|
|
|
|
|
|
void run() {
|
|
|
|
run(this::typeTestPatternSwitchTest);
|
|
|
|
run(this::typeTestPatternSwitchExpressionTest);
|
|
|
|
run(this::testBooleanSwitchExpression);
|
|
|
|
assertEquals("a", testPatternInGuard("a"));
|
|
|
|
assertEquals(null, testPatternInGuard(1));
|
2021-06-15 09:04:35 +00:00
|
|
|
runIfTrue(this::typeGuardIfTrueIfStatement);
|
|
|
|
runIfTrue(this::typeGuardIfTrueSwitchExpression);
|
|
|
|
runIfTrue(this::typeGuardIfTrueSwitchStatement);
|
2021-06-23 10:16:42 +00:00
|
|
|
runIfTrue(this::typeGuardAfterParenthesizedTrueSwitchStatement);
|
|
|
|
runIfTrue(this::typeGuardAfterParenthesizedTrueSwitchExpression);
|
|
|
|
runIfTrue(this::typeGuardAfterParenthesizedTrueIfStatement);
|
2022-05-16 07:49:26 +00:00
|
|
|
testGuardNPE();
|
2021-06-07 07:01:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void run(Function<Object, String> convert) {
|
|
|
|
assertEquals("zero", convert.apply(0));
|
|
|
|
assertEquals("one", convert.apply(1));
|
|
|
|
assertEquals("other", convert.apply(-1));
|
|
|
|
assertEquals("any", convert.apply(""));
|
|
|
|
}
|
|
|
|
|
2021-06-15 09:04:35 +00:00
|
|
|
void runIfTrue(Function<Object, String> convert) {
|
|
|
|
assertEquals("true", convert.apply(0));
|
|
|
|
assertEquals("second", convert.apply(2));
|
|
|
|
assertEquals("any", convert.apply(""));
|
|
|
|
}
|
|
|
|
|
2021-06-07 07:01:30 +00:00
|
|
|
String typeTestPatternSwitchTest(Object o) {
|
|
|
|
switch (o) {
|
2022-05-16 07:49:26 +00:00
|
|
|
case Integer i when i == 0: return "zero";
|
|
|
|
case Integer i when i == 1: return "one";
|
2021-06-07 07:01:30 +00:00
|
|
|
case Integer i: return "other";
|
|
|
|
case Object x: return "any";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
String typeTestPatternSwitchExpressionTest(Object o) {
|
|
|
|
return switch (o) {
|
2022-05-16 07:49:26 +00:00
|
|
|
case Integer i when i == 0 -> "zero";
|
|
|
|
case Integer i when i == 1 -> { yield "one"; }
|
2021-06-07 07:01:30 +00:00
|
|
|
case Integer i -> "other";
|
|
|
|
case Object x -> "any";
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
String testBooleanSwitchExpression(Object o) {
|
|
|
|
String x;
|
|
|
|
if (switch (o) {
|
2022-05-16 07:49:26 +00:00
|
|
|
case Integer i when i == 0 -> (x = "zero") != null;
|
|
|
|
case Integer i when i == 1 -> { x = "one"; yield true; }
|
2021-06-07 07:01:30 +00:00
|
|
|
case Integer i -> { x = "other"; yield true; }
|
|
|
|
case Object other -> (x = "any") != null;
|
|
|
|
}) {
|
|
|
|
return x;
|
|
|
|
} else {
|
|
|
|
throw new IllegalStateException("TODO - needed?");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-15 09:04:35 +00:00
|
|
|
String typeGuardIfTrueSwitchStatement(Object o) {
|
|
|
|
Object o2 = "";
|
|
|
|
switch (o) {
|
2022-05-16 07:49:26 +00:00
|
|
|
case Integer i when i == 0 && i < 1 && o2 instanceof String s: o = s + String.valueOf(i); return "true";
|
|
|
|
case Integer i when i == 0 || i > 1: o = String.valueOf(i); return "second";
|
2021-06-15 09:04:35 +00:00
|
|
|
case Object x: return "any";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
String typeGuardIfTrueSwitchExpression(Object o) {
|
|
|
|
Object o2 = "";
|
|
|
|
return switch (o) {
|
2022-05-16 07:49:26 +00:00
|
|
|
case Integer i when i == 0 && i < 1 && o2 instanceof String s: o = s + String.valueOf(i); yield "true";
|
|
|
|
case Integer i when i == 0 || i > 1: o = String.valueOf(i); yield "second";
|
2021-06-15 09:04:35 +00:00
|
|
|
case Object x: yield "any";
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
String typeGuardIfTrueIfStatement(Object o) {
|
|
|
|
Object o2 = "";
|
2022-05-16 07:49:26 +00:00
|
|
|
if (o != null && o instanceof Integer i && i == 0 && i < 1 && (o = i) != null && o2 instanceof String s) {
|
2021-06-15 09:04:35 +00:00
|
|
|
return s != null ? "true" : null;
|
2022-05-16 07:49:26 +00:00
|
|
|
} else if (o != null && o instanceof Integer i && (i == 0 || i > 1) && (o = i) != null) {
|
2021-06-15 09:04:35 +00:00
|
|
|
return "second";
|
|
|
|
} else {
|
|
|
|
return "any";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-23 10:16:42 +00:00
|
|
|
String typeGuardAfterParenthesizedTrueSwitchStatement(Object o) {
|
|
|
|
switch (o) {
|
2022-05-16 07:49:26 +00:00
|
|
|
case (Integer i) when i == 0: o = String.valueOf(i); return "true";
|
|
|
|
case (Integer i) when i == 2: o = String.valueOf(i); return "second";
|
2021-06-23 10:16:42 +00:00
|
|
|
case Object x: return "any";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
String typeGuardAfterParenthesizedTrueSwitchExpression(Object o) {
|
|
|
|
return switch (o) {
|
2022-05-16 07:49:26 +00:00
|
|
|
case (Integer i) when i == 0: o = String.valueOf(i); yield "true";
|
|
|
|
case (Integer i) when i == 2: o = String.valueOf(i); yield "second";
|
2021-06-23 10:16:42 +00:00
|
|
|
case Object x: yield "any";
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
String typeGuardAfterParenthesizedTrueIfStatement(Object o) {
|
2022-05-16 07:49:26 +00:00
|
|
|
if (o != null && o instanceof (Integer i) && i == 0) {
|
2021-06-23 10:16:42 +00:00
|
|
|
return "true";
|
2022-05-16 07:49:26 +00:00
|
|
|
} else if (o != null && o instanceof (Integer i) && i == 2 && (o = i) != null) {
|
2021-06-23 10:16:42 +00:00
|
|
|
return "second";
|
|
|
|
} else {
|
|
|
|
return "any";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-07 07:01:30 +00:00
|
|
|
String testPatternInGuard(Object o) {
|
2022-05-16 07:49:26 +00:00
|
|
|
if (o instanceof CharSequence cs && cs instanceof String s) {
|
2021-06-07 07:01:30 +00:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2022-05-16 07:49:26 +00:00
|
|
|
void testGuardNPE() {
|
|
|
|
assertEquals("empty", guardNPE(""));
|
|
|
|
assertEquals("A", guardNPE("A"));
|
|
|
|
assertEquals("other", guardNPE(1));
|
|
|
|
try {
|
|
|
|
guardNPE(null);
|
|
|
|
throw new AssertionError("Expected exception missing.");
|
|
|
|
} catch (NullPointerException ex) {
|
|
|
|
//expected
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
String guardNPE(Object o) {
|
|
|
|
return switch (o) {
|
|
|
|
case null, String s when s.isEmpty() -> "empty";
|
|
|
|
case String s -> s;
|
|
|
|
case Object x -> "other";
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2021-06-07 07:01:30 +00:00
|
|
|
void assertEquals(String expected, String actual) {
|
|
|
|
if (!Objects.equals(expected, actual)) {
|
|
|
|
throw new AssertionError("Expected: " + expected + ", but got: " + actual);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|