用自己的话总结一下jvm的工作原理:
1.java对象都是以class的对象存在的,所以在编译的时候可以看到,java文件中有内部类的时候,会编译成javafilename$innerClassName.class的这种形式
2.jvm加载class的时候有一个顺序,jvm是通过classpath加载class的,class path 分为三层,bootstrap,extension,application
---------------------------转自:http://7-sun.com/text/5501.html
Bootstrap Class Loader ,Extension Class Loader ,Application Class Loader三种Class Loader是JVM 系统已经事先实现。
Bootstrap Class Loader 采用的是C或其他相应语言编写(根据JRE 操作系统版本不同而不同),其他两种Class Loader 均采用Java 语言编写,他们的实现为ExtClassLoader 与AppClassLoader 两个内部静态类,位于sun.misc.Launcher内,而Launcher则位于rt.jar中。
以下网址可以下载JDK 的源码实现:
http://download.java.net/openjdk/jdk6/
通过源码可以是我们更进一步了解Class Loader 的内部实现及流程。
Bootstrap Class Loader
Bootstrap Class Loader 的实现位于${openjdk}\hotspot\src\share\vm\classfile 目录下的 classLoader.cpp 与classLoader.hpp,打开源代码可以发现Bootstrap Class Loader是采用C++ 编写的,这就是上一节代码中为什么得不到Bootstrap Class Loader 对象的原因。
该ClassLoader 可以做到根据类文件名来加载类,这其实就和java.lang.ClassLoader 的loadClass 功能非常相近了。
ClassFileStream* ClassPathDirEntry::open_stream(const char* name) { ... jio_snprintf(path, sizeof(path), "%s%s%s", _dir, os::file_separator(), name) ... }
可以看到path=_dir + 文件分隔符 + name,得到path 后,就是读取文件并存放到byte[] 的过程了。
要想path 直接可用,得有两个前提:
(1) _dir事先已经配置好;
(2)name 也应该先经过了处理;
例如:
java.lang.String,会先被转换成java/lang/String.class;
又假如:
_dir 为/usr/local/java/src;
name 为com/test/HelloWorld.class;
path 最终即为/usr/local/java/src/com/test/HelloWorld.class;
加载类文件过程:
instanceKlassHandle ClassLoader::load_classfile(symbolHandle h_name, TRAPS) { ResourceMark rm(THREAD); EventMark m("loading class " INTPTR_FORMAT, (address)h_name()); ThreadProfilerMark tpm(ThreadProfilerMark::classLoaderRegion); stringStream st; // st.print() uses too much stack space while handling a StackOverflowError // st.print("%s.class", h_name->as_utf8()); st.print_raw(h_name->as_utf8()); st.print_raw(".class"); char* name = st.as_string(); // Lookup stream for parsing .class file ClassFileStream* stream = NULL; int classpath_index = 0; { PerfClassTraceTime vmtimer(perf_sys_class_lookup_time(), ((JavaThread*) THREAD)->get_thread_stat()->perf_timers_addr(), PerfClassTraceTime::CLASS_LOAD); ClassPathEntry* e = _first_entry; while (e != NULL) { stream = e->open_stream(name); if (stream != NULL) { break; } e = e->next(); ++classpath_index; } } instanceKlassHandle h(THREAD, klassOop(NULL)); if (stream != NULL) { // class file found, parse it ClassFileParser parser(stream); Handle class_loader; Handle protection_domain; symbolHandle parsed_name; instanceKlassHandle result = parser.parseClassFile(h_name, class_loader, protection_domain, parsed_name, false, CHECK_(h)); // add to package table if (add_package(name, classpath_index, THREAD)) { h = result; } } return h; }
初始化过程:
void ClassLoader::initialize() { ... // lookup zip library entry points load_zip_library(); // initialize search path setup_bootstrap_search_path(); ... }
其中load_zip_library() 即加载zip 类库,从而便于处理jar (可以认为jar 是一种特殊的zip 压缩);而setup_bootstrap_search_path() 即为初始化BootstrapClassLoader 的查找路径,其实就是在初始化上文中每个ClassPathEntry 的_dir。
打开${openjdk}\hotspot\src\share\vm\runtime\os.cpp 文件,找到set_boot_path()方法,会发现Bootstrap Class Loader 已经将所要加载的类库jar 文件固化至代码中了,这些就是JVM 启动时必须要加载的文件。
bool os::set_boot_path(char fileSep, char pathSep) { const char* home = Arguments::get_java_home(); int home_len = (int)strlen(home); static const char* meta_index_dir_format = "%/lib/"; static const char* meta_index_format = "%/lib/meta-index"; char* meta_index = format_boot_path(meta_index_format, home, home_len, fileSep, pathSep); if (meta_index == NULL) return false; char* meta_index_dir = format_boot_path(meta_index_dir_format, home, home_len, fileSep, pathSep); if (meta_index_dir == NULL) return false; Arguments::set_meta_index_path(meta_index, meta_index_dir); // Any modification to the JAR-file list, for the boot classpath must be // aligned with install/install/make/common/Pack.gmk. Note: boot class // path class JARs, are stripped for StackMapTable to reduce download size. static const char classpath_format[] = "%/lib/resources.jar:" "%/lib/rt.jar:" "%/lib/sunrsasign.jar:" "%/lib/jsse.jar:" "%/lib/jce.jar:" "%/lib/charsets.jar:" // ## TEMPORARY hack to keep the legacy launcher working when // ## only the boot module is installed (cf. j.l.ClassLoader) "%/lib/modules/jdk.boot.jar:" "%/classes"; char* sysclasspath = format_boot_path(classpath_format, home, home_len, fileSep, pathSep); if (sysclasspath == NULL) return false; Arguments::set_sysclasspath(sysclasspath); return true; }
本人对C++ 不甚了解,所以无法就代码及流程做更进一步的讲解,希望大家谅解,也欢迎大家补充。
AppClassLoader 与ExtClassLoader
打开sun.misc.Launcher 的源代码,源代码位置在${openjdk}\jdk\src\share\classes\sun\misc\Launcher.java ,ExtClassLoader 与AppClassLoader 是Launcher 类的内部类,代码如下:
/* * The class loader used for loading installed extensions. */ static class ExtClassLoader extends URLClassLoader {} /** * The class loader used for loading from java.class.path. * runs in a restricted security context. */ static class AppClassLoader extends URLClassLoader {}
从继承关系可以向上一直追溯直至java.lang.ClassLoader 实现类,两者的继承关系如下:
java.lang.Object --- java.lang.ClassLoader --- java.security.SecureClassLoader --- java.net.URLClassLoader --- sun.misc.Launcher$ExtClassLoader java.lang.Object --- java.lang.ClassLoader --- java.security.SecureClassLoader --- java.net.URLClassLoader --- sun.misc.Launcher$AppClassLoader
从下图可以更直观的观察出他们的继承关系(引用自网络):
AppClassLoader 与ExtClassLoader 基本都是调用父类的方法,只是在其中加入了一些相应的业务逻辑,所以这里就不分析他们了,在ClassLoader这节中会更详细的分析。
相关推荐
- **标记-整理算法 (Mark-Compact)**:先标记待删除的对象,然后将存活的对象都向一端移动,随后清理掉端边界以外的内存。 #### 四、总结 JVM的工作原理及其垃圾回收机制对于理解和优化Java应用程序至关重要。通过...
在本资源中,我们主要关注的是"JAVA开发的打字软件源程序源码整理"这一主题。这通常意味着我们能够获得一个用Java编程语言编写的打字练习软件的完整源代码,这对于学习Java编程,尤其是对GUI(图形用户界面)设计、...
- 删除节点:找到待删除节点,根据节点子树数量进行不同的处理,之后可能需要进行旋转操作来恢复平衡。 #### 12. TCP如何保证可靠传输?三次握手过程? - **知识点概述**:TCP协议提供了可靠的端到端通信服务。 - ...
然而,不同的JVM实现可能会选择不同的垃圾回收算法,如复制算法、标记-整理算法、标记-压缩算法等,每种算法都有其优缺点和适用场景。 增量式GC(Incremental GC)是为了解决GC运行时导致的程序暂停问题。在标准的...
4. **分代收集算法**:根据对象生命周期的不同,将内存分为年轻代和老年代,年轻代采用复制算法,老年代采用标记整理算法,结合了不同算法的优点,是现代JVM的主流策略。 最后,不同的Java虚拟机实现提供了多种垃圾...
JVM GC(垃圾回收)原理 - **概述**:JVM的GC机制负责自动回收不再使用的对象占用的内存空间,以避免内存泄漏等问题。 - **主要算法**: - **标记-清除**:标记不再使用的对象,然后清理这些标记的对象。 - **...
了解类加载机制有助于理解Java程序的运行原理,可以参考《深入理解Java虚拟机》深入学习。 这些知识点不仅在面试中常见,也是开发中不可或缺的基础,理解并掌握它们能够提升你在Android开发领域的专业水平。
标题《java初学脑图v1.0.1》和描述“整理的一些关于java学习脑图,后续进一步完善,设计简单的java基础以及框架、技术,涉及集合、线程”,意味着本文将聚焦在Java初学者需要掌握的核心概念和基础知识点上,包括面向...
**基本思想**:快速排序采用分治的思想,选择一个基准值,通过一趟排序将待排序的记录分割成独立的两部分,其中一部分记录的关键字均比另一部分的关键字小,然后分别对这两部分继续进行排序,以达到整个序列有序的...
4. **GC算法**:Java中的垃圾收集(Garbage Collection)包括分代收集、标记-清除、复制、标记-整理等算法,用于自动回收不再使用的内存。 5. **死锁检测**:通过资源分配图或检测循环等待条件来识别死锁。 6. **...
- 冒泡排序是一种简单的排序算法,它重复地遍历待排序的数列,一次比较两个元素,如果他们的顺序错误就把他们交换过来。 42. **String and StringBuffer的区别** - `String`是不可变的,每次修改都会创建一个新...