第七章 日志记录器
第 7 章包括日志,该组件是用来记录错误信息和其他信息的。
这一章比较简单,类图如下:
根据名字我想大家都能猜出来三个实现类都是做什么的,一个按常规输出到控制台,一个按错误模式输出到控制点,一个输出到文件。
Logger类中有一个参数,verbosity,用来表示日志等级,默认为ERROR。
public static final int FATAL = Integer.MIN_VALUE;
在输出日志的时候,只有给定的message的verbosity小于默认的才会输出。
在Bootstrap启动类设置FileLogger
Bootstrap.java System.setProperty("catalina.base", System.getProperty("user.dir")); FileLogger logger = new FileLogger(); logger.setPrefix("FileLog_"); logger.setSuffix(".txt"); logger.setTimestamp(true); logger.setDirectory("webroot"); context.setLogger(logger); 在使用的时候,调用log(message)即可 private void log(String message) { Logger logger = connector.getContainer().getLogger(); if (logger != null) logger.log(threadName + " " + message); }
第八章 载入器
第 8 章解释了加载器(loader)。加载器是一个重要的 Catalina 模块,负责加载 servlet 和一个 web 应用所需的其他类。这章还展示了如何实现应用的重新加载。
类图如下:
为了自定义载入器是为了以下三个目标:
1在载入类中指定某些规则;
2缓存已经载入的类;
3实现类的预载入;
一个一个说:
第一条,在载入类中指定某些规则
我们设定了两个数组变量,triggers与packageTriggers,这个两个数组里面放置不能访问的类与包。当我们加载一个类的时候提前检查类名,包名。这里有一个问题,我们加载类的顺序是:
所有加载过的类都要进行缓存,所以首先需要检查本地缓存。
· 如果无法再本地缓存找到类,使用 java.langClassLoader 类的 findLoaderClass 方法在缓存查找类、
· 如果在两个缓存中都无法找到该类,使用系统的类加载器避免从 J2EE 类中覆盖来的 web 应用程序。
· 如果使用了安全管理器,检查该类是否允许加载,如果该类不允许加载,则抛出 ClassNotFoundException 异常。
· 如果要加载的类使用了委派标志或者该类属于 trigger 包中,使用父加载器来加载类,如果父加载器为 null,使用系统加载器加载。
· 从当前的源中加载类
· 如果在当前的源中找不到该类并且没有使用委派标志,使用父类加载器。如果父类加载器为 null,使用系统加载器
· 如果该类仍然找不到,抛出 ClassNotFoundException 异常
就像前面几章
http://localhost:8080/Primitive
请求的PrimitiveServlet不会被第二步的类加载器加载。为什么?请看看类加载器的加载路径范围。
第二条 缓存已经裁人的类
还是说上面的那个类,它会通过
clazz = findClass(name);来加载,如果加载成功class会存储到resouEntries里,如果失败就存储到notFoundResources中。
第三条 实现类的预载入
这条看的不是很懂,DirContext也看的不是很明白。
关于动态载入:
WebLoader.java run方法(部分) while (!threadDone) { // Wait for our check interval threadSleep(); if (!started) break; try { if (!classLoader.modified()) continue; } catch (Exception e) { continue; } // Handle a need for reloading notifyContext(); break; }
modified方法会检查已经载入的类是否被修改,是否修改的表示就是最后修改的时间是否改变!
若有类被修改,调用notifyContext,在notifyContext中会新建一个线程来调用container的reload方法()?为什么要新启一个线程还用说么?
reload方法()会首先关掉子容器,loader等等,然后在重新开启#