任务调度---执行两次 问题原因总结

且不论很是热火的解决方案, 就表面感觉而言, 一个程序会执行两次, 那么必需是调用了两次程序。但从我们的配置上看来只定义了一次:

Xml代码 复制代码  收藏代码
<Host name="lanvideo.cn"  appBase="lanvideo"
     unpackWARs="true" autoDeploy="true"
     xmlValidation="false" xmlNamespaceAware="false">
     <Context path="" docBase="LanVideo" />
</Host>


目录结构:

Sh代码 复制代码  收藏代码

  

  1. $ ls lanvideo   
  2. LanVideo.war 

 

$ ls lanvideo
LanVideo.war


但从实际的启动效果来看,情况却非如此:

Sh代码 复制代码  收藏代码
 
  1. $ bin/startup.sh   
  2. $ ls lanvideo   
  3. LanVideo.war   
  4. LanVideo   
  5. ROOT 


实际解压出现LanVideo和ROOT两个目录, 并且目录里内容相同, 这就不难解释Quartz会执行两次的问题了。

问题发现了, 怎么解决呢?
先看看文档上的描述:
http://localhost:8080/docs/virtual-hosting-howto.html
这是采用了在conf/Catalina/lanvideo.cn下用ROOT.xml替代server.xml的Context的方式:
删除server.xml中Context的定义:

Xml代码 复制代码  收藏代码

 

<Host name="lanvideo.cn"  appBase="lanvideo"
     unpackWARs="true" autoDeploy="true"
     xmlValidation="false" xmlNamespaceAware="false">
</Host>


新建conf/Catalina/lanvideo.cn/ROOT.xml

Xml代码 复制代码  收藏代码
<?xml version='1.0' encoding='utf-8'?>
<Context docBase="LanVideo">
</Context>


删除已经发布的LanVideo和ROOT文件夹后重启服务器。

Sh代码 复制代码  收藏代码
 



貌似解决这个问题了。可是访问是却是404。

Sh代码 复制代码  收藏代码
 
  1. $ curl -i http://lanvideo.cn:8080  
  2. HTTP/1.1 404 Not Found   
  3. Server: Apache-Coyote/1.1  
  4. Content-Type: text/html;charset=utf-8  
  5. Content-Length: 952  
  6. Date: Tue, 28 Dec 2010 08:59:57 GMT

看来官方的指南也解决不了问题。访问还是会依赖ROOT文件夹里的程序。

走到这一步说明说明正道不可走, 走偏门:
更改配置:
Xml代码 复制代码  收藏代码

  

<Host name="lanvideo.cn"  appBase="lanvideo"
     unpackWARs="true" autoDeploy="true"
     xmlValidation="false" xmlNamespaceAware="false">
     <Context path="" docBase="LanVideo.war" />
</Host>


为什么是引向war?因为Context的unpackWAR属性会将docBase指向的war自动发布到path指定的地方。所以,利用这点,可以让它正常发布到ROOT文件夹下。

然后,在lanvideo下建立空的文件夹LanVideo,通过tomcat的发布检查,避开自动发布到LanVideo文件夹,只保留ROOT文件夹里的内容。
如此:

