69a46c25cc
Reviewed-by: mcimadamore
441 lines
19 KiB
Java
441 lines
19 KiB
Java
/*
|
|
* Copyright (c) 2020, 2023, 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.
|
|
*
|
|
*/
|
|
|
|
/*
|
|
* @test
|
|
* @enablePreview
|
|
* @run testng/othervm -Djava.lang.invoke.VarHandle.VAR_HANDLE_GUARDS=true -Djava.lang.invoke.VarHandle.VAR_HANDLE_IDENTITY_ADAPT=false -Xverify:all TestAdaptVarHandles
|
|
* @run testng/othervm -Djava.lang.invoke.VarHandle.VAR_HANDLE_GUARDS=true -Djava.lang.invoke.VarHandle.VAR_HANDLE_IDENTITY_ADAPT=true -Xverify:all TestAdaptVarHandles
|
|
* @run testng/othervm -Djava.lang.invoke.VarHandle.VAR_HANDLE_GUARDS=false -Djava.lang.invoke.VarHandle.VAR_HANDLE_IDENTITY_ADAPT=false -Xverify:all TestAdaptVarHandles
|
|
* @run testng/othervm -Djava.lang.invoke.VarHandle.VAR_HANDLE_GUARDS=false -Djava.lang.invoke.VarHandle.VAR_HANDLE_IDENTITY_ADAPT=true -Xverify:all TestAdaptVarHandles
|
|
*/
|
|
|
|
import java.lang.foreign.*;
|
|
|
|
import org.testng.annotations.*;
|
|
import static org.testng.Assert.*;
|
|
|
|
import java.lang.invoke.MethodHandle;
|
|
import java.lang.invoke.MethodHandles;
|
|
import java.lang.invoke.MethodType;
|
|
import java.lang.invoke.VarHandle;
|
|
import java.util.List;
|
|
|
|
public class TestAdaptVarHandles {
|
|
|
|
static MethodHandle S2I;
|
|
static MethodHandle I2S;
|
|
static MethodHandle CTX_I2S;
|
|
static MethodHandle O2I;
|
|
static MethodHandle I2O;
|
|
static MethodHandle S2L;
|
|
static MethodHandle S2L_EX;
|
|
static MethodHandle S2I_EX;
|
|
static MethodHandle I2S_EX;
|
|
static MethodHandle BASE_ADDR;
|
|
static MethodHandle SUM_OFFSETS;
|
|
static MethodHandle VOID_FILTER;
|
|
|
|
static {
|
|
try {
|
|
S2I = MethodHandles.lookup().findStatic(TestAdaptVarHandles.class, "stringToInt", MethodType.methodType(int.class, String.class));
|
|
I2S = MethodHandles.lookup().findStatic(TestAdaptVarHandles.class, "intToString", MethodType.methodType(String.class, int.class));
|
|
CTX_I2S = MethodHandles.lookup().findStatic(TestAdaptVarHandles.class, "ctxIntToString",
|
|
MethodType.methodType(String.class, String.class, String.class, int.class));
|
|
O2I = MethodHandles.explicitCastArguments(S2I, MethodType.methodType(int.class, Object.class));
|
|
I2O = MethodHandles.explicitCastArguments(I2S, MethodType.methodType(Object.class, int.class));
|
|
S2L = MethodHandles.lookup().findStatic(TestAdaptVarHandles.class, "stringToLong", MethodType.methodType(long.class, String.class));
|
|
S2L_EX = MethodHandles.lookup().findStatic(TestAdaptVarHandles.class, "stringToLongException", MethodType.methodType(long.class, String.class));
|
|
BASE_ADDR = MethodHandles.lookup().findStatic(TestAdaptVarHandles.class, "baseAddress", MethodType.methodType(MemorySegment.class, MemorySegment.class));
|
|
SUM_OFFSETS = MethodHandles.lookup().findStatic(TestAdaptVarHandles.class, "sumOffsets", MethodType.methodType(long.class, long.class, long.class));
|
|
VOID_FILTER = MethodHandles.lookup().findStatic(TestAdaptVarHandles.class, "void_filter", MethodType.methodType(void.class, String.class));
|
|
|
|
MethodHandle s2i_ex = MethodHandles.throwException(int.class, Throwable.class);
|
|
s2i_ex = MethodHandles.insertArguments(s2i_ex, 0, new Throwable());
|
|
S2I_EX = MethodHandles.dropArguments(s2i_ex, 0, String.class);
|
|
|
|
MethodHandle i2s_ex = MethodHandles.throwException(String.class, Throwable.class);
|
|
i2s_ex = MethodHandles.insertArguments(i2s_ex, 0, new Throwable());
|
|
I2S_EX = MethodHandles.dropArguments(i2s_ex, 0, int.class);
|
|
} catch (Throwable ex) {
|
|
throw new ExceptionInInitializerError();
|
|
}
|
|
}
|
|
|
|
static final VarHandle intHandleIndexed = ValueLayout.JAVA_INT.arrayElementVarHandle();
|
|
|
|
static final VarHandle intHandle = ValueLayout.JAVA_INT.varHandle();
|
|
|
|
static final VarHandle floatHandle = ValueLayout.JAVA_FLOAT.varHandle();
|
|
|
|
@Test
|
|
public void testFilterValue() throws Throwable {
|
|
ValueLayout layout = ValueLayout.JAVA_INT;
|
|
Arena scope = Arena.ofAuto();
|
|
MemorySegment segment = scope.allocate(layout);
|
|
VarHandle intHandle = layout.varHandle();
|
|
VarHandle i2SHandle = MethodHandles.filterValue(intHandle, S2I, I2S);
|
|
i2SHandle.set(segment, "1");
|
|
String oldValue = (String)i2SHandle.getAndAdd(segment, "42");
|
|
assertEquals(oldValue, "1");
|
|
String value = (String)i2SHandle.get(segment);
|
|
assertEquals(value, "43");
|
|
boolean swapped = (boolean)i2SHandle.compareAndSet(segment, "43", "12");
|
|
assertTrue(swapped);
|
|
oldValue = (String)i2SHandle.compareAndExchange(segment, "12", "42");
|
|
assertEquals(oldValue, "12");
|
|
value = (String)i2SHandle.toMethodHandle(VarHandle.AccessMode.GET).invokeExact(segment);
|
|
assertEquals(value, "42");
|
|
}
|
|
|
|
@Test
|
|
public void testFilterValueComposite() throws Throwable {
|
|
ValueLayout layout = ValueLayout.JAVA_INT;
|
|
Arena scope = Arena.ofAuto();
|
|
MemorySegment segment = scope.allocate(layout);
|
|
VarHandle intHandle = layout.varHandle();
|
|
MethodHandle CTX_S2I = MethodHandles.dropArguments(S2I, 0, String.class, String.class);
|
|
VarHandle i2SHandle = MethodHandles.filterValue(intHandle, CTX_S2I, CTX_I2S);
|
|
i2SHandle = MethodHandles.insertCoordinates(i2SHandle, 1, "a", "b");
|
|
i2SHandle.set(segment, "1");
|
|
String oldValue = (String)i2SHandle.getAndAdd(segment, "42");
|
|
assertEquals(oldValue, "ab1");
|
|
String value = (String)i2SHandle.get(segment);
|
|
assertEquals(value, "ab43");
|
|
boolean swapped = (boolean)i2SHandle.compareAndSet(segment, "43", "12");
|
|
assertTrue(swapped);
|
|
oldValue = (String)i2SHandle.compareAndExchange(segment, "12", "42");
|
|
assertEquals(oldValue, "ab12");
|
|
value = (String)i2SHandle.toMethodHandle(VarHandle.AccessMode.GET).invokeExact(segment);
|
|
assertEquals(value, "ab42");
|
|
}
|
|
|
|
@Test
|
|
public void testFilterValueLoose() throws Throwable {
|
|
ValueLayout layout = ValueLayout.JAVA_INT;
|
|
Arena scope = Arena.ofAuto();
|
|
MemorySegment segment = scope.allocate(layout);
|
|
VarHandle intHandle = layout.varHandle();
|
|
VarHandle i2SHandle = MethodHandles.filterValue(intHandle, O2I, I2O);
|
|
i2SHandle.set(segment, "1");
|
|
String oldValue = (String)i2SHandle.getAndAdd(segment, "42");
|
|
assertEquals(oldValue, "1");
|
|
String value = (String)i2SHandle.get(segment);
|
|
assertEquals(value, "43");
|
|
boolean swapped = (boolean)i2SHandle.compareAndSet(segment, "43", "12");
|
|
assertTrue(swapped);
|
|
oldValue = (String)i2SHandle.compareAndExchange(segment, "12", "42");
|
|
assertEquals(oldValue, "12");
|
|
value = (String)(Object)i2SHandle.toMethodHandle(VarHandle.AccessMode.GET).invokeExact(segment);
|
|
assertEquals(value, "42");
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadFilterCarrier() {
|
|
MethodHandles.filterValue(floatHandle, S2I, I2S);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadFilterUnboxArity() {
|
|
VarHandle floatHandle = ValueLayout.JAVA_INT.varHandle();
|
|
MethodHandles.filterValue(floatHandle, S2I.bindTo(""), I2S);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadFilterBoxArity() {
|
|
VarHandle intHandle = ValueLayout.JAVA_INT.varHandle();
|
|
MethodHandles.filterValue(intHandle, S2I, I2S.bindTo(42));
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadFilterBoxPrefixCoordinates() {
|
|
VarHandle intHandle = ValueLayout.JAVA_INT.varHandle();
|
|
MethodHandles.filterValue(intHandle,
|
|
MethodHandles.dropArguments(S2I, 1, int.class),
|
|
MethodHandles.dropArguments(I2S, 1, long.class));
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadFilterBoxException() {
|
|
VarHandle intHandle = ValueLayout.JAVA_INT.varHandle();
|
|
MethodHandles.filterValue(intHandle, I2S, S2L_EX);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadFilterUnboxException() {
|
|
VarHandle intHandle = ValueLayout.JAVA_INT.varHandle();
|
|
MethodHandles.filterValue(intHandle, S2L_EX, I2S);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalStateException.class)
|
|
public void testBadFilterBoxHandleException() {
|
|
VarHandle intHandle = ValueLayout.JAVA_INT.varHandle();
|
|
VarHandle vh = MethodHandles.filterValue(intHandle, S2I, I2S_EX);
|
|
try (Arena arena = Arena.ofConfined()) {
|
|
MemorySegment seg = arena.allocate(ValueLayout.JAVA_INT);
|
|
vh.set(seg, "42");
|
|
String x = (String) vh.get(seg); // should throw
|
|
}
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalStateException.class)
|
|
public void testBadFilterUnboxHandleException() {
|
|
VarHandle intHandle = ValueLayout.JAVA_INT.varHandle();
|
|
VarHandle vh = MethodHandles.filterValue(intHandle, S2I_EX, I2S);
|
|
try (Arena arena = Arena.ofConfined()) {
|
|
MemorySegment seg = arena.allocate(ValueLayout.JAVA_INT);
|
|
vh.set(seg, "42"); // should throw
|
|
}
|
|
}
|
|
|
|
@Test
|
|
public void testFilterCoordinates() throws Throwable {
|
|
ValueLayout layout = ValueLayout.JAVA_INT;
|
|
Arena scope = Arena.ofAuto();
|
|
MemorySegment segment = scope.allocate(layout);
|
|
VarHandle intHandle_longIndex = MethodHandles.filterCoordinates(intHandleIndexed, 0, BASE_ADDR, S2L);
|
|
intHandle_longIndex.set(segment, "0", 1);
|
|
int oldValue = (int)intHandle_longIndex.getAndAdd(segment, "0", 42);
|
|
assertEquals(oldValue, 1);
|
|
int value = (int)intHandle_longIndex.get(segment, "0");
|
|
assertEquals(value, 43);
|
|
boolean swapped = (boolean)intHandle_longIndex.compareAndSet(segment, "0", 43, 12);
|
|
assertTrue(swapped);
|
|
oldValue = (int)intHandle_longIndex.compareAndExchange(segment, "0", 12, 42);
|
|
assertEquals(oldValue, 12);
|
|
value = (int)intHandle_longIndex.toMethodHandle(VarHandle.AccessMode.GET).invokeExact(segment, "0");
|
|
assertEquals(value, 42);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadFilterCoordinatesNegativePos() {
|
|
MethodHandles.filterCoordinates(intHandle, -1, SUM_OFFSETS);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadFilterCoordinatesPosTooBig() {
|
|
MethodHandles.filterCoordinates(intHandle, 1, SUM_OFFSETS);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadFilterCoordinatesWrongFilterType() {
|
|
MethodHandles.filterCoordinates(intHandleIndexed, 1, S2I);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadFilterCoordinatesWrongFilterException() {
|
|
MethodHandles.filterCoordinates(intHandleIndexed, 1, S2L_EX);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadFilterCoordinatesTooManyFilters() {
|
|
MethodHandles.filterCoordinates(intHandleIndexed, 1, S2L, S2L);
|
|
}
|
|
|
|
@Test
|
|
public void testInsertCoordinates() throws Throwable {
|
|
ValueLayout layout = ValueLayout.JAVA_INT;
|
|
Arena scope = Arena.ofAuto();
|
|
MemorySegment segment = scope.allocate(layout);
|
|
VarHandle intHandle_longIndex = MethodHandles.insertCoordinates(intHandleIndexed, 0, segment, 0L);
|
|
intHandle_longIndex.set(1);
|
|
int oldValue = (int)intHandle_longIndex.getAndAdd(42);
|
|
assertEquals(oldValue, 1);
|
|
int value = (int)intHandle_longIndex.get();
|
|
assertEquals(value, 43);
|
|
boolean swapped = (boolean)intHandle_longIndex.compareAndSet(43, 12);
|
|
assertTrue(swapped);
|
|
oldValue = (int)intHandle_longIndex.compareAndExchange(12, 42);
|
|
assertEquals(oldValue, 12);
|
|
value = (int)intHandle_longIndex.toMethodHandle(VarHandle.AccessMode.GET).invokeExact();
|
|
assertEquals(value, 42);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadInsertCoordinatesNegativePos() {
|
|
MethodHandles.insertCoordinates(intHandle, -1, 42);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadInsertCoordinatesPosTooBig() {
|
|
MethodHandles.insertCoordinates(intHandle, 1, 42);
|
|
}
|
|
|
|
@Test(expectedExceptions = ClassCastException.class)
|
|
public void testBadInsertCoordinatesWrongCoordinateType() {
|
|
MethodHandles.insertCoordinates(intHandleIndexed, 1, "Hello");
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadInsertCoordinatesTooManyValues() {
|
|
MethodHandles.insertCoordinates(intHandleIndexed, 1, 0L, 0L);
|
|
}
|
|
|
|
@Test
|
|
public void testPermuteCoordinates() throws Throwable {
|
|
ValueLayout layout = ValueLayout.JAVA_INT;
|
|
Arena scope = Arena.ofAuto();
|
|
MemorySegment segment = scope.allocate(layout);
|
|
VarHandle intHandle_swap = MethodHandles.permuteCoordinates(intHandleIndexed,
|
|
List.of(long.class, MemorySegment.class), 1, 0);
|
|
intHandle_swap.set(0L, segment, 1);
|
|
int oldValue = (int)intHandle_swap.getAndAdd(0L, segment, 42);
|
|
assertEquals(oldValue, 1);
|
|
int value = (int)intHandle_swap.get(0L, segment);
|
|
assertEquals(value, 43);
|
|
boolean swapped = (boolean)intHandle_swap.compareAndSet(0L, segment, 43, 12);
|
|
assertTrue(swapped);
|
|
oldValue = (int)intHandle_swap.compareAndExchange(0L, segment, 12, 42);
|
|
assertEquals(oldValue, 12);
|
|
value = (int)intHandle_swap.toMethodHandle(VarHandle.AccessMode.GET).invokeExact(0L, segment);
|
|
assertEquals(value, 42);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadPermuteCoordinatesTooManyCoordinates() {
|
|
MethodHandles.permuteCoordinates(intHandle, List.of(int.class, int.class), new int[2]);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadPermuteCoordinatesTooFewCoordinates() {
|
|
MethodHandles.permuteCoordinates(intHandle, List.of());
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadPermuteCoordinatesIndexTooBig() {
|
|
MethodHandles.permuteCoordinates(intHandle, List.of(int.class, int.class), 3);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadPermuteCoordinatesIndexTooSmall() {
|
|
MethodHandles.permuteCoordinates(intHandle, List.of(int.class, int.class), -1);
|
|
}
|
|
|
|
@Test
|
|
public void testCollectCoordinates() throws Throwable {
|
|
ValueLayout layout = ValueLayout.JAVA_INT;
|
|
Arena scope = Arena.ofAuto();
|
|
MemorySegment segment = scope.allocate(layout);
|
|
VarHandle intHandle_sum = MethodHandles.collectCoordinates(intHandleIndexed, 1, SUM_OFFSETS);
|
|
intHandle_sum.set(segment, -2L, 2L, 1);
|
|
int oldValue = (int)intHandle_sum.getAndAdd(segment, -2L, 2L, 42);
|
|
assertEquals(oldValue, 1);
|
|
int value = (int)intHandle_sum.get(segment, -2L, 2L);
|
|
assertEquals(value, 43);
|
|
boolean swapped = (boolean)intHandle_sum.compareAndSet(segment, -2L, 2L, 43, 12);
|
|
assertTrue(swapped);
|
|
oldValue = (int)intHandle_sum.compareAndExchange(segment, -2L, 2L, 12, 42);
|
|
assertEquals(oldValue, 12);
|
|
value = (int)intHandle_sum.toMethodHandle(VarHandle.AccessMode.GET).invokeExact(segment, -2L, 2L);
|
|
assertEquals(value, 42);
|
|
}
|
|
|
|
@Test
|
|
public void testCollectCoordinatesVoidFilterType() {
|
|
VarHandle handle = MethodHandles.collectCoordinates(intHandle, 0, VOID_FILTER);
|
|
assertEquals(handle.coordinateTypes(), List.of(String.class, MemorySegment.class));
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadCollectCoordinatesNegativePos() {
|
|
MethodHandles.collectCoordinates(intHandle, -1, SUM_OFFSETS);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadCollectCoordinatesPosTooBig() {
|
|
MethodHandles.collectCoordinates(intHandle, 1, SUM_OFFSETS);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadCollectCoordinatesWrongFilterType() {
|
|
MethodHandles.collectCoordinates(intHandle, 0, SUM_OFFSETS);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadCollectCoordinatesWrongFilterException() {
|
|
MethodHandles.collectCoordinates(intHandle, 0, S2L_EX);
|
|
}
|
|
|
|
@Test
|
|
public void testDropCoordinates() throws Throwable {
|
|
ValueLayout layout = ValueLayout.JAVA_INT;
|
|
Arena scope = Arena.ofAuto();
|
|
MemorySegment segment = scope.allocate(layout);
|
|
VarHandle intHandle_dummy = MethodHandles.dropCoordinates(intHandleIndexed, 1, float.class, String.class);
|
|
intHandle_dummy.set(segment, 1f, "hello", 0L, 1);
|
|
int oldValue = (int)intHandle_dummy.getAndAdd(segment, 1f, "hello", 0L, 42);
|
|
assertEquals(oldValue, 1);
|
|
int value = (int)intHandle_dummy.get(segment, 1f, "hello", 0L);
|
|
assertEquals(value, 43);
|
|
boolean swapped = (boolean)intHandle_dummy.compareAndSet(segment, 1f, "hello", 0L, 43, 12);
|
|
assertTrue(swapped);
|
|
oldValue = (int)intHandle_dummy.compareAndExchange(segment, 1f, "hello", 0L, 12, 42);
|
|
assertEquals(oldValue, 12);
|
|
value = (int)intHandle_dummy.toMethodHandle(VarHandle.AccessMode.GET).invokeExact(segment, 1f, "hello", 0L);
|
|
assertEquals(value, 42);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadDropCoordinatesNegativePos() {
|
|
MethodHandles.dropCoordinates(intHandle, -1);
|
|
}
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
public void testBadDropCoordinatesPosTooBig() {
|
|
MethodHandles.dropCoordinates(intHandle, 2);
|
|
}
|
|
|
|
//helper methods
|
|
|
|
static int stringToInt(String s) {
|
|
return Integer.valueOf(s);
|
|
}
|
|
|
|
static String intToString(int i) {
|
|
return String.valueOf(i);
|
|
}
|
|
|
|
static long stringToLong(String s) {
|
|
return Long.valueOf(s);
|
|
}
|
|
|
|
static long stringToLongException(String s) throws Throwable {
|
|
return Long.valueOf(s);
|
|
}
|
|
|
|
static MemorySegment baseAddress(MemorySegment segment) {
|
|
return segment;
|
|
}
|
|
|
|
static long sumOffsets(long l1, long l2) {
|
|
return l1 + l2;
|
|
}
|
|
|
|
static void void_filter(String s) { }
|
|
|
|
static String ctxIntToString(String a, String b, int i) {
|
|
return a + b + String.valueOf(i);
|
|
}
|
|
}
|