public class MethodHandles extends Object
Modifier and Type | Class and Description |
---|---|
static class |
MethodHandles.Lookup
A lookup object is a factory for creating method handles,
when the creation requires access checking.
|
Modifier and Type | Method and Description |
---|---|
static MethodHandle |
arrayElementGetter(Class<?> arrayClass)
Produces a method handle giving read access to elements of an array.
|
static MethodHandle |
arrayElementSetter(Class<?> arrayClass)
Produces a method handle giving write access to elements of an array.
|
static MethodHandle |
catchException(MethodHandle target,
Class<? extends Throwable> exType,
MethodHandle handler)
Makes a method handle which adapts a target method handle,
by running it inside an exception handler.
|
static MethodHandle |
collectArguments(MethodHandle target,
int pos,
MethodHandle filter)
Adapts a target method handle by pre-processing
a sub-sequence of its arguments with a filter (another method handle).
|
static MethodHandle |
constant(Class<?> type,
Object value)
Produces a method handle of the requested return type which returns the given
constant value every time it is invoked.
|
static MethodHandle |
countedLoop(MethodHandle iterations,
MethodHandle init,
MethodHandle body)
Constructs a loop that runs a given number of iterations.
|
static MethodHandle |
countedLoop(MethodHandle start,
MethodHandle end,
MethodHandle init,
MethodHandle body)
Constructs a loop that counts over a range of numbers.
|
static MethodHandle |
doWhileLoop(MethodHandle init,
MethodHandle body,
MethodHandle pred)
Constructs a
do-while loop from an initializer, a body, and a predicate. |
static MethodHandle |
dropArguments(MethodHandle target,
int pos,
Class<?>... valueTypes)
Produces a method handle which will discard some dummy arguments
before calling some other specified target method handle.
|
static MethodHandle |
dropArguments(MethodHandle target,
int pos,
List<Class<?>> valueTypes)
Produces a method handle which will discard some dummy arguments
before calling some other specified target method handle.
|
static MethodHandle |
exactInvoker(MethodType type)
Produces a special invoker method handle which can be used to
invoke any method handle of the given type, as if by
invokeExact . |
static MethodHandle |
explicitCastArguments(MethodHandle target,
MethodType newType)
Produces a method handle which adapts the type of the
given method handle to a new type by pairwise argument and return type conversion.
|
static MethodHandle |
filterArguments(MethodHandle target,
int pos,
MethodHandle... filters)
Adapts a target method handle by pre-processing
one or more of its arguments, each with its own unary filter function,
and then calling the target with each pre-processed argument
replaced by the result of its corresponding filter function.
|
static MethodHandle |
filterReturnValue(MethodHandle target,
MethodHandle filter)
Adapts a target method handle by post-processing
its return value (if any) with a filter (another method handle).
|
static MethodHandle |
foldArguments(MethodHandle target,
int pos,
MethodHandle combiner)
Adapts a target method handle by pre-processing some of its arguments, starting at a given position, and then
calling the target with the result of the pre-processing, inserted into the original sequence of arguments just
before the folded arguments.
|
static MethodHandle |
foldArguments(MethodHandle target,
MethodHandle combiner)
Adapts a target method handle by pre-processing
some of its arguments, and then calling the target with
the result of the pre-processing, inserted into the original
sequence of arguments.
|
static MethodHandle |
guardWithTest(MethodHandle test,
MethodHandle target,
MethodHandle fallback)
Makes a method handle which adapts a target method handle,
by guarding it with a test, a boolean-valued method handle.
|
static MethodHandle |
identity(Class<?> type)
Produces a method handle which returns its sole argument when invoked.
|
static MethodHandle |
insertArguments(MethodHandle target,
int pos,
Object... values)
Provides a target method handle with one or more bound arguments
in advance of the method handle's invocation.
|
static MethodHandle |
invoker(MethodType type)
Produces a special invoker method handle which can be used to
invoke any method handle compatible with the given type, as if by
invoke . |
static MethodHandle |
iteratedLoop(MethodHandle iterator,
MethodHandle init,
MethodHandle body)
Constructs a loop that ranges over the elements produced by an
Iterator<T> . |
static MethodHandles.Lookup |
lookup()
Returns a
lookup object with
full capabilities to emulate all supported bytecode behaviors of the caller. |
static MethodHandle |
loop(MethodHandle[]... clauses)
Constructs a method handle representing a loop with several loop variables that are updated and checked upon each
iteration.
|
static MethodHandle |
permuteArguments(MethodHandle target,
MethodType newType,
int... reorder)
Produces a method handle which adapts the calling sequence of the
given method handle to a new type, by reordering the arguments.
|
static MethodHandles.Lookup |
publicLookup()
Returns a
lookup object which is trusted minimally. |
static <T extends Member> |
reflectAs(Class<T> expected,
MethodHandle target)
Performs an unchecked "crack" of a
direct method handle.
|
static MethodHandle |
spreadInvoker(MethodType type,
int leadingArgCount)
Produces a method handle which will invoke any method handle of the
given
type , with a given number of trailing arguments replaced by
a single trailing Object[] array. |
static MethodHandle |
throwException(Class<?> returnType,
Class<? extends Throwable> exType)
Produces a method handle which will throw exceptions of the given
exType . |
static MethodHandle |
tryFinally(MethodHandle target,
MethodHandle cleanup)
Makes a method handle that adapts a
target method handle by wrapping it in a try-finally block. |
static MethodHandle |
whileLoop(MethodHandle init,
MethodHandle pred,
MethodHandle body)
Constructs a
while loop from an initializer, a body, and a predicate. |
public static MethodHandles.Lookup lookup()
lookup object
with
full capabilities to emulate all supported bytecode behaviors of the caller.
These capabilities include private access to the caller.
Factory methods on the lookup object can create
direct method handles
for any member that the caller has access to via bytecodes,
including protected and private fields and methods.
This lookup object is a capability which may be delegated to trusted agents.
Do not store it in place where untrusted code can access it.
This method is caller sensitive, which means that it may return different values to different callers.
For any given caller class C
, the lookup object returned by this call
has equivalent capabilities to any lookup object
supplied by the JVM to the bootstrap method of an
invokedynamic instruction
executing in the same caller class C
.
public static MethodHandles.Lookup publicLookup()
lookup object
which is trusted minimally.
It can only be used to create method handles to
publicly accessible fields and methods.
As a matter of pure convention, the lookup class
of this lookup object will be Object
.
Discussion:
The lookup class can be changed to any other class C
using an expression of the form
publicLookup().in(C.class)
.
Since all classes have equal access to public names,
such a change would confer no new access rights.
A public lookup object is always subject to
security manager checks.
Also, it cannot access
caller sensitive methods.
public static <T extends Member> T reflectAs(Class<T> expected, MethodHandle target)
Lookup.revealDirect
on the target to obtain its symbolic reference, and then called
MethodHandleInfo.reflectAs
to resolve the symbolic reference to a member.
If there is a security manager, its checkPermission
method
is called with a ReflectPermission("suppressAccessChecks")
permission.
T
- the desired type of the result, either Member
or a subtypetarget
- a direct method handle to crack into symbolic reference componentsexpected
- a class object representing the desired result type T
SecurityException
- if the caller is not privileged to call setAccessible
NullPointerException
- if either argument is null
IllegalArgumentException
- if the target is not a direct method handleClassCastException
- if the member is not of the expected typepublic static MethodHandle arrayElementGetter(Class<?> arrayClass) throws IllegalArgumentException
int
.arrayClass
- an array typeNullPointerException
- if the argument is nullIllegalArgumentException
- if arrayClass is not an array typepublic static MethodHandle arrayElementSetter(Class<?> arrayClass) throws IllegalArgumentException
arrayClass
- the class of an arrayNullPointerException
- if the argument is nullIllegalArgumentException
- if arrayClass is not an array typepublic static MethodHandle spreadInvoker(MethodType type, int leadingArgCount)
type
, with a given number of trailing arguments replaced by
a single trailing Object[]
array.
The resulting invoker will be a method handle with the following
arguments:
MethodHandle
target
leadingArgCount
)
Object[]
array containing trailing arguments
The invoker will invoke its target like a call to invoke
with
the indicated type
.
That is, if the target is exactly of the given type
, it will behave
like invokeExact
; otherwise it behave as if asType
is used to convert the target to the required type
.
The type of the returned invoker will not be the given type
, but rather
will have all parameters except the first leadingArgCount
replaced by a single array of type Object[]
, which will be
the final parameter.
Before invoking its target, the invoker will spread the final array, apply
reference casts as necessary, and unbox and widen primitive arguments.
If, when the invoker is called, the supplied array argument does
not have the correct number of elements, the invoker will throw
an IllegalArgumentException
instead of invoking the target.
This method is equivalent to the following code (though it may be more efficient):
This method throws no reflective or security exceptions.MethodHandle invoker = MethodHandles.invoker(type); int spreadArgCount = type.parameterCount() - leadingArgCount; invoker = invoker.asSpreader(Object[].class, spreadArgCount); return invoker;
type
- the desired target typeleadingArgCount
- number of fixed arguments, to be passed unchanged to the targetNullPointerException
- if type
is nullIllegalArgumentException
- if leadingArgCount
is not in
the range from 0 to type.parameterCount()
inclusive,
or if the resulting method handle's type would have
too many parameterspublic static MethodHandle exactInvoker(MethodType type)
invokeExact
.
The resulting invoker will have a type which is
exactly equal to the desired type, except that it will accept
an additional leading argument of type MethodHandle
.
This method is equivalent to the following code (though it may be more efficient):
publicLookup().findVirtual(MethodHandle.class, "invokeExact", type)
Discussion:
Invoker method handles can be useful when working with variable method handles
of unknown types.
For example, to emulate an invokeExact
call to a variable method
handle M
, extract its type T
,
look up the invoker method X
for T
,
and call the invoker method, as X.invoke(T, A...)
.
(It would not work to call X.invokeExact
, since the type T
is unknown.)
If spreading, collecting, or other argument transformations are required,
they can be applied once to the invoker X
and reused on many M
method handle values, as long as they are compatible with the type of X
.
(Note: The invoker method is not available via the Core Reflection API.
An attempt to call java.lang.reflect.Method.invoke
on the declared invokeExact
or invoke
method will raise an
UnsupportedOperationException
.)
This method throws no reflective or security exceptions.
type
- the desired target typeIllegalArgumentException
- if the resulting method handle's type would have
too many parameterspublic static MethodHandle invoker(MethodType type)
invoke
.
The resulting invoker will have a type which is
exactly equal to the desired type, except that it will accept
an additional leading argument of type MethodHandle
.
Before invoking its target, if the target differs from the expected type,
the invoker will apply reference casts as
necessary and box, unbox, or widen primitive values, as if by asType
.
Similarly, the return value will be converted as necessary.
If the target is a variable arity method handle,
the required arity conversion will be made, again as if by asType
.
This method is equivalent to the following code (though it may be more efficient):
publicLookup().findVirtual(MethodHandle.class, "invoke", type)
Discussion:
A general method type is one which
mentions only Object
arguments and return values.
An invoker for such a type is capable of calling any method handle
of the same arity as the general type.
(Note: The invoker method is not available via the Core Reflection API.
An attempt to call java.lang.reflect.Method.invoke
on the declared invokeExact
or invoke
method will raise an
UnsupportedOperationException
.)
This method throws no reflective or security exceptions.
type
- the desired target typeIllegalArgumentException
- if the resulting method handle's type would have
too many parameterspublic static MethodHandle explicitCastArguments(MethodHandle target, MethodType newType)
If the original type and new type are equal, returns target.
The same conversions are allowed as for MethodHandle.asType
,
and some additional conversions are also applied if those conversions fail.
Given types T0, T1, one of the following conversions is applied
if possible, before or instead of any conversions done by asType
:
(x & 1) != 0
.
target
- the method handle to invoke after arguments are retypednewType
- the expected type of the new method handleNullPointerException
- if either argument is nullWrongMethodTypeException
- if the conversion cannot be madeMethodHandle.asType(java.lang.invoke.MethodType)
public static MethodHandle permuteArguments(MethodHandle target, MethodType newType, int... reorder)
The given array controls the reordering.
Call #I
the number of incoming parameters (the value
newType.parameterCount()
, and call #O
the number
of outgoing parameters (the value target.type().parameterCount()
).
Then the length of the reordering array must be #O
,
and each element must be a non-negative number less than #I
.
For every N
less than #O
, the N
-th
outgoing argument will be taken from the I
-th incoming
argument, where I
is reorder[N]
.
No argument or return value conversions are applied.
The type of each incoming argument, as determined by newType
,
must be identical to the type of the corresponding outgoing parameter
or parameters in the target method handle.
The return type of newType
must be identical to the return
type of the original target.
The reordering array need not specify an actual permutation.
An incoming argument will be duplicated if its index appears
more than once in the array, and an incoming argument will be dropped
if its index does not appear in the array.
As in the case of dropArguments
,
incoming arguments which are not mentioned in the reordering array
are may be any type, as determined only by newType
.
import static java.lang.invoke.MethodHandles.*; import static java.lang.invoke.MethodType.*; ... MethodType intfn1 = methodType(int.class, int.class); MethodType intfn2 = methodType(int.class, int.class, int.class); MethodHandle sub = ... (int x, int y) -> (x-y) ...; assert(sub.type().equals(intfn2)); MethodHandle sub1 = permuteArguments(sub, intfn2, 0, 1); MethodHandle rsub = permuteArguments(sub, intfn2, 1, 0); assert((int)rsub.invokeExact(1, 100) == 99); MethodHandle add = ... (int x, int y) -> (x+y) ...; assert(add.type().equals(intfn2)); MethodHandle twice = permuteArguments(add, intfn1, 0, 0); assert(twice.type().equals(intfn1)); assert((int)twice.invokeExact(21) == 42);
target
- the method handle to invoke after arguments are reorderednewType
- the expected type of the new method handlereorder
- an index array which controls the reorderingNullPointerException
- if any argument is nullIllegalArgumentException
- if the index array length is not equal to
the arity of the target, or if any index array element
not a valid index for a parameter of newType
,
or if two corresponding parameter types in
target.type()
and newType
are not identical,public static MethodHandle constant(Class<?> type, Object value)
Before the method handle is returned, the passed-in value is converted to the requested type. If the requested type is primitive, widening primitive conversions are attempted, else reference conversions are attempted.
The returned method handle is equivalent to identity(type).bindTo(value)
.
type
- the return type of the desired method handlevalue
- the value to returnNullPointerException
- if the type
argument is nullClassCastException
- if the value cannot be converted to the required return typeIllegalArgumentException
- if the given type is void.class
public static MethodHandle identity(Class<?> type)
type
- the type of the sole parameter and return value of the desired method handleNullPointerException
- if the argument is nullIllegalArgumentException
- if the given type is void.class
public static MethodHandle insertArguments(MethodHandle target, int pos, Object... values)
The type of the new method handle will drop the types for the bound parameters from the original target type, since the new method handle will no longer require those arguments to be supplied by its callers.
Each given argument object must match the corresponding bound parameter type. If a bound parameter type is a primitive, the argument object must be a wrapper, and will be unboxed to produce the primitive value.
The pos
argument selects which parameters are to be bound.
It may range between zero and N-L (inclusively),
where N is the arity of the target method handle
and L is the length of the values array.
target
- the method handle to invoke after the argument is insertedpos
- where to insert the argument (zero for the first)values
- the series of arguments to insertNullPointerException
- if the target or the values
array is nullMethodHandle.bindTo(java.lang.Object)
public static MethodHandle dropArguments(MethodHandle target, int pos, List<Class<?>> valueTypes)
The pos
argument may range between zero and N,
where N is the arity of the target.
If pos
is zero, the dummy arguments will precede
the target's real arguments; if pos
is N
they will come after.
Example:
import static java.lang.invoke.MethodHandles.*; import static java.lang.invoke.MethodType.*; ... MethodHandle cat = lookup().findVirtual(String.class, "concat", methodType(String.class, String.class)); assertEquals("xy", (String) cat.invokeExact("x", "y")); MethodType bigType = cat.type().insertParameterTypes(0, int.class, String.class); MethodHandle d0 = dropArguments(cat, 0, bigType.parameterList().subList(0,2)); assertEquals(bigType, d0.type()); assertEquals("yz", (String) d0.invokeExact(123, "x", "y", "z"));
This method is also equivalent to the following code:
dropArguments
(target, pos, valueTypes.toArray(new Class[0]))
target
- the method handle to invoke after the arguments are droppedvalueTypes
- the type(s) of the argument(s) to droppos
- position of first argument to drop (zero for the leftmost)NullPointerException
- if the target is null,
or if the valueTypes
list or any of its elements is nullIllegalArgumentException
- if any element of valueTypes
is void.class
,
or if pos
is negative or greater than the arity of the target,
or if the new method handle's type would have too many parameterspublic static MethodHandle dropArguments(MethodHandle target, int pos, Class<?>... valueTypes)
The pos
argument may range between zero and N,
where N is the arity of the target.
If pos
is zero, the dummy arguments will precede
the target's real arguments; if pos
is N
they will come after.
Example:
import static java.lang.invoke.MethodHandles.*; import static java.lang.invoke.MethodType.*; ... MethodHandle cat = lookup().findVirtual(String.class, "concat", methodType(String.class, String.class)); assertEquals("xy", (String) cat.invokeExact("x", "y")); MethodHandle d0 = dropArguments(cat, 0, String.class); assertEquals("yz", (String) d0.invokeExact("x", "y", "z")); MethodHandle d1 = dropArguments(cat, 1, String.class); assertEquals("xz", (String) d1.invokeExact("x", "y", "z")); MethodHandle d2 = dropArguments(cat, 2, String.class); assertEquals("xy", (String) d2.invokeExact("x", "y", "z")); MethodHandle d12 = dropArguments(cat, 1, int.class, boolean.class); assertEquals("xz", (String) d12.invokeExact("x", 12, true, "z"));
This method is also equivalent to the following code:
dropArguments
(target, pos, Arrays.asList(valueTypes))
target
- the method handle to invoke after the arguments are droppedvalueTypes
- the type(s) of the argument(s) to droppos
- position of first argument to drop (zero for the leftmost)NullPointerException
- if the target is null,
or if the valueTypes
array or any of its elements is nullIllegalArgumentException
- if any element of valueTypes
is void.class
,
or if pos
is negative or greater than the arity of the target,
or if the new method handle's type would have
too many parameterspublic static MethodHandle filterArguments(MethodHandle target, int pos, MethodHandle... filters)
The pre-processing is performed by one or more method handles,
specified in the elements of the filters
array.
The first element of the filter array corresponds to the pos
argument of the target, and so on in sequence.
Null arguments in the array are treated as identity functions, and the corresponding arguments left unchanged. (If there are no non-null elements in the array, the original target is returned.) Each filter is applied to the corresponding argument of the adapter.
If a filter F
applies to the N
th argument of
the target, then F
must be a method handle which
takes exactly one argument. The type of F
's sole argument
replaces the corresponding argument type of the target
in the resulting adapted method handle.
The return type of F
must be identical to the corresponding
parameter type of the target.
It is an error if there are elements of filters
(null or not)
which do not correspond to argument positions in the target.
Example:
import static java.lang.invoke.MethodHandles.*; import static java.lang.invoke.MethodType.*; ... MethodHandle cat = lookup().findVirtual(String.class, "concat", methodType(String.class, String.class)); MethodHandle upcase = lookup().findVirtual(String.class, "toUpperCase", methodType(String.class)); assertEquals("xy", (String) cat.invokeExact("x", "y")); MethodHandle f0 = filterArguments(cat, 0, upcase); assertEquals("Xy", (String) f0.invokeExact("x", "y")); // Xy MethodHandle f1 = filterArguments(cat, 1, upcase); assertEquals("xY", (String) f1.invokeExact("x", "y")); // xY MethodHandle f2 = filterArguments(cat, 0, upcase, upcase); assertEquals("XY", (String) f2.invokeExact("x", "y")); // XY
Here is pseudocode for the resulting adapter:
V target(P... p, A[i]... a[i], B... b); A[i] filter[i](V[i]); T adapter(P... p, V[i]... v[i], B... b) { return target(p..., f[i](v[i])..., b...); }
target
- the method handle to invoke after arguments are filteredpos
- the position of the first argument to filterfilters
- method handles to call initially on filtered argumentsNullPointerException
- if the target is null
or if the filters
array is nullIllegalArgumentException
- if a non-null element of filters
does not match a corresponding argument type of target as described above,
or if the pos+filters.length
is greater than target.type().parameterCount()
,
or if the resulting method handle's type would have
too many parameterspublic static MethodHandle collectArguments(MethodHandle target, int pos, MethodHandle filter)
If the filter returns a value, the target must accept that value as
its argument in position pos
, preceded and/or followed by
any arguments not passed to the filter.
If the filter returns void, the target must accept all arguments
not passed to the filter.
No arguments are reordered, and a result returned from the filter
replaces (in order) the whole subsequence of arguments originally
passed to the adapter.
The argument types (if any) of the filter
replace zero or one argument types of the target, at position pos
,
in the resulting adapted method handle.
The return type of the filter (if any) must be identical to the
argument type of the target at position pos
, and that target argument
is supplied by the return value of the filter.
In all cases, pos
must be greater than or equal to zero, and
pos
must also be less than or equal to the target's arity.
Example:
import static java.lang.invoke.MethodHandles.*; import static java.lang.invoke.MethodType.*; ... MethodHandle deepToString = publicLookup() .findStatic(Arrays.class, "deepToString", methodType(String.class, Object[].class)); MethodHandle ts1 = deepToString.asCollector(String[].class, 1); assertEquals("[strange]", (String) ts1.invokeExact("strange")); MethodHandle ts2 = deepToString.asCollector(String[].class, 2); assertEquals("[up, down]", (String) ts2.invokeExact("up", "down")); MethodHandle ts3 = deepToString.asCollector(String[].class, 3); MethodHandle ts3_ts2 = collectArguments(ts3, 1, ts2); assertEquals("[top, [up, down], strange]", (String) ts3_ts2.invokeExact("top", "up", "down", "strange")); MethodHandle ts3_ts2_ts1 = collectArguments(ts3_ts2, 3, ts1); assertEquals("[top, [up, down], [strange]]", (String) ts3_ts2_ts1.invokeExact("top", "up", "down", "strange")); MethodHandle ts3_ts2_ts3 = collectArguments(ts3_ts2, 1, ts3); assertEquals("[top, [[up, down, strange], charm], bottom]", (String) ts3_ts2_ts3.invokeExact("top", "up", "down", "strange", "charm", "bottom"));
Here is pseudocode for the resulting adapter:
T target(A...,V,C...); V filter(B...); T adapter(A... a,B... b,C... c) { V v = filter(b...); return target(a...,v,c...); } // and if the filter has no arguments: T target2(A...,V,C...); V filter2(); T adapter2(A... a,C... c) { V v = filter2(); return target2(a...,v,c...); } // and if the filter has a void return: T target3(A...,C...); void filter3(B...); void adapter3(A... a,B... b,C... c) { filter3(b...); return target3(a...,c...); }
A collection adapter collectArguments(mh, 0, coll)
is equivalent to
one which first "folds" the affected arguments, and then drops them, in separate
steps as follows:
If the target method handle consumes no arguments besides than the result (if any) of the filtermh = MethodHandles.dropArguments(mh, 1, coll.type().parameterList()); //step 2 mh = MethodHandles.foldArguments(mh, coll); //step 1
coll
, then collectArguments(mh, 0, coll)
is equivalent to filterReturnValue(coll, mh)
.
If the filter method handle coll
consumes one argument and produces
a non-void result, then collectArguments(mh, N, coll)
is equivalent to filterArguments(mh, N, coll)
.
Other equivalences are possible but would require argument permutation.target
- the method handle to invoke after filtering the subsequence of argumentspos
- the position of the first adapter argument to pass to the filter,
and/or the target argument which receives the result of the filterfilter
- method handle to call on the subsequence of argumentsNullPointerException
- if either argument is nullIllegalArgumentException
- if the return type of filter
is non-void and is not the same as the pos
argument of the target,
or if pos
is not between 0 and the target's arity, inclusive,
or if the resulting method handle's type would have
too many parametersfoldArguments(java.lang.invoke.MethodHandle, java.lang.invoke.MethodHandle)
,
filterArguments(java.lang.invoke.MethodHandle, int, java.lang.invoke.MethodHandle...)
,
filterReturnValue(java.lang.invoke.MethodHandle, java.lang.invoke.MethodHandle)
public static MethodHandle filterReturnValue(MethodHandle target, MethodHandle filter)
If the target returns a value, the filter must accept that value as its only argument. If the target returns void, the filter must accept no arguments.
The return type of the filter replaces the return type of the target in the resulting adapted method handle. The argument type of the filter (if any) must be identical to the return type of the target.
Example:
import static java.lang.invoke.MethodHandles.*; import static java.lang.invoke.MethodType.*; ... MethodHandle cat = lookup().findVirtual(String.class, "concat", methodType(String.class, String.class)); MethodHandle length = lookup().findVirtual(String.class, "length", methodType(int.class)); System.out.println((String) cat.invokeExact("x", "y")); // xy MethodHandle f0 = filterReturnValue(cat, length); System.out.println((int) f0.invokeExact("x", "y")); // 2
Here is pseudocode for the resulting adapter:
V target(A...); T filter(V); T adapter(A... a) { V v = target(a...); return filter(v); } // and if the target has a void return: void target2(A...); T filter2(); T adapter2(A... a) { target2(a...); return filter2(); } // and if the filter has a void return: V target3(A...); void filter3(V); void adapter3(A... a) { V v = target3(a...); filter3(v); }
target
- the method handle to invoke before filtering the return valuefilter
- method handle to call on the return valueNullPointerException
- if either argument is nullIllegalArgumentException
- if the argument list of filter
does not match the return type of target as described abovepublic static MethodHandle foldArguments(MethodHandle target, MethodHandle combiner)
The pre-processing is performed by combiner
, a second method handle.
Of the arguments passed to the adapter, the first N
arguments
are copied to the combiner, which is then called.
(Here, N
is defined as the parameter count of the combiner.)
After this, control passes to the target, with any result
from the combiner inserted before the original N
incoming
arguments.
If the combiner returns a value, the first parameter type of the target
must be identical with the return type of the combiner, and the next
N
parameter types of the target must exactly match the parameters
of the combiner.
If the combiner has a void return, no result will be inserted,
and the first N
parameter types of the target
must exactly match the parameters of the combiner.
The resulting adapter is the same type as the target, except that the first parameter type is dropped, if it corresponds to the result of the combiner.
(Note that dropArguments
can be used to remove any arguments
that either the combiner or the target does not wish to receive.
If some of the incoming arguments are destined only for the combiner,
consider using asCollector
instead, since those
arguments will not need to be live on the stack on entry to the
target.)
Example:
import static java.lang.invoke.MethodHandles.*; import static java.lang.invoke.MethodType.*; ... MethodHandle trace = publicLookup().findVirtual(java.io.PrintStream.class, "println", methodType(void.class, String.class)) .bindTo(System.out); MethodHandle cat = lookup().findVirtual(String.class, "concat", methodType(String.class, String.class)); assertEquals("boojum", (String) cat.invokeExact("boo", "jum")); MethodHandle catTrace = foldArguments(cat, trace); // also prints "boo": assertEquals("boojum", (String) catTrace.invokeExact("boo", "jum"));
Here is pseudocode for the resulting adapter:
// there are N arguments in A... T target(V, A[N]..., B...); V combiner(A...); T adapter(A... a, B... b) { V v = combiner(a...); return target(v, a..., b...); } // and if the combiner has a void return: T target2(A[N]..., B...); void combiner2(A...); T adapter2(A... a, B... b) { combiner2(a...); return target2(a..., b...); }
target
- the method handle to invoke after arguments are combinedcombiner
- method handle to call initially on the incoming argumentsNullPointerException
- if either argument is nullIllegalArgumentException
- if combiner
's return type
is non-void and not the same as the first argument type of
the target, or if the initial N
argument types
of the target
(skipping one matching the combiner
's return type)
are not identical with the argument types of combiner
public static MethodHandle guardWithTest(MethodHandle test, MethodHandle target, MethodHandle fallback)
Here is pseudocode for the resulting adapter:
Note that the test arguments (boolean test(A...); T target(A...,B...); T fallback(A...,B...); T adapter(A... a,B... b) { if (test(a...)) return target(a..., b...); else return fallback(a..., b...); }
a...
in the pseudocode) cannot
be modified by execution of the test, and so are passed unchanged
from the caller to the target or fallback as appropriate.test
- method handle used for test, must return booleantarget
- method handle to call if test passesfallback
- method handle to call if test failsNullPointerException
- if any argument is nullIllegalArgumentException
- if test
does not return boolean,
or if all three method types do not match (with the return
type of test
changed to match that of the target).public static MethodHandle catchException(MethodHandle target, Class<? extends Throwable> exType, MethodHandle handler)
The target and handler must have the same corresponding
argument and return types, except that handler may omit trailing arguments
(similarly to the predicate in guardWithTest
).
Also, the handler must have an extra leading parameter of exType
or a supertype.
Here is pseudocode for the resulting adapter:
Note that the saved arguments (T target(A..., B...); T handler(ExType, A...); T adapter(A... a, B... b) { try { return target(a..., b...); } catch (ExType ex) { return handler(ex, a...); } }
a...
in the pseudocode) cannot
be modified by execution of the target, and so are passed unchanged
from the caller to the handler, if the handler is invoked.
The target and handler must return the same type, even if the handler
always throws. (This might happen, for instance, because the handler
is simulating a finally
clause).
To create such a throwing handler, compose the handler creation logic
with throwException
,
in order to create a method handle of the correct return type.
target
- method handle to callexType
- the type of exception which the handler will catchhandler
- method handle to call if a matching exception is thrownNullPointerException
- if any argument is nullIllegalArgumentException
- if handler
does not accept
the given exception type, or if the method handle types do
not match in their return types and their
corresponding parameterstryFinally(MethodHandle, MethodHandle)
public static MethodHandle throwException(Class<?> returnType, Class<? extends Throwable> exType)
exType
.
The method handle will accept a single argument of exType
,
and immediately throw it as an exception.
The method type will nominally specify a return of returnType
.
The return type may be anything convenient: It doesn't matter to the
method handle's behavior, since it will never return normally.returnType
- the return type of the desired method handleexType
- the parameter type of the desired method handleNullPointerException
- if either argument is nullpublic static MethodHandle loop(MethodHandle[]... clauses)
Intuitively, every loop is formed by one or more "clauses", each specifying a local iteration value and/or a loop exit. Each iteration of the loop executes each clause in order. A clause can optionally update its iteration variable; it can also optionally perform a test and conditional loop exit. In order to express this logic in terms of method handles, each clause will determine four actions:
Some of these clause parts may be omitted according to certain rules, and useful default behavior is provided in this case. See below for a detailed description.
Each clause function, with the exception of clause initializers, is able to observe the entire loop state,
because it will be passed all current iteration variable values, as well as all incoming loop
parameters. Most clause functions will not need all of this information, but they will be formally connected as
if by dropArguments(java.lang.invoke.MethodHandle, int, java.util.List<java.lang.Class<?>>)
.
Given a set of clauses, there is a number of checks and adjustments performed to connect all the parts of the
loop. They are spelled out in detail in the steps below. In these steps, every occurrence of the word "must"
corresponds to a place where IllegalArgumentException
may be thrown if the required constraint is not met
by the inputs to the loop combinator. The term "effectively identical", applied to parameter type lists, means
that they must be identical, or else one list must be a proper prefix of the other.
Step 0: Determine clause structure.
MethodHandle[][]
must be non-null
and contain at least one element.
null
s or sub-arrays longer than four elements.
null
elements to length
four. Padding takes place by appending elements to the array.
null
s are disregarded.
Step 1A: Determine iteration variables.
void
; else if one is omitted, use the other's return type; else
use the common return type (they must be identical).
void
.
Step 1B: Determine loop parameters.
null
parameter lists.
Step 1C: Determine loop return type.
void
as the loop return type.
Step 1D: Check other types.
boolean
return type.
(Implementation Note: Steps 1A, 1B, 1C, 1D are logically independent of each other, and may be performed in any order.)
Step 2: Determine parameter lists.
Step 3: Fill in omitted functions.
null
/zero/false
/void
type. (For this purpose, a constant void
is simply a
function which does nothing and returns void
; it can be obtained from another constant function by
type conversion.)
void
iteration variables of preceding clauses. (This will turn the loop variable into a local loop invariant.)
true
function. (This will keep the loop going, as far
as this clause is concerned.)
null
/zero/false
/void
function of the
loop return type.
Step 4: Fill in missing parameter types.
Final observations.
void
) iteration variables followed by loop parameters.
Loop execution.
void
values
are saved (as the common prefix) into locals. These locals are loop varying (unless their steps are identity
functions, as noted above).
void
iteration values (in clause order) and then the loop inputs (in argument order).
false
.
void
result from a step function call is used to update the corresponding loop variable. The
updated value is immediately visible to all subsequent function calls.
false
, the corresponding fini function is called, and the resulting value
is returned from the loop as a whole.
Here is pseudocode for the resulting loop handle. In the code, V
/v
represent the types / values
of loop variables; A
/a
, those of arguments passed to the resulting loop; and R
, the
result types of finalizers as well as of the resulting loop.
V... init...(A...); boolean pred...(V..., A...); V... step...(V..., A...); R fini...(V..., A...); R loop(A... a) { V... v... = init...(a...); for (;;) { for ((v, p, s, f) in (v..., pred..., step..., fini...)) { v = s(v..., a...); if (!p(v..., a...)) { return f(v..., a...); } } } }
// iterative implementation of the factorial function as a loop handle static int one(int k) { return 1; } int inc(int i, int acc, int k) { return i + 1; } int mult(int i, int acc, int k) { return i * acc; } boolean pred(int i, int acc, int k) { return i < k; } int fin(int i, int acc, int k) { return acc; } // assume MH_one, MH_inc, MH_mult, MH_pred, and MH_fin are handles to the above methods // null initializer for counter, should initialize to 0 MethodHandle[] counterClause = new MethodHandle[]{null, MH_inc}; MethodHandle[] accumulatorClause = new MethodHandle[]{MH_one, MH_mult, MH_pred, MH_fin}; MethodHandle loop = MethodHandles.loop(counterClause, accumulatorClause); assertEquals(120, loop.invoke(5));
clauses
- an array of arrays (4-tuples) of MethodHandle
s adhering to the rules described above.IllegalArgumentException
- in case any of the constraints described above is violated.whileLoop(MethodHandle, MethodHandle, MethodHandle)
,
doWhileLoop(MethodHandle, MethodHandle, MethodHandle)
,
countedLoop(MethodHandle, MethodHandle, MethodHandle)
,
iteratedLoop(MethodHandle, MethodHandle, MethodHandle)
public static MethodHandle whileLoop(MethodHandle init, MethodHandle pred, MethodHandle body)
while
loop from an initializer, a body, and a predicate. This is a convenience wrapper for
the generic loop combinator.
The loop handle's result type is the same as the sole loop variable's, i.e., the result type of init
.
The parameter type list of init
also determines that of the resulting handle. The pred
handle
must have an additional leading parameter of the same type as init
's result, and so must the body
. These constraints follow directly from those described for the generic loop combinator.
Here is pseudocode for the resulting loop handle. In the code, V
/v
represent the type / value of
the sole loop variable as well as the result type of the loop; and A
/a
, that of the argument
passed to the loop.
V init(A); boolean pred(V, A); V body(V, A); V whileLoop(A a) { V v = init(a); while (pred(v, a)) { v = body(v, a); } return v; }
// implement the zip function for lists as a loop handle List<String> initZip(Iterator<String> a, Iterator<String> b) { return new ArrayList<>(); } boolean zipPred(List<String> zip, Iterator<String> a, Iterator<String> b) { return a.hasNext() && b.hasNext(); } List<String> zipStep(List<String> zip, Iterator<String> a, Iterator<String> b) { zip.add(a.next()); zip.add(b.next()); return zip; } // assume MH_initZip, MH_zipPred, and MH_zipStep are handles to the above methods MethodHandle loop = MethodHandles.doWhileLoop(MH_initZip, MH_zipStep, MH_zipPred); List<String> a = Arrays.asList("a", "b", "c", "d"); List<String> b = Arrays.asList("e", "f", "g", "h"); List<String> zipped = Arrays.asList("a", "e", "b", "f", "c", "g", "d", "h"); assertEquals(zipped, (List<String>) loop.invoke(a.iterator(), b.iterator()));
MethodHandle whileLoop(MethodHandle init, MethodHandle pred, MethodHandle body) { MethodHandle[] checkExit = {null, null, pred, identity(init.type().returnType())}, varBody = {init, body}; return loop(checkExit, varBody); }
init
- initializer: it should provide the initial value of the loop variable. This controls the loop's
result type. Passing null
or a void
init function will make the loop's result type
void
.pred
- condition for the loop, which may not be null
.body
- body of the loop, which may not be null
.IllegalArgumentException
- if any argument has a type inconsistent with the loop structureloop(MethodHandle[][])
public static MethodHandle doWhileLoop(MethodHandle init, MethodHandle body, MethodHandle pred)
do-while
loop from an initializer, a body, and a predicate. This is a convenience wrapper
for the generic loop combinator.
The loop handle's result type is the same as the sole loop variable's, i.e., the result type of init
.
The parameter type list of init
also determines that of the resulting handle. The pred
handle
must have an additional leading parameter of the same type as init
's result, and so must the body
. These constraints follow directly from those described for the generic loop combinator.
Here is pseudocode for the resulting loop handle. In the code, V
/v
represent the type / value of
the sole loop variable as well as the result type of the loop; and A
/a
, that of the argument
passed to the loop.
V init(A); boolean pred(V, A); V body(V, A); V doWhileLoop(A a) { V v = init(a); do { v = body(v, a); } while (pred(v, a)); return v; }
// int i = 0; while (i < limit) { ++i; } return i; => limit int zero(int limit) { return 0; } int step(int i, int limit) { return i + 1; } boolean pred(int i, int limit) { return i < limit; } // assume MH_zero, MH_step, and MH_pred are handles to the above methods MethodHandle loop = MethodHandles.doWhileLoop(MH_zero, MH_step, MH_pred); assertEquals(23, loop.invoke(23));
MethodHandle doWhileLoop(MethodHandle init, MethodHandle body, MethodHandle pred) { MethodHandle[] clause = { init, body, pred, identity(init.type().returnType()) }; return loop(clause); }
init
- initializer: it should provide the initial value of the loop variable. This controls the loop's
result type. Passing null
or a void
init function will make the loop's result type
void
.pred
- condition for the loop, which may not be null
.body
- body of the loop, which may not be null
.IllegalArgumentException
- if any argument has a type inconsistent with the loop structureloop(MethodHandle[][])
public static MethodHandle countedLoop(MethodHandle iterations, MethodHandle init, MethodHandle body)
int
initialized from the
iterations
handle evaluation result. The counter is passed to the body
function, so that must
accept an initial int
argument. The result of the loop execution is the final value of the additional
local state. This is a convenience wrapper for the generic loop
combinator.
The result type and parameter type list of init
determine those of the resulting handle. The iterations
handle must accept the same parameter types as init
but return an int
. The body
handle must accept the same parameter types as well, preceded by an int
parameter for the counter,
and a parameter of the same type as init
's result. These constraints follow directly from those described
for the generic loop combinator.
Here is pseudocode for the resulting loop handle. In the code, V
/v
represent the type / value of
the sole loop variable as well as the result type of the loop; and A
/a
, that of the argument
passed to the loop.
int iterations(A); V init(A); V body(int, V, A); V countedLoop(A a) { int end = iterations(a); V v = init(a); for (int i = 0; i < end; ++i) { v = body(i, v, a); } return v; }
// String s = "Lambdaman!"; for (int i = 0; i < 13; ++i) { s = "na " + s; } return s; // => a variation on a well known theme String start(String arg) { return arg; } String step(int counter, String v, String arg) { return "na " + v; } // assume MH_start and MH_step are handles to the two methods above MethodHandle fit13 = MethodHandles.constant(int.class, 13); MethodHandle loop = MethodHandles.countedLoop(fit13, MH_start, MH_step); assertEquals("na na na na na na na na na na na na na Lambdaman!", loop.invoke("Lambdaman!"));
MethodHandle countedLoop(MethodHandle iterations, MethodHandle init, MethodHandle body) { return countedLoop(null, iterations, init, body); // null => constant zero }
iterations
- a handle to return the number of iterations this loop should run.init
- initializer for additional loop state. This determines the loop's result type.
Passing null
or a void
init function will make the loop's result type
void
.body
- the body of the loop, which must not be null
.
It must accept an initial int
parameter (for the counter), and then any
additional loop-local variable plus loop parameters.IllegalArgumentException
- if any argument has a type inconsistent with the loop structurepublic static MethodHandle countedLoop(MethodHandle start, MethodHandle end, MethodHandle init, MethodHandle body)
int
that will be
initialized to the int
value returned from the evaluation of the start
handle and run to the
value returned from end
(exclusively) with a step width of 1. The counter value is passed to the body
function in each iteration; it has to accept an initial int
parameter
for that. The result of the loop execution is the final value of the additional local state
obtained by running init
.
This is a
convenience wrapper for the generic loop combinator.
The constraints for the init
and body
handles are the same as for countedLoop(MethodHandle, MethodHandle, MethodHandle)
. Additionally, the start
and end
handles
must return an int
and accept the same parameters as init
.
Here is pseudocode for the resulting loop handle. In the code, V
/v
represent the type / value of
the sole loop variable as well as the result type of the loop; and A
/a
, that of the argument
passed to the loop.
int start(A); int end(A); V init(A); V body(int, V, A); V countedLoop(A a) { int s = start(a); int e = end(a); V v = init(a); for (int i = s; i < e; ++i) { v = body(i, v, a); } return v; }
MethodHandle countedLoop(MethodHandle start, MethodHandle end, MethodHandle init, MethodHandle body) { MethodHandle returnVar = dropArguments(identity(init.type().returnType()), 0, int.class, int.class); // assume MH_increment and MH_lessThan are handles to x+1 and x<y of type int MethodHandle[] indexVar = {start, MH_increment}, // i = start; i = i+1 loopLimit = {end, null, MH_lessThan, returnVar }, // i<end bodyClause = {init, dropArguments(body, 1, int.class)}; // v = body(i, v); return loop(indexVar, loopLimit, bodyClause); }
start
- a handle to return the start value of the loop counter.
If it is null
, a constant zero is assumed.end
- a non-null
handle to return the end value of the loop counter (the loop will run to end-1
).init
- initializer for additional loop state. This determines the loop's result type.
Passing null
or a void
init function will make the loop's result type
void
.body
- the body of the loop, which must not be null
.
It must accept an initial int
parameter (for the counter), and then any
additional loop-local variable plus loop parameters.IllegalArgumentException
- if any argument has a type inconsistent with the loop structurepublic static MethodHandle iteratedLoop(MethodHandle iterator, MethodHandle init, MethodHandle body)
Iterator<T>
.
The iterator will be produced by the evaluation of the iterator
handle.
If this handle is passed as null
the method Iterable.iterator()
will be used instead,
and will be applied to a leading argument of the loop handle.
Each value produced by the iterator is passed to the body
, which must accept an initial T
parameter.
The result of the loop execution is the final value of the additional local state
obtained by running init
.
This is a convenience wrapper for the
generic loop combinator, and the constraints imposed on the body
handle follow directly from those described for the latter.
Here is pseudocode for the resulting loop handle. In the code, V
/v
represent the type / value of
the loop variable as well as the result type of the loop; T
/t
, that of the elements of the
structure the loop iterates over, and A
/a
, that of the argument passed to the loop.
Iterator<T> iterator(A); // defaults to Iterable::iterator V init(A); V body(T,V,A); V iteratedLoop(A a) { Iterator<T> it = iterator(a); V v = init(a); for (T t : it) { v = body(t, v, a); } return v; }
The type T
may be either a primitive or reference.
Since type Iterator<T>
is erased in the method handle representation to the raw type
Iterator
, the iteratedLoop
combinator adjusts the leading argument type for body
to Object
as if by the asType
conversion method.
Therefore, if an iterator of the wrong type appears as the loop is executed,
runtime exceptions may occur as the result of dynamic conversions performed by asType
.
// reverse a list List<String> reverseStep(String e, List<String> r, List<String> l) { r.add(0, e); return r; } List<String> newArrayList(List<String> l) { return new ArrayList<>(); } // assume MH_reverseStep, MH_newArrayList are handles to the above methods MethodHandle loop = MethodHandles.iteratedLoop(null, MH_newArrayList, MH_reverseStep); List<String> list = Arrays.asList("a", "b", "c", "d", "e"); List<String> reversedList = Arrays.asList("e", "d", "c", "b", "a"); assertEquals(reversedList, (List<String>) loop.invoke(list));
MethodHandle iteratedLoop(MethodHandle iterator, MethodHandle init, MethodHandle body) { // assume MH_next and MH_hasNext are handles to methods of Iterator Class<?> itype = iterator.type().returnType(); Class<?> ttype = body.type().parameterType(0); MethodHandle returnVar = dropArguments(identity(init.type().returnType()), 0, itype); MethodHandle nextVal = MH_next.asType(MH_next.type().changeReturnType(ttype)); MethodHandle[] iterVar = {iterator, null, MH_hasNext, returnVar}, // it = iterator(); while (it.hasNext) bodyClause = {init, filterArgument(body, 0, nextVal)}; // v = body(t, v, a); return loop(iterVar, bodyClause); }
iterator
- a handle to return the iterator to start the loop.
Passing null
will make the loop call Iterable.iterator()
on the first
incoming value.init
- initializer for additional loop state. This determines the loop's result type.
Passing null
or a void
init function will make the loop's result type
void
.body
- the body of the loop, which must not be null
.
It must accept an initial T
parameter (for the iterated values), and then any
additional loop-local variable plus loop parameters.IllegalArgumentException
- if any argument has a type inconsistent with the loop structurepublic static MethodHandle tryFinally(MethodHandle target, MethodHandle cleanup)
target
method handle by wrapping it in a try-finally
block.
Another method handle, cleanup
, represents the functionality of the finally
block. Any exception
thrown during the execution of the target
handle will be passed to the cleanup
handle. The
exception will be rethrown, unless cleanup
handle throws an exception first. The
value returned from the cleanup
handle's execution will be the result of the execution of the
try-finally
handle.
The cleanup
handle will be passed one or two additional leading arguments.
The first is the exception thrown during the
execution of the target
handle, or null
if no exception was thrown.
The second is the result of the execution of the target
handle, or, if it throws an exception,
a null
, zero, or false
value of the required type is supplied as a placeholder.
The second argument is not present if the target
handle has a void
return type.
(Note that, except for argument type conversions, combinators represent void
values in parameter lists
by omitting the corresponding paradoxical arguments, not by inserting null
or zero values.)
The target
and cleanup
handles' return types must be the same. Their parameter type lists also
must be the same, but the cleanup
handle must accept one or two more leading parameters:
Throwable
, which will carry the exception thrown by the target
handle (if any); and
target
and cleanup
, which will carry
the result from the execution of the target
handle.
This parameter is not present if the target
returns void
.
The pseudocode for the resulting adapter looks as follows. In the code, V
represents the result type of
the try/finally
construct; A
/a
, the types and values of arguments to the resulting
handle consumed by the cleanup; and B
/b
, those of arguments to the resulting handle discarded by
the cleanup.
V target(A..., B...); V cleanup(Throwable, V, A...); V adapter(A... a, B... b) { V result = (zero value for V); Throwable throwable = null; try { result = target(a..., b...); } catch (Throwable t) { throwable = t; throw t; } finally { result = cleanup(throwable, result, a...); } return result; }
Note that the saved arguments (a...
in the pseudocode) cannot
be modified by execution of the target, and so are passed unchanged
from the caller to the cleanup, if it is invoked.
The target and cleanup must return the same type, even if the cleanup
always throws.
To create such a throwing cleanup, compose the cleanup logic
with throwException
,
in order to create a method handle of the correct return type.
Note that tryFinally
never converts exceptions into normal returns.
In rare cases where exceptions must be converted in that way, first wrap
the target with catchException(MethodHandle, Class, MethodHandle)
to capture an outgoing exception, and then wrap with tryFinally
.
target
- the handle whose execution is to be wrapped in a try
block.cleanup
- the handle that is invoked in the finally block.try-finally
block composed of the two arguments.NullPointerException
- if any argument is nullIllegalArgumentException
- if cleanup
does not accept
the required leading arguments, or if the method handle types do
not match in their return types and their
corresponding trailing parameterscatchException(MethodHandle, Class, MethodHandle)
public static MethodHandle foldArguments(MethodHandle target, int pos, MethodHandle combiner)
This method is closely related to foldArguments(MethodHandle, MethodHandle)
, but allows to control the
position in the parameter list at which folding takes place. The argument controlling this, pos
, is a
zero-based index. The aforementioned method foldArguments(MethodHandle, MethodHandle)
assumes position
0.
import static java.lang.invoke.MethodHandles.*; import static java.lang.invoke.MethodType.*; ... MethodHandle trace = publicLookup().findVirtual(java.io.PrintStream.class, "println", methodType(void.class, String.class)) .bindTo(System.out); MethodHandle cat = lookup().findVirtual(String.class, "concat", methodType(String.class, String.class)); assertEquals("boojum", (String) cat.invokeExact("boo", "jum")); MethodHandle catTrace = foldArguments(cat, 1, trace); // also prints "jum": assertEquals("boojum", (String) catTrace.invokeExact("boo", "jum"));
Here is pseudocode for the resulting adapter:
// there are N arguments in A... T target(Z..., V, A[N]..., B...); V combiner(A...); T adapter(Z... z, A... a, B... b) { V v = combiner(a...); return target(z..., v, a..., b...); } // and if the combiner has a void return: T target2(Z..., A[N]..., B...); void combiner2(A...); T adapter2(Z... z, A... a, B... b) { combiner2(a...); return target2(z..., a..., b...); }
target
- the method handle to invoke after arguments are combinedpos
- the position at which to start folding and at which to insert the folding result; if this is 0
, the effect is the same as for foldArguments(MethodHandle, MethodHandle)
.combiner
- method handle to call initially on the incoming argumentsNullPointerException
- if either argument is nullIllegalArgumentException
- if combiner
's return type
is non-void and not the same as the argument type at position pos
of
the target signature, or if the N
argument types at position pos
of the target signature
(skipping one matching the combiner
's return type)
are not identical with the argument types of combiner
foldArguments(MethodHandle, MethodHandle)
Submit a bug or feature
For further API reference and developer documentation, see Java SE Documentation. That documentation contains more detailed, developer-targeted descriptions, with conceptual overviews, definitions of terms, workarounds, and working code examples.
Copyright © 1993, 2016, Oracle and/or its affiliates. All rights reserved.
DRAFT 9-internal+0-2016-01-26-133437.ivan.openjdk9onspinwait