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

com.jfinal.template.expr.ast.SharedMethodKit Maven / Gradle / Ivy

The newest version!
/**
 * Copyright (c) 2011-2023, James Zhan 詹波 ([email protected]).
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License 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.jfinal.template.expr.ast;

import java.util.ArrayList;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Set;
import java.util.HashMap;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import com.jfinal.kit.HashKit;
import com.jfinal.kit.ReflectKit;
import com.jfinal.kit.SyncWriteMap;

/**
 * SharedMethodKit
 */
public class SharedMethodKit {

	private static final Set excludedMethodKey = new HashSet();

	static {
		Method[] methods = Object.class.getMethods();
		for (Method method : methods) {
			Long key = getSharedMethodKey(method.getName(), method.getParameterTypes());
			excludedMethodKey.add(key);
		}
	}

	private final List sharedMethodList = new ArrayList();
	private final HashMap methodCache = new SyncWriteMap(512, 0.25F);

	public SharedMethodInfo getSharedMethodInfo(String methodName, Object[] argValues) {
		Class[] argTypes = MethodKit.getArgTypes(argValues);
		Long key = getSharedMethodKey(methodName, argTypes);
		SharedMethodInfo method = methodCache.get(key);
		if (method == null) {
			method = doGetSharedMethodInfo(methodName, argTypes);
			if (method != null) {
				methodCache.putIfAbsent(key, method);
			}
			// shared method 不支持 null safe,不缓存: methodCache.putIfAbsent(key, Void.class)
		}
		return method;
	}

	private SharedMethodInfo doGetSharedMethodInfo(String methodName, Class[] argTypes) {
		for (SharedMethodInfo smi : sharedMethodList) {
			if (smi.getName().equals(methodName)) {
				Class[] paraTypes = smi.getParameterTypes();
				if (MethodKit.matchFixedArgTypes(paraTypes, argTypes)) {	// 无条件优先匹配固定参数方法
					return smi;
				}
				if (smi.isVarArgs() && MethodKit.matchVarArgTypes(paraTypes, argTypes)) {
					return smi;
				}
			}
		}
		return null;
	}

	public void addSharedMethod(Object sharedMethodFromObject) {
		addSharedMethod(sharedMethodFromObject.getClass(), sharedMethodFromObject);
	}

	public void addSharedMethod(Class sharedMethodFromClass) {
		addSharedMethod(sharedMethodFromClass, ReflectKit.newInstance(sharedMethodFromClass));
	}

	public void addSharedStaticMethod(Class sharedStaticMethodFromClass) {
		addSharedMethod(sharedStaticMethodFromClass, null);
	}

	public void removeSharedMethod(String methodName) {
		Iterator it = sharedMethodList.iterator();
		while(it.hasNext()) {
			SharedMethodInfo smi = it.next();
			if (smi.getName().equals(methodName)) {
				it.remove();
				methodCache.remove(smi.getKey());
			}
		}
	}

	public void removeSharedMethod(Class sharedClass) {
		Iterator it = sharedMethodList.iterator();
		while(it.hasNext()) {
			SharedMethodInfo smi = it.next();
			if (smi.getClazz() == sharedClass) {
				it.remove();
				methodCache.remove(smi.getKey());
			}
		}
	}

	public void removeSharedMethod(Method method) {
		Iterator it = sharedMethodList.iterator();
		while(it.hasNext()) {
			SharedMethodInfo current = it.next();
			String methodName = method.getName();
			if (current.getName().equals(methodName)) {
				Long key = getSharedMethodKey(methodName, method.getParameterTypes());
				if (current.getKey().equals(key)) {
					it.remove();
					methodCache.remove(current.getKey());
				}
			}
		}
	}

	public void removeSharedMethod(String methodName, Class... paraTypes) {
		Long key = getSharedMethodKey(methodName, paraTypes);
		sharedMethodList.removeIf(sharedMethodInfo -> sharedMethodInfo.getKey().equals(key));
		methodCache.remove(key);
	}

	private synchronized void addSharedMethod(Class sharedClass, Object target) {
		if (MethodKit.isForbiddenClass(sharedClass)) {
			throw new IllegalArgumentException("Forbidden class: " + sharedClass.getName());
		}

		Method[] methods = sharedClass.getMethods();
		for (Method method : methods) {
			Long key = getSharedMethodKey(method.getName(), method.getParameterTypes());
			if (excludedMethodKey.contains(key)) {
				continue ;
			}

			for (SharedMethodInfo smi : sharedMethodList) {
				if (smi.getKey().equals(key)) {
					throw new RuntimeException("The shared method is already exists : " + smi.toString());
				}
			}

			if (target != null) {
				sharedMethodList.add(new SharedMethodInfo(key, sharedClass, method, target));
			} else if (Modifier.isStatic(method.getModifiers())) { 	// target 为 null 时添加 static method
				sharedMethodList.add(new SharedMethodInfo(key, sharedClass, method, null));
			}
		}
	}

	private static Long getSharedMethodKey(String methodName, Class[] argTypes) {
		long hash = HashKit.FNV_OFFSET_BASIS_64;
		hash ^= methodName.hashCode();
		hash *= HashKit.FNV_PRIME_64;

		if (argTypes != null) {
			for (int i=0; i type = argTypes[i];
				if (type != null) {
					hash ^= type.getName().hashCode();
					hash *= HashKit.FNV_PRIME_64;
				} else {
					hash ^= "null".hashCode();
					hash *= HashKit.FNV_PRIME_64;
				}
			}
		}
		return hash;
	}

	static class SharedMethodInfo extends MethodInfo {
		final Object target;

		private SharedMethodInfo(Long key, Class clazz, Method method, Object target) {
			super(key, clazz, method);
			this.target = target;
		}

		public Object invoke(Object... args) throws ReflectiveOperationException {
			return super.invoke(target, args);
		}

		Class getClazz() {
			return clazz;
		}
	}
}





© 2015 - 2024 Weber Informatics LLC | Privacy Policy