All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.ibm.cloud.objectstorage.util.ReflectionMethodInvoker Maven / Gradle / Ivy

/*
 * Copyright (c) 2019. Amazon.com, Inc. or its affiliates. All Rights Reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License").
 * You may not use this file except in compliance with the License.
 * A copy of the License is located at
 *
 * http://aws.amazon.com/apache2.0
 *
 * or in the "license" file accompanying this file. This file is distributed
 * on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either
 * express or implied. See the License for the specific language governing
 * permissions and limitations under the License.
 */
package com.ibm.cloud.objectstorage.util;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

import com.ibm.cloud.objectstorage.SdkClientException;
import com.ibm.cloud.objectstorage.annotation.SdkInternalApi;

/**
 * This class acts as a proxy to invoke a specific method on objects of a specific class. It will use the JDK's
 * reflection library to find and invoke the method.
 * 

* The relatively expensive call to find the correct method on the class is lazy and will not be performed until the * first invocation or until the invoker is explicitly initialized. Once found, the method is cached so repeated * calls to initialize() or invoke() will not incur the reflection cost of searching for the method on the class. *

* Example: * {@code * ReflectionMethodInvoker invoker = * new ReflectionMethodInvoker(String.class, Integer.class, "indexOf", String.class, int.class); * invoker.invoke("ababab", "ab", 1); // This is equivalent to calling "ababab".indexOf("ab", 1); * } * @param The class type that has the method to be invoked. * @param The expected return type of the method invocation. */ @SdkInternalApi public class ReflectionMethodInvoker { private final Class clazz; private final String methodName; private final Class returnType; private final Class[] parameterTypes; private Method targetMethod; /** * Construct an instance of {@code ReflectionMethodInvoker}. *

* This constructor will not make any reflection calls as part of initialization; i.e. no validation of the * existence of the given method signature will occur. * @param clazz The class that has the method to be invoked. * @param returnType The expected return class of the method invocation. The object returned by the invocation * will be cast to this class. * @param methodName The name of the method to invoke. * @param parameterTypes The classes of the parameters of the method to invoke. */ public ReflectionMethodInvoker(Class clazz, Class returnType, String methodName, Class... parameterTypes) { this.clazz = clazz; this.methodName = methodName; this.returnType = returnType; this.parameterTypes = parameterTypes; } /** * Attempt to invoke the method this proxy targets for on the given object with the given arguments. If the * invoker has not yet been initialized, an attempt to initialize the invoker will be made first. If the call * succeeds the invoker will be in an initialized state after this call and future calls to the same method will * not incur an initialization cost. If the call fails because the target method could not be found, the invoker * will remain in an uninitialized state after the exception is thrown. * @param obj The object to invoke the method on. * @param args The arguments to pass to the method. These arguments must match the signature of the method. * @return The returned value of the method cast to the 'returnType' class that this proxy was initialized with. * @throws NoSuchMethodException if the JVM could not find a method matching the signature specified in the * initialization of this proxy. * @throws SdkClientException if any other exception is thrown when attempting to invoke the method or by the * method itself. The cause of this exception will be the exception that was actually thrown. */ public R invoke(T obj, Object... args) throws NoSuchMethodException { Method targetMethod = getTargetMethod(); try { Object rawResult = targetMethod.invoke(obj, args); return returnType.cast(rawResult); } catch (IllegalAccessException e) { throw new SdkClientException(e); } catch (InvocationTargetException e) { throw new SdkClientException(e); } } /** * Initializes the method invoker by finding and caching the target method from the target class that will be used * for subsequent invoke operations. If the invoker is already initialized this call will do nothing. * @throws NoSuchMethodException if the JVM could not find a method matching the signature specified in the * initialization of this proxy. */ public void initialize() throws NoSuchMethodException { getTargetMethod(); } /** * Gets the initialization state of the invoker. * @return true if the invoker has been initialized and the method has been cached for invocation; otherwise false. */ public boolean isInitialized() { return targetMethod != null; } private Method getTargetMethod() throws NoSuchMethodException { if (targetMethod != null) { return targetMethod; } try { targetMethod = clazz.getMethod(methodName, parameterTypes); return targetMethod; } catch (NullPointerException e) { throw new SdkClientException(e); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy