// HelloController.javapackagehello;importorg.springframework.web.bind.annotation.RestController;importorg.springframework.web.bind.annotation.RequestMapping;@RestControllerpublicclassHelloController{@RequestMapping("/")publicStringindex(){return"Greetings from Spring Boot!";}}// Application.javapackagehello;importorg.springframework.boot.SpringApplication;importorg.springframework.boot.autoconfigure.SpringBootApplication;@SpringBootApplicationpublicclassApplication{publicstaticvoidmain(String[]args){SpringApplication.run(Application.class,args);}}
// Generated by Maven Shade PluginManifest-Version:1.0Implementation-Title:gs-spring-bootImplementation-Version:0.1.0Built-By:qianwpImplementation-Vendor-Id:org.springframeworkCreated-By:ApacheMaven3.5.4Build-Jdk:1.8.0_191Implementation-URL:https://projects.spring.io/spring-boot/#/spring-boot-starter-parent/gs-spring-bootMain-Class:hello.Application// Generated by SpringBootLoader PluginManifest-Version:1.0Implementation-Title:gs-spring-bootImplementation-Version:0.1.0Built-By:qianwpImplementation-Vendor-Id:org.springframeworkSpring-Boot-Version:2.0.5.RELEASEMain-Class:org.springframework.boot.loader.JarLauncherStart-Class:hello.ApplicationSpring-Boot-Classes:BOOT-INF/classes/Spring-Boot-Lib:BOOT-INF/lib/Created-By:ApacheMaven3.5.4Build-Jdk:1.8.0_191Implementation-URL:https://projects.spring.io/spring-boot/#/spring-boot-starter-parent/gs-spring-boot
publicclassJarLauncher{...staticvoidmain(String[]args){newJarLauncher().launch(args);}protectedvoidlaunch(String[]args){try{JarFile.registerUrlProtocolHandler();ClassLoadercl=createClassLoader(getClassPathArchives());launch(args,getMainClass(),cl);}catch(Exceptionex){ex.printStackTrace();System.exit(1);}}protectedvoidlaunch(String[]args,Stringmcls,ClassLoadercl){Runnablerunner=createMainMethodRunner(mcls,args,cl);ThreadrunnerThread=newThread(runner);runnerThread.setContextClassLoader(classLoader);runnerThread.setName(Thread.currentThread().getName());runnerThread.start();}}classMainMethodRunner{@Overridepublicvoidrun(){try{Threadth=Thread.currentThread();ClassLoadercl=th.getContextClassLoader();Class<?>mc=cl.loadClass(this.mainClassName);Methodmm=mc.getDeclaredMethod("main",String[].class);if(mm==null){thrownewIllegalStateException(this.mainClassName+" does not have a main method");}mm.invoke(null,newObject[]{this.args});}catch(Exceptionex){ex.printStackTrace();System.exit(1);}}}
又一个问题来了,当 JVM 遇到一个不认识的类,BOOT-INF/lib 目录里又有那么多 jar 包,它是如何知道去哪个 jar 包里加载呢?我们继续看这个特别的 ClassLoader 的源码
privatevoidfindPackage(finalStringname){intlastDot=name.lastIndexOf('.');if(lastDot!=-1){StringpackageName=name.substring(0,lastDot);if(getPackage(packageName)==null){try{definePackage(name,packageName);}catch(Exceptionex){// Swallow and continue}}}}privatefinalHashMap<String,Package>packages=newHashMap<>();protectedPackagegetPackage(Stringname){Packagepkg;synchronized(packages){pkg=packages.get(name);}if(pkg==null){if(parent!=null){pkg=parent.getPackage(name);}else{pkg=Package.getSystemPackage(name);}if(pkg!=null){synchronized(packages){Packagepkg2=packages.get(name);if(pkg2==null){packages.put(name,pkg);}else{pkg=pkg2;}}}}returnpkg;}privatevoiddefinePackage(Stringname,StringpackageName){Stringpath=name.replace('.','/').concat(".class");for(URLurl:getURLs()){try{if(url.getContent()instanceofJarFile){JarFilejf=(JarFile)url.getContent();if(jf.getJarEntryData(path)!=null&&jf.getManifest()!=null){definePackage(packageName,jf.getManifest(),url);returnnull;}}}catch(IOExceptionex){// Ignore}}returnnull;}