package com.coderising.jvm.engine; import java.util.HashMap; import java.util.Map; import com.coderising.jvm.clz.ClassFile; import com.coderising.jvm.constant.MethodRefInfo; import com.coderising.jvm.loader.ClassFileLoader; import com.coderising.jvm.method.Method; public class MethodArea { public static final MethodArea instance = new MethodArea(); /** * 注意:我们做了极大的简化, ClassLoader 只有一个, 实际JVM中的ClassLoader,是一个双亲委托的模型 * 方法区: * 1.存储类的数据,如字符串常量,数字常量,字段引用,方法引用,类引用及名称和类型 * 2.存储方法字节码 * * 加载类到方法区: * 将classloader加载到MethodArea */ private ClassFileLoader clzLoader = null; Map<String,ClassFile> map = new HashMap<String,ClassFile>(); //HashMap<className,classFile> private MethodArea(){ } public static MethodArea getInstance(){ return instance; } public void setClassFileLoader(ClassFileLoader clzLoader){ this.clzLoader = clzLoader; } public Method getMainMethod(String className){ ClassFile clzFile = this.findClassFile(className); return clzFile.getMainMethod(); } public ClassFile findClassFile(String className){ //非常简化的实现,类装载器可以是多个,这里简化实现 if(map.get(className) != null){ return map.get(className); } // 看来该class 文件还没有load过 ClassFile clzFile = this.clzLoader.loadClass(className); map.put(className, clzFile); return clzFile; } public Method getMethod(String className, String methodName, String paramAndReturnType){ ClassFile clz = this.findClassFile(className); Method m = clz.getMethod(methodName, paramAndReturnType); if(m == null){ throw new RuntimeException("method can't be found : \n" + "class: " + className + "method: " + methodName + "paramAndReturnType: " + paramAndReturnType); } return m; } public Method getMethod(MethodRefInfo methodRef){ ClassFile clz = this.findClassFile(methodRef.getClassName()); Method m = clz.getMethod(methodRef.getMethodName(), methodRef.getParamAndReturnType()); if(m == null){ throw new RuntimeException("method can't be found : " + methodRef.toString()); } return m; } }