`
esffor
  • 浏览: 1381643 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Portal开源实现-Liferay的Portlet Session处理(2)

阅读更多
二、LIFERAY中的实现

LIFERAY在构建ActionRequestImpl和RenderRequestImpl时,会设置PORTLET SESSION,如下代码所示:

public RenderRequestImpl(HttpServletRequest req, Portlet portlet,
        CachePortlet cachePortlet,
        PortletContext portletCtx,
        WindowState windowState, PortletMode portletMode,
        PortletPreferences prefs, String layoutId) 
...{
   ...
  _req 
= dynamicReq;
  _portlet 
= portlet;
  _cachePortlet 
= cachePortlet;
  _portalCtx 
= new PortalContextImpl();
  _portletCtx 
= portletCtx;
  _windowState 
= windowState;
  _portletMode 
= portletMode;
  _prefs 
= prefs;
  _ses 
= new PortletSessionImpl(
   _req.getSession(), _portletName, _portletCtx);
   ...
 }

从兰色的部分(  _ses = new PortletSessionImpl(_req.getSession(),_portletName, _portletCtx);  )我们可以看到,这个PORTLET SESSION其实就是PORTAL SYSTEM的 SESSION 对象。
所以无论request调用getSession()或者getPortletSession()都将获取Portal 系统的SESSION 对象,而无论该PORTLET  是或者不是属于PORTAL SYSTEM上下文。而且即使不同PORTAL APPLICATION的PORTLET也将使用同一个SESSION 对象(PORTAL 系统)。
也就是说,对于某一个PORTLET来说,如果有对其的SESSION进行的操作,并没有真正的在该APPLICATION上下文中的SESSION进行操作,而是在PORTAL系统上下文的SESSION中进行操作。

而且LIFERAY提供getPortletSession来获取PortletSession对象,而不是getSession()方法,所以即使getPortletSession()可以获取正确的Session对象,开发人员由于习惯问题,也因使用getSession()而得不到。

另外如果调用request.getSession(true)还可能会出现错误,因为LIFERAY在包含某一个PORTLET内容是,调用PortletRequestDispatcherImpl.include()方法,该方法将生成PortletServletRequest 和PortletServletResponse,请见如下代码:

 

PortletServletRequest portletServletReq = new PortletServletRequest(
    httpReq, reqImpl, pathInfo, queryString, requestURI,
    servletPath);

   PortletServletResponse portletServletRes 
=
    
new PortletServletResponse(
     resImpl.getHttpServletResponse(), resImpl);
而PortletServletRequest的构造函数是如下定义的:
public PortletServletRequest(HttpServletRequest req,
         RenderRequest renderRequest, String pathInfo,
         String queryString, String requestURI,
         String servletPath) 
...{

  
super(req);

  _ses 
= req.getSession();
  _renderRequest 
= renderRequest;
  _pathInfo 
= pathInfo;
  _queryString 
= queryString;
  _requestURI 
= requestURI;
  _servletPath 
= servletPath;
 }


所以其SESSION依然是PORTAL系统上下文的。然后问题就出在这里,PortletServletRequest实现了getSession()方法,但是没有实现getSession(boolen create)方法,如果用户在此阶段调用getSession(true)的话,在某些情况下就会抛出NullPointerException

原因见如下代码(请注意我添加的注释部分)
//ApplicationHttpRequest:  

 

 public HttpSession getSession(boolean create) ...{

        
if (crossContext) ...{
            
            
// There cannot be a session if no context has been assigned yet
            if (context == null)
                
return (null);

            
// Return the current session if it exists and is valid
            if (session != null)
                
return (session.getSession());
     
// 我的注释:这里将获取PORTAL系统的SESSION对象。
            HttpSession other = super.getSession(false);
            
if (create && (other == null)) ...{
                
// First create a session in the first context: the problem is
                
// that the top level request is the only one which can 
                
// create the cookie safely
                other = super.getSession(true);
            }

            
if (other != null...{
                Session localSession 
= null;
                
try ...{
                    
// 我的注释:this context did not have the session with session id. It can just be found in the Portal
                    
// context. So here it will return a null value.
                    localSession =
                        context.getManager().findSession(other.getId());
                    localSession.access(); 
//我的注释:Here, localSession is null. So it throws a NullPointException.
                }
 catch (IOException e) ...{
                    
// Ignore
                }

                
if (localSession == null...{
                    localSession 
= context.getManager().createEmptySession();
                    localSession.setNew(
true);
                    localSession.setValid(
true);
                    localSession.setCreationTime(System.currentTimeMillis());
                    localSession.setMaxInactiveInterval
                        (context.getManager().getMaxInactiveInterval());
                    localSession.setId(other.getId());
                }

                session 
= localSession;
                
return session.getSession();
            }

            
return null;

        }
 else ...{
            
return super.getSession(create);
        }


    }


 


分享到:
评论

相关推荐

    liferay-portal详解

    Liferay Portal 是一个基于 Java 的开源门户平台,提供了一个灵活的框架来构建企业门户和 web 应用程序。本文档将从架构解析、portal 规范、portlet 容器、portlet 生命周期、liferay portal 工作原理等方面对 ...

    Liferay_Portal_门户解决方案

    Liferay Portal 是一个完整的门户解决方案,基于 J2EE 的应用,使用了 EJB 及 JMS 等技术,前台界面部分使用 Struts MVC 框架,基于 XML 的 portlet 配置文件可以自由地动态扩展,使用了 Web 服务来支持一些远程信息...

    liferay跨组件间通信实例

    Liferay作为一个开源的企业级门户平台,提供了多种方式来实现组件间的通信,以支持复杂的应用场景。本实例将深入探讨如何在Liferay中进行跨组件通信,并以`test-portlet`为例进行详细讲解。 首先,我们需要理解...

    Liferay 6.0.5 portal.properties 配置文件详解

    Liferay 是一个开源的企业级门户平台,用于构建和管理企业网站、社交网络和工作流程。在 Liferay 中,`portal.properties` 文件是一个至关重要的配置文件,它定义了系统的全局属性和行为。对于 Liferay 6.0.5 版本,...

    liferay开发指南6.0

    Liferay是一款开源的企业级门户平台,能够实现跨系统的资源整合,为用户提供一站式的解决方案。它具备丰富的特性和功能,支持多种认证方式(如LDAP、SQL)、个性化门户布局自定义、多语言支持、主流J2EE应用服务器...

    LiferayPortal

    Liferay Portal的部署可以通过手动或自动化工具(如Ant脚本)实现,确保自定义Portlet能够顺利集成到门户环境中。 - **手动部署**:涉及复制Portlet包至指定目录、更新部署描述文件等步骤。 - **Ant自动部署**:...

    liferay开发文档

    这部分包括了创建Portlet类的基础知识、Portlet开发中的重要基类GenericPortlet、Portlet标签的使用以及Portal对象(如Request、Response、PortletConfig、Session和Preference对象)的介绍。 在编写自定义Portlet...

    lifeary开发

    2. 在portlet的web.xml中配置Struts2的前端控制器Filter,以启用Struts2处理请求。 三、使用Hibernate与Spring 在Lifeary中,通常会结合Hibernate进行数据持久化,与Spring进行事务管理和依赖注入。以下是具体操作...

    wicket指南

    5. **JSR 168/286 支持**:Wicket 兼容portlet标准,可以无缝集成到portlet容器,如Apache Pluto或Liferay Portal。 **二、Wicket 开发流程** 1. **创建页面类**:在Wicket中,每个页面都有一个对应的Java类,类中...

Global site tag (gtag.js) - Google Analytics