最近,遇到了一个从未遇到的问题,就是同一tomcat下启动两个web项目,一个是业务系统,也是最主要的部分,另一个是bbs论坛项目,是项目的一个小部分,要求就是要在业务系统登录之后,点击论坛主页就能跳入论坛页面并且同时能携带登录之后的用户名。对于这个问题,网上解答也很多,但是我都大致试了一下,没能解决,最后在同事的帮助下才解决了。
首页要在tomcat下面修改下配置,在tomcat/conf/context.xml中给<Context > 增加一个属性,为<Context crossContext="true">,
并在同一目录中将server.xml修改,其中部分原配置为:
<Host name="localhost" appBase="webapps"
unpackWARs="true" autoDeploy="true"
xmlValidation="false" xmlNamespaceAware="false">
<!-- SingleSignOn valve, share authentication between web applications
Documentation at: /docs/config/valve.html -->
<!--
<Valve className="org.apache.catalina.authenticator.SingleSignOn" />
-->
<!-- Access log processes all example.
Documentation at: /docs/config/valve.html -->
<!--
<Valve className="org.apache.catalina.valves.AccessLogValve" directory="logs"
prefix="localhost_access_log." suffix=".txt" pattern="common" resolveHosts="false"/>
-->
</Host>
修改之后为:
<Host name="localhost" appBase="webapps"
unpackWARs="true" autoDeploy="true"
xmlValidation="false" xmlNamespaceAware="false">
<Context path="/szdai" docBase="E:\\works\\szdai\\WebContent"></Context>
<Context path="/bbs" docBase="E:\\works\\bbs\\WebContent"></Context>
<!-- SingleSignOn valve, share authentication between web applications
Documentation at: /docs/config/valve.html -->
<!--
<Valve className="org.apache.catalina.authenticator.SingleSignOn" />
-->
<!-- Access log processes all example.
Documentation at: /docs/config/valve.html -->
<!--
<Valve className="org.apache.catalina.valves.AccessLogValve" directory="logs"
prefix="localhost_access_log." suffix=".txt" pattern="common" resolveHosts="false"/>
-->
</Host>
<Context>中的docBase为项目的存放路径,path相当于项目的根目录。两个项目名为szdai和bbs
然后就是开始写java代码,由于我们用的是spring MVC,所以在szdai项目中我们做如下处理:
@RequestMapping("bbsHome.htm")
public void bbsHome(HttpServletRequest request,
HttpServletResponse response) throws Exception {
ServletContext szdaiContext = request.getSession().getServletContext();
ServletContext bbsContext = szdaiContext.getContext("/bbs");
MemberInfoEO memberInfo = (MemberInfoEO)request.getSession().getAttribute("member");
request.setAttribute("member", memberInfo);
bbsContext.getRequestDispatcher("/realBBSHome.htm").forward(request, response);
}
将用户信息存放在request中,然后通过转发到bbs项目中。
在bbs中,处理如下:
@RequestMapping("realBBSHome.htm")
public void bbsHome(HttpServletRequest request,HttpServletResponse response) throws IOException{
HttpSession session = request.getSession();
Object memeberInfoObject = request.getAttribute("member");
if(memeberInfoObject!=null){
com.szd.bbs.eo.MemberInfoEO bbsMemberInfo = transformToBBSObject(memeberInfoObject);
session.setAttribute("member", bbsMemberInfo);
Cookie cookie = new Cookie("JSESSIONID",session.getId());
cookie.setMaxAge(2000);
cookie.setPath("/bbs");
response.addCookie(cookie);
}
response.sendRedirect("/bbs");
}
这里有几个关键要注意:由于szdai和bbs是两个不同的项目,所以它们的实体类MemberInfoEO也是不同的,虽然你在bbs里面通过request.getAttribute("member");得到了它的值,但是你也不能获取到用户名,所以你还要通过java反射机制通过得到szdai里面的MemberInfoEO的属性值,然后赋值给bbs里面的MemberInfoEO属性的get方法,然后才能获取到它的用户名。这里还有一点要注意,有可能你存取了session的值,但未必在这个项目的其他方法里面能获取到session。这是为什么呢?由于session的SessionID是保存在客户端的,用Cookie保存的,用户提交数据时,会将这一SessionID提交到服务端,来存取session数据。所以上面所说的如果你不能取到session的值,
那需要我们手动的把sessionID保存在cookie中,操作如下:
Cookie cookie = new Cookie("JSESSIONID",session.getId());
cookie.setMaxAge(2000);
cookie.setPath("/bbs");
response.addCookie(cookie);
如果你的bbs项目是另外用的一个域名,那么你可以通过response.sendRedirect()来重定向。