Sh代码 复制代码  收藏代码
 
  1. $ ls lanvideo   
  2. LanVideo.war   
  3. LanVideo   
  4. ROOT   
  5. $ ls lanvideo/LanVideo   
  6. $ ls lanvideo/ROOT   
  7. ...   
  8. $ curl -I http://lanvideo.cn:8080  
  9. HTTP/1.1 200 OK   
  10. Server: Apache-Coyote/1.1  
  11. Set-Cookie: JSESSIONID=7B057642D076C8B9F33DA7A4401EE288; Path=/   
  12. Content-Type: text/html;charset=utf-8  
  13. Content-Length: 3915  
  14. Date: Tue, 28 Dec 2010 09:34:19 GMT  


     
     
    搞定!


    附:
    网上很多方案是说更改appBase, 我猜那些人也是这样找到的偏门吧, 不过妄称appBase设定错误, 我可不敢苟同。

    --------------------------------------------------------第二---------------------------------------------------------------

    前提:

        1、存在某个应用:hello

        2、该应用存放路径:D:\apache-tomcat-5.5.17\webapps\hello

        3、Tomcat的server.xml部分配置信息如下:

    Java代码 复制代码
    ...
    <Context path="" reloadable="true" docBase="hello"/>
    ...



        (这么做的目的就是为了将hello应用设为根,访问IP时就直接访问该应用)

        4、hello下有一个 servlet(目的是为了系统在启动时执行某些初始化的操作),该servlet里存在init()方法,目前该方法里只打印“hello word”字符串;

         5、hello 应用的 web.xml 部分配置信息如下:

    Java代码 复制代码
    <servlet>
        <servlet-name>HelloServlet</servlet-name>
        <servlet-class>HelloServlet</servlet-class>
        <load-on-startup>1</load-on-startup>
    </servlet>


    问题:
        在启动 tomcat5 时,控制台窗口会打印出两行“hello word”字符串?

    问题原因:
        因为hello应用本来就放在Tomat的默认webapp目录下(tomcat在启动时肯定会加载1次),然后又在server.xml中做了配置,为了达到访问根就可以访问hello应用(这样tomcat就又加载1次);结果,Tomcat就会加载两次。

    解决办法:
        办法1、不要将 hello 应用放在Tomat的默认webapp目录下,把它移出去,然后在server.xml中修改docBase的值为hello应用在新位置的绝对路径就可以了;
        办法2、删除掉server.xml中 Context 的手动配置,这样就不会加载两次,因为hello应用在webapp下,所以在访问时,就只能是:http://ip:port/hello 这样来访问了。


    文章原地址:
    http://hi.baidu.com/wpf512/blog/item/3b941786aa12b73667096e0f.html

    转载“tomcat中三种部署项目的方法”
    http://252401762.iteye.com/blog/287241


    第一种方法:在tomcat中的conf目录中,在server.xml中的,<host/>节点中添加:
    <Context path="/hello" docBase="D:eclipse3.2.2forwebtoolsworkspacehelloWebRoot" debug="0" privileged="true">
    </Context>
    至于Context 节点属性,可详细见相关文档。



    第二种方法:将web项目文件件拷贝到webapps 目录中。



    第三种方法:很灵活,在conf目录中,新建 Catalina(注意大小写)\localhost目录,在该目录中新建一个xml文件,名字可以随意取,只要和当前文件中的文件名不重复就行了,该xml文件的内容为:
    <Context path="/hello" docBase="D:eclipse3.2.2forwebtoolsworkspacehelloWebRoot" debug="0" privileged="true">
    </Context>



    第3个方法有个优点,可以定义别名。服务器端运行的项目名称为path,外部访问的URL则使用XML的文件名。这个方法很方便的隐藏了项目的名称,对一些项目名称被固定不能更换,但外部访问时又想换个路径,非常有效。

    第2、3还有优点,可以定义一些个性配置,如数据源的配置等。

    还有一篇 详细的



    此处主要讲述Tomcat部署发布JSP应用程序的三种方法
        1、直接放到Webapps目录下
         Tomcat的Webapps目录是Tomcat默认的应用目录,当服务器启动时,会加载所有这个目录下的应用。也可以将JSP程序打包成一个 war包放在目录下,服务器会自动解开这个war包,并在这个目录下生成一个同名的文件夹。一个war包就是有特性格式的jar包,它是将一个Web程序的所有内容进行压缩得到。具体如何打包,可以使用许多开发工具的IDE环境,如Eclipse、NetBeans、ant、JBuilder等。也可以用 cmd 命令:jar -cvf applicationname.war package.*;
    甚至可以在程序执行中打包:
    try{
    string strjavahome = system.getproperty("java.home");
    strjavahome = strjavahome.substring(0,strjavahome.lastindexof(\\))+"\\bin\\";
    runtime.getruntime().exec("cmd /c start "+strjavahome+"jar cvf hello.war c:\\tomcat5.0\\webapps\\root\\*");

    catch(exception e){system.out.println(e);}

         webapps这个默认的应用目录也是可以改变。打开Tomcat的conf目录下的server.xml文件,找到下面内容:
    <Host name="localhost" debug="0" appBase="webapps" unpackWARs="true" autoDeloy="true" xmlValidation="falase" xmlNamespaceAware="false">

       2、在server.xml中指定
         在Tomcat的配置文件中,一个Web应用就是一个特定的Context,可以通过在server.xml中新建Context里部署一个JSP应用程序。打开server.xml文件,在Host标签内建一个Context,内容如下。
    <Context path="/myapp" reloadable="true" docBase="D:\myapp" workDir="D:\myapp\work"/>
         其中path是虚拟路径,docBase是JSP应用程序的物理路径,workDir是这个应用的工作目录,存放运行是生成的于这个应用相关的文件。

       3、创建一个Context文件
         以上两种方法,Web应用被服务器加载后都会在Tomcat的conf\catalina\localhost目录下生成一个XML文件,其内容如下:
    <Context path="/admin" docBase="${catalina.home}/server/webapps/admin" debug="0" privileged="true"></Context>
    可以看出,文件中描述一个应用程序的Context信息,其内容和server.xml中的Context信息格式是一致的,文件名便是虚拟目录名。您可以直接建立这样的一个xml文件,放在Tomcat的conf\catalina\localhost目录下。例子如下:
    注意:删除一个Web应用同时也要删除webapps下相应的文件夹祸server.xml中相应的Context,还要将Tomcat的conf
    \catalina\localhost目录下相应的xml文件删除。否则Tomcat仍会岸配置去加载。。。

    --------------------------------------------------------第三---------------------------------------------------------------------------

    Web工程启动的时候,我要注册一个定时任务,每天的凌晨触发,所以使用了Quartz框架,其他没有任何外部包的依赖。
    在Servlet的init方法里面启动这个schedule。 Job的run方法中每次执行的时候都会打info级别的日志处理。
    然后在web.xml文件中注册这个Servlet,设置了load-on-startup的值为3

    Xml代码 复制代码


    在IDE里面启动tomcat是没有问题的,但是将这个工程打包发布到服务器的tomcat目录下面去后再启动,每次执行Job的时候会打两次日志,执行两次run方法。

    在tomcat/conf/server.xml中的配置是这样的:

    Xml代码 复制代码



    tomcat的加载顺序是
    1)server.xml中<Host>/<Context>
    2) conf/<engine name>/<host name>/context xml
    3) webapps/war
    4) webapps/文件夹

    host下每个context的name唯一,加载序列中如果前面加载了相同名称的context,则不会在后面继续加载,你这个要么<Host>/<Context>中path配置与webapps里的文件夹同名,要么将这个文件夹放webapps外的其他地方,然后以绝对路径指过去

     

    ---------------------------------------------------------------第四-------------------------------------------------

    是TOMCAT的配置文件server.xml配置不当引起的,

    请看下面这段配置就是错误的:

    <Host name="www.***.com" debug="0" appBase="webapps" unpackWARs="true" autoDeploy="true" xmlValidation="false" xmlNamespaceAware="false">
    <Logger className="org.apache.catalina.logger.FileLogger" directory="logs" prefix="localhost_log." suffix=".txt" timestamp="true" />
    <Context path="" docBase="/tomcat/webapps/***" debug="1" />
    <Context path="/***2" docBase="/tomcat/webapps/***2" debug="1" />
    </Host>


    正确的配置如下

    <Host name="www.***.com" debug="0" appBase="" unpackWARs="true" autoDeploy="true" xmlValidation="false" xmlNamespaceAware="false">
    <Logger className="org.apache.catalina.logger.FileLogger" directory="logs" prefix="localhost_log." suffix=".txt" timestamp="true" />
    <Context path="" docBase="/tomcat/webapps/***" debug="1" />
    <Context path="/***2" docBase="/tomcat/webapps/***2" debug="1" />
    </Host>
    这两段的区别是第二段去除了appBase="webapps"中的webapps变成了appBase="",因为web应用程序都是放在webapps这个目录下的,如果不把“webapps“去掉,这里会调用一次quartz的任务调度,在接下来的“<Context path”中又会调用一次quartz的任务调度,所以就重复了2次

     

     以上是服务器配置问题 如果是配置问题 则不管你采用何种技术来实现定时任务都会有这个问题  如果是spring  quartz 则有可能是因为项目中集成了不同的技术 在web.xml 中配置了两个监听 加载了两次spring的配置文件。。。。。

     

    评论
    添加红包

    请填写红包祝福语或标题

    红包个数最小为10个

    红包金额最低5元

    当前余额3.43前往充值 >
    需支付:10.00
    成就一亿技术人!
    领取后你会自动成为博主和红包主的粉丝 规则
    hope_wisdom
    发出的红包
    实付
    使用余额支付
    点击重新获取
    扫码支付
    钱包余额 0

    抵扣说明:

    1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
    2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

    余额充值