【0】README
1)本文部分文字描述转自:“深入剖析tomcat”,旨在学习“tomcat(19)Manager应用程序的servlet类” 的相关知识;
2)Manager应用程序用来管理已经部署的web 应用程序;在tomcat7中,manager应用程序位于 %CATALINA_HOME%webapps 目录中;
【1】使用Manager应用程序
1)Manager应用程序中的主servlet类是 ManagerServlet类。在tomcat4中, 该类位于 org.apache.catalina.servlets包下; 而在tomcat5中,该类位于 org.apache.catalina.manager下,是作为 WEB-INF/lib 目录下的一个JAR 文件部署的;
<?xml version="1.0" encoding="ISO-8859-1"?>
<!-- tomcat7 下面的 web.xml -->Licensed to the Apache Software Foundation (ASF) under one or morecontributor license agreements.  See the NOTICE file distributed withthis work for additional information regarding copyright ownership.The ASF licenses this file to You under the Apache License, Version 2.0(the "License"); you may not use this file except in compliance withthe License.  You may obtain a copy of the License athttp://www.apache.org/licenses/LICENSE-2.0Unless required by applicable law or agreed to in writing, softwaredistributed under the License is distributed on an "AS IS" BASIS,WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.See the License for the specific language governing permissions andlimitations under the License.
--><web-app xmlns="http://java.sun.com/xml/ns/javaee"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://java.sun.com/xml/ns/javaeehttp://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"version="3.0"metadata-complete="true"><display-name>Tomcat Manager Application</display-name><description>A scriptable management web application for the Tomcat Web Server;Manager lets you view, load/unload/etc particular web applications.</description><servlet><servlet-name>Manager</servlet-name><servlet-class>org.apache.catalina.manager.ManagerServlet</servlet-class><init-param><param-name>debug</param-name><param-value>2</param-value></init-param></servlet><servlet><servlet-name>HTMLManager</servlet-name><servlet-class>org.apache.catalina.manager.HTMLManagerServlet</servlet-class><init-param><param-name>debug</param-name><param-value>2</param-value></init-param><!-- Uncomment this to show proxy sessions from the Backup manager or aStoreManager in the sessions list for an application<init-param><param-name>showProxySessions</param-name><param-value>true</param-value></init-param>--><multipart-config><!-- 50MB max --><max-file-size>52428800</max-file-size><max-request-size>52428800</max-request-size><file-size-threshold>0</file-size-threshold></multipart-config></servlet><servlet><servlet-name>Status</servlet-name><servlet-class>org.apache.catalina.manager.StatusManagerServlet</servlet-class><init-param><param-name>debug</param-name><param-value>0</param-value></init-param></servlet><servlet><servlet-name>JMXProxy</servlet-name><servlet-class>org.apache.catalina.manager.JMXProxyServlet</servlet-class></servlet><!-- Define the Manager Servlet Mapping --><servlet-mapping><servlet-name>Manager</servlet-name><url-pattern>/text/*</url-pattern></servlet-mapping><servlet-mapping><servlet-name>Status</servlet-name><url-pattern>/status/*</url-pattern></servlet-mapping><servlet-mapping><servlet-name>JMXProxy</servlet-name><url-pattern>/jmxproxy/*</url-pattern></servlet-mapping><servlet-mapping><servlet-name>HTMLManager</servlet-name><url-pattern>/html/*</url-pattern></servlet-mapping><filter><filter-name>SetCharacterEncoding</filter-name><filter-class>org.apache.catalina.filters.SetCharacterEncodingFilter</filter-class><init-param><param-name>encoding</param-name><param-value>UTF-8</param-value></init-param></filter><filter-mapping><filter-name>SetCharacterEncoding</filter-name><url-pattern>/*</url-pattern></filter-mapping><filter><filter-name>CSRF</filter-name><filter-class>org.apache.catalina.filters.CsrfPreventionFilter</filter-class><init-param><param-name>entryPoints</param-name><param-value>/html,/html/,/html/list,/index.jsp</param-value></init-param></filter><filter-mapping><filter-name>CSRF</filter-name><servlet-name>HTMLManager</servlet-name><servlet-name>jsp</servlet-name></filter-mapping><!-- Define a Security Constraint on this Application --><!-- NOTE:  None of these roles are present in the default users file --><security-constraint><web-resource-collection><web-resource-name>HTML Manager interface (for humans)</web-resource-name><url-pattern>/html/*</url-pattern></web-resource-collection><auth-constraint><role-name>manager-gui</role-name></auth-constraint></security-constraint><security-constraint><web-resource-collection><web-resource-name>Text Manager interface (for scripts)</web-resource-name><url-pattern>/text/*</url-pattern></web-resource-collection><auth-constraint><role-name>manager-script</role-name></auth-constraint></security-constraint><security-constraint><web-resource-collection><web-resource-name>JMX Proxy interface</web-resource-name><url-pattern>/jmxproxy/*</url-pattern></web-resource-collection><auth-constraint><role-name>manager-jmx</role-name></auth-constraint></security-constraint><security-constraint><web-resource-collection><web-resource-name>Status interface</web-resource-name><url-pattern>/status/*</url-pattern></web-resource-collection><auth-constraint><role-name>manager-gui</role-name><role-name>manager-script</role-name><role-name>manager-jmx</role-name><role-name>manager-status</role-name></auth-constraint></security-constraint><!-- Define the Login Configuration for this Application --><login-config><auth-method>BASIC</auth-method><realm-name>Tomcat Manager Application</realm-name></login-config><!-- Security roles referenced by this web application --><security-role><description>The role that is required to access the HTML Manager pages</description><role-name>manager-gui</role-name></security-role><security-role><description>The role that is required to access the text Manager pages</description><role-name>manager-script</role-name></security-role><security-role><description>The role that is required to access the HTML JMX Proxy</description><role-name>manager-jmx</role-name></security-role><security-role><description>The role that is required to access to the Manager Status pages</description><role-name>manager-status</role-name></security-role><error-page><error-code>401</error-code><location>/WEB-INF/jsp/401.jsp</location></error-page><error-page><error-code>403</error-code><location>/WEB-INF/jsp/403.jsp</location></error-page><error-page><error-code>404</error-code><location>/WEB-INF/jsp/404.jsp</location></error-page></web-app>
对以上代码的分析(Analysis):
A1)第1个servlet是 org.apache.catalina.manager.ManagerServlet;第2个是 org.apache.catalina.manager.HTMLManagerServlet;
A2)org.apache.catalina.manager.ManagerServlet 的函数列表如下:
protected void list(PrintWriter writer) {if (debug >= 1)log("list: Listing contexts for virtual host '" +host.getName() + "'");writer.println(sm.getString("managerServlet.listed",host.getName()));Container[] contexts = host.findChildren();for (int i = 0; i < contexts.length; i++) {Context context = (Context) contexts[i];String displayPath = context.getPath();if( displayPath.equals("") )displayPath = "/";if (context != null ) {if (context.getAvailable()) {writer.println(sm.getString("managerServlet.listitem",displayPath,"running","" + context.getManager().findSessions().length,context.getDocBase()));} else {writer.println(sm.getString("managerServlet.listitem",displayPath,"stopped","0",context.getDocBase()));}}}}protected void reload(PrintWriter writer, String path) {if (debug >= 1)log("restart: Reloading web application at '" + path + "'");if ((path == null) || (!path.startsWith("/") && path.equals(""))) {writer.println(sm.getString("managerServlet.invalidPath",RequestUtil.filter(path)));return;}String displayPath = path;if( path.equals("/") )path = "";try {Context context = (Context) host.findChild(path);if (context == null) {writer.println(sm.getString("managerServlet.noContext",RequestUtil.filter(displayPath)));return;}// It isn't possible for the manager to reload itselfif (context.getPath().equals(this.context.getPath())) {writer.println(sm.getString("managerServlet.noSelf"));return;}context.reload();writer.println(sm.getString("managerServlet.reloaded", displayPath));} catch (Throwable t) {log("ManagerServlet.reload[" + displayPath + "]", t);writer.println(sm.getString("managerServlet.exception",t.toString()));}}protected void resources(PrintWriter writer, String type) {if (debug >= 1) {if (type != null) {log("resources:  Listing resources of type " + type);} else {log("resources:  Listing resources of all types");}}// Is the global JNDI resources context available?if (global == null) {writer.println(sm.getString("managerServlet.noGlobal"));return;}// Enumerate the global JNDI resources of the requested typeif (type != null) {writer.println(sm.getString("managerServlet.resourcesType",type));} else {writer.println(sm.getString("managerServlet.resourcesAll"));}Class clazz = null;try {if (type != null) {clazz = Class.forName(type);}} catch (Throwable t) {log("ManagerServlet.resources[" + type + "]", t);writer.println(sm.getString("managerServlet.exception",t.toString()));return;}printResources(writer, "", global, type, clazz);}protected void printResources(PrintWriter writer, String prefix,javax.naming.Context namingContext,String type, Class clazz) {try {NamingEnumeration items = namingContext.listBindings("");while (items.hasMore()) {Binding item = (Binding) items.next();if (item.getObject() instanceof javax.naming.Context) {printResources(writer, prefix + item.getName() + "/",(javax.naming.Context) item.getObject(), type, clazz);} else {if ((clazz != null) &&(!(clazz.isInstance(item.getObject())))) {continue;}writer.print(prefix + item.getName());writer.print(':');writer.print(item.getClassName());// Do we want a description if available?writer.println();}}} catch (Throwable t) {log("ManagerServlet.resources[" + type + "]", t);writer.println(sm.getString("managerServlet.exception",t.toString()));}}protected void roles(PrintWriter writer) {if (debug >= 1) {log("roles:  List security roles from user database");}// Look up the UserDatabase instance we should useUserDatabase database = null;try {InitialContext ic = new InitialContext();database = (UserDatabase) ic.lookup("java:comp/env/users");} catch (NamingException e) {writer.println(sm.getString("managerServlet.userDatabaseError"));log("java:comp/env/users", e);return;}if (database == null) {writer.println(sm.getString("managerServlet.userDatabaseMissing"));return;}// Enumerate the available roleswriter.println(sm.getString("managerServlet.rolesList"));Iterator roles = database.getRoles();if (roles != null) {while (roles.hasNext()) {Role role = (Role) roles.next();writer.print(role.getRolename());writer.print(':');if (role.getDescription() != null) {writer.print(role.getDescription());}writer.println();}}}protected void serverinfo(PrintWriter writer) {if (debug >= 1)log("serverinfo");try {StringBuffer props = new StringBuffer();props.append("OK - Server info");props.append("\nTomcat Version: ");props.append(ServerInfo.getServerInfo());props.append("\nOS Name: ");props.append(System.getProperty("os.name"));props.append("\nOS Version: ");props.append(System.getProperty("os.version"));props.append("\nOS Architecture: ");props.append(System.getProperty("os.arch"));props.append("\nJVM Version: ");props.append(System.getProperty("java.runtime.version"));props.append("\nJVM Vendor: ");props.append(System.getProperty("java.vm.vendor"));writer.println(props.toString());} catch (Throwable t) {getServletContext().log("ManagerServlet.serverinfo",t);writer.println(sm.getString("managerServlet.exception",t.toString()));}}protected void sessions(PrintWriter writer, String path, int idle) {if (debug >= 1) {log("sessions: Session information for web application at '" + path + "'");if (idle >= 0)log("sessions: Session expiration for " + idle + " minutes '" + path + "'");}if ((path == null) || (!path.startsWith("/") && path.equals(""))) {writer.println(sm.getString("managerServlet.invalidPath",RequestUtil.filter(path)));return;}String displayPath = path;if( path.equals("/") )path = "";try {Context context = (Context) host.findChild(path);if (context == null) {writer.println(sm.getString("managerServlet.noContext",RequestUtil.filter(displayPath)));return;}Manager manager = context.getManager() ;if(manager == null) {writer.println(sm.getString("managerServlet.noManager",RequestUtil.filter(displayPath)));return;               }int maxCount = 60;int maxInactiveInterval = manager.getMaxInactiveInterval()/60;int histoInterval = maxInactiveInterval / maxCount;if ( histoInterval * maxCount < maxInactiveInterval ) histoInterval++;maxCount = maxInactiveInterval / histoInterval;if ( histoInterval * maxCount < maxInactiveInterval ) maxCount++;writer.println(sm.getString("managerServlet.sessions", displayPath));writer.println(sm.getString("managerServlet.sessiondefaultmax","" + maxInactiveInterval));Session [] sessions = manager.findSessions();int [] timeout = new int[maxCount];int notimeout = 0;int expired = 0;long now = System.currentTimeMillis();for (int i = 0; i < sessions.length; i++) {int time = (int)((now-sessions[i].getLastAccessedTimeInternal())/1000);if (idle >= 0 && time >= idle*60) {sessions[i].expire();idle++;}time=time/60/histoInterval;if (time < 0)notimeout++;else if (time >= maxCount)timeout[maxCount-1]++;elsetimeout[time]++;}if (timeout[0] > 0)writer.println(sm.getString("managerServlet.sessiontimeout","<" + histoInterval, "" + timeout[0]));for (int i = 1; i < maxCount-1; i++) {if (timeout[i] > 0)writer.println(sm.getString("managerServlet.sessiontimeout","" + (i)*histoInterval + " - <" + (i+1)*histoInterval,"" + timeout[i]));}if (timeout[maxCount-1] > 0)writer.println(sm.getString("managerServlet.sessiontimeout",">=" + maxCount*histoInterval,"" + timeout[maxCount-1]));if (notimeout > 0)writer.println(sm.getString("managerServlet.sessiontimeout","unlimited","" + notimeout));if (idle >= 0)writer.println(sm.getString("managerServlet.sessiontimeout","" + idle,"expired " + expired));} catch (Throwable t) {log("ManagerServlet.sessions[" + displayPath + "]", t);writer.println(sm.getString("managerServlet.exception",t.toString()));}}
protected void undeploy(PrintWriter writer, String path) {if (debug >= 1)log("undeploy: Undeploying web application at '" + path + "'");if ((path == null) || (!path.startsWith("/") && path.equals(""))) {writer.println(sm.getString("managerServlet.invalidPath",RequestUtil.filter(path)));return;}String displayPath = path;if( path.equals("/") )path = "";try {// Validate the Context of the specified applicationContext context = (Context) host.findChild(path);if (context == null) {writer.println(sm.getString("managerServlet.noContext",RequestUtil.filter(displayPath)));return;}// Identify the appBase of the owning Host of this Context (if any)String appBase = null;File appBaseDir = null;if (context.getParent() instanceof Host) {appBase = ((Host) context.getParent()).getAppBase();appBaseDir = new File(appBase);if (!appBaseDir.isAbsolute()) {appBaseDir = new File(System.getProperty("catalina.base"),appBase);}}if (!isDeployed(path)) {writer.println(sm.getString("managerServlet.notDeployed",RequestUtil.filter(displayPath)));return;}if (!isServiced(path)) {addServiced(path);try {// Try to stop the context first to be nicer((Lifecycle) context).stop();} catch (Throwable t) {// Ignore}try {File war = new File(getAppBase(), getDocBase(path) + ".war");File dir = new File(getAppBase(), getDocBase(path));File xml = new File(configBase, getConfigFile(path) + ".xml");if (war.exists()) {war.delete();} else if (dir.exists()) {undeployDir(dir);} else {xml.delete();}// Perform new deploymentcheck(path);} finally {removeServiced(path);}}writer.println(sm.getString("managerServlet.undeployed",displayPath));} catch (Throwable t) {log("ManagerServlet.undeploy[" + displayPath + "]", t);writer.println(sm.getString("managerServlet.exception",t.toString()));}}

2)在部署描述符中,还有 security-constraint 元素:(also tomcat7)
 <!-- Define a Security Constraint on this Application --><!-- NOTE:  None of these roles are present in the default users file --><security-constraint><web-resource-collection><web-resource-name>HTML Manager interface (for humans)</web-resource-name><url-pattern>/html/*</url-pattern></web-resource-collection><auth-constraint><role-name>manager-gui</role-name></auth-constraint></security-constraint><security-constraint><web-resource-collection><web-resource-name>Text Manager interface (for scripts)</web-resource-name><url-pattern>/text/*</url-pattern></web-resource-collection><auth-constraint><role-name>manager-script</role-name></auth-constraint></security-constraint><security-constraint><web-resource-collection><web-resource-name>JMX Proxy interface</web-resource-name><url-pattern>/jmxproxy/*</url-pattern></web-resource-collection><auth-constraint><role-name>manager-jmx</role-name></auth-constraint></security-constraint><security-constraint><web-resource-collection><web-resource-name>Status interface</web-resource-name><url-pattern>/status/*</url-pattern></web-resource-collection><auth-constraint><role-name>manager-gui</role-name><role-name>manager-script</role-name><role-name>manager-jmx</role-name><role-name>manager-status</role-name></auth-constraint></security-constraint>
对上述代码的分析(Analysis):
A1)上面的配置意思是: 整个应用程序被限制为只有带manager前缀的用户角色的用户才能访问;
A2)auth-login 元素指明了权限认证使用 BASIC 身份验证方式,只有当用户输入正确的用户名和密码时才能访问受限资源;
 <login-config><auth-method>BASIC</auth-method><realm-name>Tomcat Manager Application</realm-name></login-config>

3)在tomcat中,用户和角色列表都存储在 tomcat-users.xml 中,该文件位于 %CATALINA_HOME%/conf 目录下。因此,要访问 Manager应用程序,必须添加一个 manager角色(如manager-gui)和一个拥有该角色的用户, 例如:
<tomcat-users><role rolename="<span style="font-family: Arial, Helvetica, sans-serif;">manager-gui</span><span style="font-family: Arial, Helvetica, sans-serif;">"/></span><user username="tomcat" password="tomcat" roles="manager-gui"/>
</tomcat-users>
对以上代码的分析(Analysis):通过这个 tomcat-users.xml 文件,就可以以tomcat为用户名,以tomcat为密码来访问 Manager 应用程序了;
【2】ContainerServlet接口(org.apache.catalina.ContainerServlet接口)
1)intro:实现该接口的 servlet类可以访问表示该 servlet实例的StandardWrapper对象,通过访问该Wrapper实例,它也就可以访问表示当前 web 应用程序的Context实例,以及该Context实例内的部署器(StandardHost类的实例)等对象;
public interface ContainerServlet {   public Wrapper getWrapper();public void setWrapper(Wrapper wrapper);
} 

2)Catalina实例会调用实现了 ContainerServlet接口的servlet类的 setWrapper方法:将该引用传递给表示该servlet类 的 StandardWrapper实例;

【3】初始化 ManagerServlet
1)intro:servlet 对象由一个 org.apache.catalina.core.StandardWrapper 实例表示,第一次调用servlet实例时,会调用 StandardWrapper.loadServlet()方法,然后调用 servlet实例的init()方法;
(下面的图片借用自tomcat(5)servlet容器 章节5.7)
【4】列出已经部署的web 应用程序 
1)通过 http://localhost:8888/manager/list 列出已经部署的web 应用程序;
2)上面的URL 会调用 ManagerServlet.list()方法, 源码如下:
 protected void list(PrintWriter writer) { // org.apache.catalina.manager.ManagerServlet.list().if (debug >= 1)log("list: Listing contexts for virtual host '" +deployer.getName() + "'");writer.println(sm.getString("managerServlet.listed",deployer.getName()));String contextPaths[] = deployer.findDeployedApps();for (int i = 0; i < contextPaths.length; i++) {Context context = deployer.findDeployedApp(contextPaths[i]);String displayPath = contextPaths[i];if( displayPath.equals("") )displayPath = "/";if (context != null ) {if (context.getAvailable()) {writer.println(sm.getString("managerServlet.listitem",displayPath,"running","" + context.getManager().findSessions().length,context.getDocBase()));} else {writer.println(sm.getString("managerServlet.listitem",displayPath,"stopped","0",context.getDocBase()));}}}}
对以上代码的分析(Analysis):
A1)list()方法会调用部署器的 findDeployedApps()方法来获取catalina中所有已经部署了的 Context的路径;
A2)然后,对path 数组进行迭代,获取每一个Context,然后检查该 Context是否可用;
A3)对每个可用的 Context:list()方法输出上下文路径,字符串 "running",用户session 的数量以及文档的根路径;
A4)对那些不可用的 Context:list()方法输出上下文路径,字符串 "stopped",0 以及文档的根路径;
【5】启动web 应用程序
1)使用如下URL 来启动某个web 应用程序: http://localhost:8888/manager/start?path=ContextPath , 而ContextPath是想要启动web 应用程序的路径;
2)在接收到该URL后,会调用 ManagerServlet.start() 方法;
protected void start(PrintWriter writer, String path) { //org.apache.catalina.manager.ManagerServlet.start().if (debug >= 1)log("start: Starting web application at '" + path + "'");if ((path == null) || (!path.startsWith("/") && path.equals(""))) {writer.println(sm.getString("managerServlet.invalidPath", path));return;}String displayPath = path;if( path.equals("/") )path = "";try {Context context = deployer.findDeployedApp(path); // highlight line.if (context == null) {writer.println(sm.getString("managerServlet.noContext", displayPath));return;}deployer.start(path);// highlight line.if (context.getAvailable())writer.println(sm.getString("managerServlet.started", displayPath));elsewriter.println(sm.getString("managerServlet.startFailed", displayPath));} //......}
public void start(String contextPath) throws IOException { //org.apahce.catalina.core.StandardHostDeployer.start().// Validate the format and state of our argumentsif (contextPath == null)throw new IllegalArgumentException(sm.getString("standardHost.pathRequired"));if (!contextPath.equals("") && !contextPath.startsWith("/"))throw new IllegalArgumentException(sm.getString("standardHost.pathFormat", contextPath));Context context = findDeployedApp(contextPath);if (context == null)throw new IllegalArgumentException(sm.getString("standardHost.pathMissing", contextPath));host.log("standardHost.start " + contextPath);try {((Lifecycle) context).start(); // highlight line.启动Context容器.} //......}
对以上代码的分析(Analysis):在执行一些检查工作后,start()方法会在 try代码块中调用部署器的 findDeployedApp()方法,该方法返回参数path 指向的Context对象。如果context不是null,start()方法调用部署器的start()方法来启动该 web 应用程序;
【6】关闭web 应用程序 
1)intro:使用如下URL 关闭某个web 应用程序:  http://localhost:8888/manager/stop?path=ContextPath;
2)在接收到该URL后,会调用 ManagerServlet.stop() 方法;
protected void stop(PrintWriter writer, String path) { //org.apache.catalina.manager.ManagerServlet.stop().if (debug >= 1)log("stop: Stopping web application at '" + path + "'");if ((path == null) || (!path.startsWith("/") && path.equals(""))) {writer.println(sm.getString("managerServlet.invalidPath", path));return;}String displayPath = path;if( path.equals("/") )path = "";try {Context context = deployer.findDeployedApp(path); //highlight line.if (context == null) {writer.println(sm.getString("managerServlet.noContext", displayPath));return;}// It isn't possible for the manager to stop itselfif (context.getPath().equals(this.context.getPath())) {writer.println(sm.getString("managerServlet.noSelf"));return;}deployer.stop(path); // highlight line.writer.println(sm.getString("managerServlet.stopped", displayPath));} //.....}
public void stop(String contextPath) throws IOException { //org.apahce.catalina.core.StandardHostDeployer.start().// Validate the format and state of our argumentsif (contextPath == null)throw new IllegalArgumentException(sm.getString("standardHost.pathRequired"));if (!contextPath.equals("") && !contextPath.startsWith("/"))throw new IllegalArgumentException(sm.getString("standardHost.pathFormat", contextPath));Context context = findDeployedApp(contextPath);if (context == null)throw new IllegalArgumentException(sm.getString("standardHost.pathMissing", contextPath));host.log("standardHost.stop " + contextPath);try {((Lifecycle) context).stop(); //highlight line. 停止该容器.} catch (LifecycleException e) {host.log("standardHost.stop " + contextPath + ": ", e);throw new IllegalStateException("standardHost.stop " + contextPath + ": " + e);}}
public synchronized void stop() throws LifecycleException { // org.apache.catalina.core.StandardContext.stop().// Validate and update our current component stateif (!started)throw new LifecycleException(sm.getString("containerBase.notStarted", logName()));if (debug >= 1)log("Stopping");// Notify our interested LifecycleListenerslifecycle.fireLifecycleEvent(BEFORE_STOP_EVENT, null);// Mark this application as unavailable while we shut downsetAvailable(false);// Binding threadClassLoader oldCCL = bindThread();// Stop our filtersfilterStop();// Stop our application listenerslistenerStop();// Finalize our character set mappersetCharsetMapper(null);if ((manager != null) && (manager instanceof Lifecycle)) {((Lifecycle) manager).stop();}// Normal container shutdown processingif (debug >= 1)log("Processing standard container shutdown");// Notify our interested LifecycleListenerslifecycle.fireLifecycleEvent(STOP_EVENT, null);started = false;try {// Stop the Valves in our pipeline (including the basic), if anyif (pipeline instanceof Lifecycle) {((Lifecycle) pipeline).stop();}// Stop our child containers, if anyContainer children[] = findChildren();for (int i = 0; i < children.length; i++) {if (children[i] instanceof Lifecycle)((Lifecycle) children[i]).stop();}// Stop our Mappers, if anyMapper mappers[] = findMappers();for (int i = 0; i < mappers.length; i++) {if (mappers[(mappers.length-1)-i] instanceof Lifecycle)((Lifecycle) mappers[(mappers.length-1)-i]).stop();}// Stop resourcesresourcesStop();if ((realm != null) && (realm instanceof Lifecycle)) {((Lifecycle) realm).stop();}if ((cluster != null) && (cluster instanceof Lifecycle)) {((Lifecycle) cluster).stop();}if ((logger != null) && (logger instanceof Lifecycle)) {((Lifecycle) logger).stop();}if ((loader != null) && (loader instanceof Lifecycle)) {((Lifecycle) loader).stop();}} finally {// Unbinding threadunbindThread(oldCCL);}// Reset application contextcontext = null;// Notify our interested LifecycleListenerslifecycle.fireLifecycleEvent(AFTER_STOP_EVENT, null); if (debug >= 1)log("Stopping complete");}
Conclusion)
C1)本文讲解了如何使用一个专用的接口(ContainerServlet)来创建一个可以访问 Catalina内部类的 servlet类;
C2)可以用来管理已部署web 应用程序的 Manager应用程序展示了如何从 Wrapper对象中获取其他 Catalina对象;

tomcat(19)Manager应用程序的servlet类相关推荐

  1. 求解:Tomcat实例化Servlet类异常?

    求解:运行web项目时报错实例化servlet类异常 初学者,接触不到一年. jdk使用的是jdk13,按照老师要求并未使用集成开发环境,使用的是记事本进行编写,javaBean和Servlet可以正 ...

  2. tomcat容器是如何创建servlet类实例?用到了什么原理?

    当容器启动时,会读取在webapps目录下所有的web应用中的web.xml文件,然后对xml文件进行解析, 并读取servlet注册信息.然后,将每个应用中注册的servlet类都进行加载,并通过反 ...

  3. Servlet第一篇【介绍Servlet、HTTP协议、WEB目录结构、编写入门Servlet程序、Servlet生命周期】...

    tags: Servlet 什么是Serlvet?# Servlet其实就是一个遵循Servlet开发的java类.Serlvet是由服务器调用的,运行在服务器端. 为什么要用到Serlvet? 我们 ...

  4. Servlet第一篇【介绍Servlet、HTTP协议、WEB目录结构、编写入门Servlet程序、Servlet生命周期】

    什么是Servlet? Servlet其实就是一个遵循Servlet开发的java类.Servlet是由服务器调用的,运行在服务器端. 为什么要用到Servlet? 我们编写java程序想要在网上实现 ...

  5. 微信小程序与servlet交互

    学了一段时间的小程序,现在记录一下 之前写了jsp和servlet之间是怎么传输数据和返回数据的,那么微信小程序和servlet是怎么进行交互的,前端怎么进入后台. 开发准备: 微信开发工具:4090 ...

  6. 【JavaWeb】Servlet系列——响应HTML代码、Servlet连接数据库、IDEA开发Servlet程序、Servlet对象的生命周期、GenericServelet适配器模式

    文章目录 10 向浏览器响应一段HTML代码 11 在Servlet中连接数据库,怎么做? 12 在集成开发环境当中开发Servlet程序 12.1 集成开发工具很多,其中目前使用比较多的是: 12. ...

  7. 实例化Servlet类ServletTest异常的处理方式

    实例化Servlet类ServletTest异常的处理方式 Tomcat与Servlet的版本对应关系 例如: Tomcat9.0x对应的Servlet最低版本为Servlet4.0, 如果Tomca ...

  8. 实例化Servlet类异常

    最近初学Servlet,在编写第一个Servlet程序时,遇到了一个令我很苦恼的问题.java代码和XML配置文件中的内容都没有问题,也在Edit Configurations中修改了server的n ...

  9. servlet类中@WebServlet 报错,不变色

    加上javax.servlet.annotation.WebServlet后依然是报错. 解决方法:到安装tomcat的目录下找到lib,里面有个servlet-api.jar复制到目录中,Build ...

最新文章

  1. 【建模必备】遗传算法的定义与生物学基础
  2. 消除 Xcode7 中 directory not found for option 'xxxx' 警告
  3. POJ--2488 A Knight's Journeyb
  4. Android性能优化典范(转)
  5. FZU - 2037 -Maximum Value Problem(规律题)
  6. ppt里quicktime不可用_PPT插入音乐、视频的全部疑难问题,一站式帮你解决!
  7. 电子政务的着力点---紫云舆情服务
  8. 2022电工杯数学建模A题目思路总结分享
  9. matlab系统辨识尝试之详细过程1,系统辨识工具箱教程
  10. logback控制台彩色日志输出
  11. 【公告】CSDN 博客将进行数据库维护
  12. 怎样学构图-平面设计中有哪几项构图
  13. React Fullpage
  14. 看完你就明白的锁系列之自旋锁
  15. 全屏的微粒子3D动画特效
  16. html 苏宁首页,简单实现
  17. Java——动态创建一维数组和二维数组
  18. 深度学习模型的训练(大总结)
  19. storyboard搭建项目_用Storyboard实例化控制器:
  20. 【6】opencv采用映射技术实现鱼眼镜头校正和鱼眼镜头还原全景图。

热门文章

  1. 2021牛客第四场-E Tree Xor-线段树区间异或
  2. Codeforces Round #700 (Div. 1Div. 2)
  3. P1972 [SDOI2009]HH的项链
  4. P4592 [TJOI2018]异或
  5. 牛客网dp专题 数位dp
  6. CF650E-Clockwork Bomb【并查集】
  7. P6378-[PA2010]Riddle【2-SAT】
  8. atcoder E - Greedy Ant(最优解等价+dp)
  9. 4152. [AMPPZ2014]The Captain(稠密图最短路)
  10. 3 年 Java 应该具备的技能体系