• 嵌入式 Tomcat (Embedded Tomcat)


    嵌入式 Tomcat 作为嵌入 Java 应用程序的库, 你可以在 mvnrepository 下载 发行版Jar 以及源码

    https://mvnrepository.com/search?q=embed+tomcat

    作为最基本的依赖, 你需要以下几个库

    • Tomcat Embed Core
    • Tomcat Embed Logging JULI
    • Tomcat Annotations API

    概念

    • Connector: tomcat监听相关的配置对象。
    import org.apache.catalina.connector.Connector;
    
    Connector connector = new Connector();
    conn.setProperty("address", hostname); // 监听地址
    conn.setPort(port); // 监听端口
    
    // 关联到tomcat实例,启动监听
    tomcat.setConnector(connector);
    tomcat.start();
    tomcat.getServer().await();
    
    • 工作目录,绝对路径,该目录下必须有一个"webapps"目录,同时Tomcat还会生成一个work目录,因此建议使用当前目录"."
    tomcat.setBaseDir(new File(".").getAbsolutePath());
    
    // 指定任意工作目录,自动创建webapps目录
        private void configDir(String baseDir) {
            File root = new File(baseDir);
            if (!root.isDirectory() && !root.mkdirs()) {
                throw new RuntimeException("请提供Tomcat工作目录");
            }
            String path4root = root.getAbsolutePath();
            tomcat.setBaseDir(path4root);
            File webapps = new File(path4root + "/webapps");
            if (!webapps.isDirectory() && !webapps.mkdirs()) {
                throw new RuntimeException("无法创建webapps目录");
            }
        }
    
    • 上下文、上下文目录
      添加上下文时,Tomcat会在工作目录下生成目录work/Tomcat/{Hostname}/{contextPath}
      相关代码:
    tomcat.setHostname("my_tomcat");
    
    tomcat.addContext(contextPath, docBase) // docBase:您可以为此目录或WAR文件指定绝对路径名,或者相对于所属主机的appBase目录的相对路径名,除非在server.xml中定义了Context元素,或者docBase不在主机的appBase下,否则不得设置此字段的值
    
    • 上下文映射
    context = tomcat.addContext("", new File(baseDir).getAbsolutePath());
    Tomcat.addServlet(context, "default", new HelloServlet()); // HelloServlet : HttpServlet
    context.addServletMappingDecoded("/", "default");
    

    现在, 让我们把 tomcat 跑起来

    package develon.test;
    
    import java.io.File;
    import java.io.IOException;
    import java.io.Writer;
    
    import javax.servlet.ServletException;
    import javax.servlet.http.HttpServlet;
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;
    
    import org.apache.catalina.Context;
    import org.apache.catalina.LifecycleException;
    import org.apache.catalina.connector.Connector;
    import org.apache.catalina.startup.Tomcat;
    
    public final class Main {
    	File tmpDir = new File("F:\Game\tomcat");
    	Tomcat tomcat = new Tomcat();
    	
    	public static void main(String[] args) throws Throwable {
    		new Main().init();
    	}
    	
    	private void init() throws Throwable {
    		Runtime.getRuntime().addShutdownHook(new Thread(() -> {
    				try {
    					tomcat.destroy();
    				} catch (LifecycleException e) {
    					e.printStackTrace();
    				}
    			})
    		);
    		test();
    	}
    
    	private void test() throws Throwable {
    		tomcat.setBaseDir(tmpDir.getAbsolutePath()); // 设置工作目录
    		tomcat.setHostname("localhost"); // 主机名, 将生成目录: {工作目录}/work/Tomcat/{主机名}/ROOT
    		System.out.println("工作目录: " + tomcat.getServer().getCatalinaBase().getAbsolutePath());
    
    		tomcat.setPort(80);
    		Connector conn = tomcat.getConnector(); // Tomcat 9.0 必须调用 Tomcat#getConnector() 方法之后才会监听端口
    		System.out.println("连接器设置完成: " + conn);
    		
    		// contextPath要使用的上下文映射,""表示根上下文
    		// docBase上下文的基础目录,用于静态文件。相对于服务器主目录必须存在 ({主目录}/webapps/{docBase})
    		Context ctx = tomcat.addContext("", /*{webapps}/~*/ "/ROOT");
    
            Tomcat.addServlet(ctx, "globalServlet", new HttpServlet() {
    			private static final long serialVersionUID = 1L;
    
    			@Override
                protected void service(HttpServletRequest request, HttpServletResponse response)
                        throws ServletException, IOException {
                    response.setCharacterEncoding("UTF-8");
                    response.setContentType("text/plain");
                    response.setHeader("Server", "Embedded Tomcat");
                    try (Writer writer = response.getWriter()) {
                        writer.write("Hello, Embedded Tomcat!");
                        writer.flush();
                    }
                }
            });
            ctx.addServletMappingDecoded("/", "globalServlet");
    
    		tomcat.start();
    		System.out.println("tomcat 已启动");
    		tomcat.getServer().await();
    	}
    
    }
    

    tomcat 嵌入正常, 让我们继续, 如何令 tomcat 加载 Spring Framework ?

    嵌入式 tomcat 集成 Spring 框架

    package develon.tomc;
    
    import java.util.HashSet;
    
    import org.apache.catalina.Context;
    import org.apache.catalina.LifecycleEvent;
    import org.apache.catalina.LifecycleListener;
    import org.apache.catalina.LifecycleState;
    import org.apache.catalina.startup.Tomcat;
    import org.springframework.web.SpringServletContainerInitializer;
    
    public class Main {
    	Tomcat tomcat;
    	
    	{
    		tomcat = new Tomcat();
    //		tomcat.setAddDefaultWebXmlToWebapp(false);
    //		tomcat.noDefaultWebXmlPath();
    	}
    
    	public void run() throws Throwable {
    		tomcat.setBaseDir("F:\Game\tomcat");
    		tomcat.setHostname("localhost");
    		tomcat.setPort(80);
    //		tomcat.enableNaming();
    		
    //		tomcat.getHost().setAutoDeploy(false);
    //		tomcat.getEngine().setBackgroundProcessorDelay(-1);
    		
    		Context ctx = tomcat.addContext("", "ROOT");
    		
    		ctx.addLifecycleListener(new LifecycleListener() {
    			public void lifecycleEvent(LifecycleEvent event) {
    //				System.out.println(event.getLifecycle().getState().name());
    				if (event.getLifecycle().getState() == LifecycleState.STARTING_PREP) {
    					try {
    						new SpringServletContainerInitializer().onStartup(new HashSet<Class<?>>() {
    							private static final long serialVersionUID = 1L;
    							{
    								add(WebAppInitializer.class);
    							}
    						}, ctx.getServletContext());
    					} catch (Throwable e) {
    						e.printStackTrace();
    					}
    				}
    			}
    		});
    		
    //		tomcat.init();
    		tomcat.getConnector();
    		tomcat.start();
    		tomcat.getServer().await();
    	}
    	
    	public static void main(String[] args) throws Throwable {
    		new Main().run();
    	}
    }
    

    其中 WebAppInitializer 是继承 AbstractAnnotationConfigDispatcherServletInitializer 的一个配置类
    由于 AbstractAnnotationConfigDispatcherServletInitializer 继承了 SpringServletContainerInitializer, 所以可以简写为

    		Context ctx = tomcat.addContext("", "ROOT");
    		
    		ctx.addLifecycleListener(new LifecycleListener() {
    			public void lifecycleEvent(LifecycleEvent event) {
    				if (event.getLifecycle().getState() == LifecycleState.STARTING_PREP) {
    					try {
    						new WebAppInitializer().onStartup(ctx.getServletContext());
    					} catch (Throwable e) {
    						e.printStackTrace();
    					}
    				}
    			}
    		});
    

    这种方式好像会报一个错误, 不过可以忽略它, 但是注意这是一个运行时异常, 我们最好捕获 Throwable, 否则程序直接退出了
    (经查, 是由于注射 dispacherServlet 两次造成的, 实际上第一次已经注射完成了)

    java.lang.IllegalStateException: Failed to register servlet with name 'dispatcher'. Check if there is another servlet registered under the same name.
    	at org.springframework.web.servlet.support.AbstractDispatcherServletInitializer.registerDispatcherServlet(AbstractDispatcherServletInitializer.java:90)
    	at org.springframework.web.servlet.support.AbstractDispatcherServletInitializer.onStartup(AbstractDispatcherServletInitializer.java:63)
    	at develon.tomc.Main$1.lifecycleEvent(Main.java:37)
    

    然后我们还能用闭包进一步简化程序, 并且把烦人的栈痕迹删除

    		ctx.addLifecycleListener((LifecycleEvent event) -> {
    			if (event.getLifecycle().getState() == LifecycleState.STARTING_PREP) {
    				try {
    					new WebAppInitializer().onStartup(ctx.getServletContext());
    				} catch (Throwable e) {
    //					e.printStackTrace();
    				}
    			}
    		});
    

    我用 kotlin 简单地封装了一个 EmbeddedTomcat 类

    import org.apache.catalina.startup.Tomcat
    import org.apache.catalina.Context
    import org.apache.catalina.LifecycleState
    
    class EmbeddedTomcat {
    	var tomcat: Tomcat = Tomcat()
    	var ctx: Context? = null
    	
    	init {
    		
    	}
    	
    	/** 初始化嵌入式 tomcat */
    	fun init() {
    		tomcat.setBaseDir("""F:\Game\tomcat""")
    		tomcat.setHostname("localhost")
    		tomcat.setPort(80)
    		
    		ctx = tomcat.addContext("", "ROOT")
    	}
    	
    	/** 开始监听服务 */
    	fun run() {
    		tomcat.getConnector()
    		tomcat.start()
    		tomcat.getServer().await()
    	}
    	
    	/** 启动 Spring 框架, 注射 DispatcherServlet */
    	fun spring() {
    		var tyusya = false
    		ctx?.addLifecycleListener {
    			if (tyusya == false && it.getLifecycle().getState() == LifecycleState.STARTING_PREP) {
    				println("开始注射 -> ${ it.getLifecycle().getState() }")
    				val sctx = ctx?.getServletContext()
    				try {
    					WebAppInitializer().onStartup(sctx)
    					println("完成")
    					tyusya = true
    				} catch(e: Throwable) {
    					println("失败: ${ e.message }")
    				}
    			}
    		}
    	}
    
    	fun spring2() { // 调用了 removeLifecycleListener 移除 tomcat 生命周期监听器
    		ctx?.addLifecycleListener(object : LifecycleListener {
    			override fun lifecycleEvent(it: LifecycleEvent) {
    				if (it.getLifecycle().getState() == LifecycleState.STARTING_PREP) {
    					println("开始注射 DispatcherServlet -> ${ it.getLifecycle().getState() }")
    					try {
    						WebAppInitializer().onStartup(ctx?.getServletContext())
    						println("注射完成")
    						ctx?.removeLifecycleListener(this)
    					} catch(e: Throwable) {
    						println("注射失败: ${ e.message }")
    					}
    				}
    			}
    		})
    	}
    }
    
    fun main() {
    	val tomcat = EmbeddedTomcat()
    	tomcat.init()
    	tomcat.spring()
    	tomcat.run()
    }
    
    import org.springframework.web.servlet.support.AbstractAnnotationConfigDispatcherServletInitializer
    import org.springframework.context.annotation.ComponentScan
    
    @ComponentScan(basePackageClasses = [DefController::class])
    class WebAppInitializer : AbstractAnnotationConfigDispatcherServletInitializer() {
    	override fun getRootConfigClasses() = null
    	override fun getServletMappings() = arrayOf("/")
    	override fun getServletConfigClasses() = arrayOf(WebAppInitializer::class.java)
    }
    
  • 相关阅读:
    分享一个在线生成站点地图SiteMap制作工具
    去掉tppabs冗余代码,怎样批量去掉tppabs代码
    js文字无缝滚动
    史上最全的Win8快捷键大全
    原码, 反码, 补码
    Java [Leetcode 136]Single Number
    拉格朗日对偶
    随机森林与GBDT
    GBDT(Gradient Boosting Decision Tree)算法&协同过滤算法
    Java [Leetcode 165]Compare Version Numbers
  • 原文地址:https://www.cnblogs.com/develon/p/11602969.html
Copyright © 2020-2023  润新知