如果正在阅读本文的你是一名开发者,那么一定听说过Eclipse和Intellij IDEA这两款可以称之为神器的集成开发环境(IDE)。
Eclipse以其代码开源、插件众多、扩展方便等特点,可以通过配置支持多种开发语言,用户众多。
Intellij IDEA则是分为社区版和专业版。由于是商业公司在维护和更新,因此专业版有更丰富的功能,使用起来更便捷,但如果只使用社区版,则一大块功能不能使用。
当然,我们今天要讨论的不是两个IDE孰优孰劣,而是简要分析下Tomcat在这两个使用量巨大的IDE中是如何工作的。
1. Eclipse
首先来看下Eclipse,通过配置将本地已经下载好的Tomcat添加到Server中。之后,在部署Web应用的时候,就可以选择已经配置好的Tomcat。此时,通过启动的Log,我们来看下其背后的原理。
仔细看上面的蓝框内,是一个类似tomcat的webapps这个应用部署目录的文件路径,而且命令也是wtp.deploy。而wtp也是Eclipse的一个插件。我们再顺着这个路径,在本地打开看下。
这个是上层目录,我们看到基本上除了Tomcat的bin目录和lib目录,剩下的差不多都在了。
其中conf下的文件有这些
可以看出和Tomcat的配置文件一个样子。
使用jps命令查看下具体执行时的参数,就基本理解了
上面这张图中,-D后面是JVM启动时传递的参数,其中catalina.base,是代表Tomcat具体执行时,所有这些配置文件一类的资源具体查找的路径,所有这些配置,在Tomcat启动后,都会被解析得到,所以这些独立位置的配置文件才能生效。
那如果要在一个IDE里跑多个Tomcat该怎么办呢?
有了上面的分析你一定会说,so easy.只需要指定不同的catalina.base,在其对应的路径下配置不同的端口就可以了。
对,影响一个操作系统中是否可以同时存在多个Tomcat,或者引申开来,即多个应用程序,无非是说端口这一类的资源没有被占用即可。
而Tomcat,在server.xml中可以配置Http通道,AJP通道这两个是涉及到端口的,只要端口号使用不同的即可。
另外一个经常被忽略的是,Server下标签下的port,默认是8005,这个也是会冲突的。需要注意下。
有了上面的分析,我们可以大致理清头绪了。在Eclipse里,Tomcat的执行是通过指定不同的catalina.base来实现自定义不同的通道端口的配置,应用文件部署目录等,从而可以在IDE内方便的使用之。
2. Intellij IDEA
下面再来看下Intellij IDEA。在启动Tomcat时,Log一开始就能看到类似于下面的内容:
注意蓝框中对应的内容,和在上面分析Catalina.base基本一致。为了严谨,还是要看一眼这个目录内对应的内容
可以看出,实现思路和上面的分析基本一致,我们不再多谈。
但这里和Eclipse的插件实现的区别是,这里并没有指定wtp.deploy这个类似的属性,所以在上图的webapps目录内并没有我们要运行的应用的内容。那这个时候,IDEA内部是怎么实现应用的部署的呢?
在IDEA里,向tomcat部署一个应用,启动时,其实并不会在本地的tomcat中找到该应用的目录,或者实际运行的目录下有该应用。仔细观察发现,IDEA是通过Tomcat的MBean,动态的向tomcat增加了一个Context,即一个应用。这样直接指定了应用的路径,访问路径等。
例如下面的调用链:
TCP Connection(2)-127.0.0.1@1379 daemon, prio=5, in group ‘RMI Runtime’, status: ‘RUNNING’
at org.springframework.web.context.ContextLoaderListener.
at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:150)
at org.apache.catalina.core.ContainerBase.addChildInternal(ContainerBase.java:725)
at org.apache.catalina.core.ContainerBase.addChild(ContainerBase.java:701)
at org.apache.catalina.core.StandardHost.addChild(StandardHost.java:717)
at org.apache.catalina.startup.HostConfig.manageApp(HostConfig.java:1585)
at org.apache.catalina.mbeans.MBeanFactory.createStandardContext(MBeanFactory.java:463)
at org.apache.catalina.mbeans.MBeanFactory.createStandardContext(MBeanFactory.java:413)
at sun.reflect.NativeMethodAccessorImpl.invoke0(NativeMethodAccessorImpl.java:-1)
at javax.management.remote.rmi.RMIConnectionImpl.doOperation(RMIConnectionImpl.java:1466)
at java.lang.Thread.run(Thread.java:745)
同时还能观察到这样的调用链,因为只有tomcat启动之后,它的MBeanServer才能够被访问。所以如下是启动一个空tomcat。
main@1, prio=5, in group ‘main’, status: ‘RUNNING’
at java.net.DualStackPlainSocketImpl.accept0(DualStackPlainSocketImpl.java:-1)
at java.net.PlainSocketImpl.accept(PlainSocketImpl.java:199)
at java.net.ServerSocket.implAccept(ServerSocket.java:545)
at java.net.ServerSocket.accept(ServerSocket.java:513)
at org.apache.catalina.core.StandardServer.await(StandardServer.java:446)
at java.lang.reflect.Method.invoke(Method.java:483)
at org.apache.catalina.startup.Bootstrap.start(Bootstrap.java:351)
at org.apache.catalina.startup.Bootstrap.main(Bootstrap.java:485)
通过上面简单的分析,IDE中Tomcat的具体执行原理,已经明了。