how tomcat works 总结 三

时间:2023-03-09 20:26:35
how tomcat works 总结 三

第七章 日志记录器

第 7 章包括日志,该组件是用来记录错误信息和其他信息的。

这一章比较简单,类图如下:

how tomcat works 总结 三

根据名字我想大家都能猜出来三个实现类都是做什么的,一个按常规输出到控制台,一个按错误模式输出到控制点,一个输出到文件。

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 应用所需的其他类。这章还展示了如何实现应用的重新加载。

类图如下:

how tomcat works 总结 三

为了自定义载入器是为了以下三个目标:

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等等,然后在重新开启#