`

有关 EJB3,无状态会话Bean 与 Spring 依赖注入 Singleton 的困惑!

阅读更多
最近我在学习 EJB3。关于它的无状态会话Bean,我有一些困惑。

按照 EJB3 规范,容器会为每一种无状态会话 Bean 创建一定数量的实例,然后将它们保存在一个缓冲池(pool) 里面,然后根据调用的情况,将它们在调用者之间分配。书上特别强调了,因为这些 bean 都是无状态的,所以自由度很高。比如说,一个对象调用了无状态会话 Bean 的方法A,执行了一段时间;然后,又调用了同一个 Bean 的方法B。但实际上,它调用的是实例 1 的方法A,和实例 2 的方法B。只不过这一切,对调用者来说都是透明的。

这就是我的困惑所在。我们都知道,可以用无状态的 Singleton 来封装一些领域服务。比如说,UserService,就可以是一个 Singleton class。在实际使用时,可以用 IoC 容器的依赖注入方法得到它的一个(全 JVM 唯一的)Singleton 实例,然后再调用它里面的方法,比如说:
public class ... {
    private UserService userService;
    public void setUserService(UserService v) {
        // IoC 容器将会注入 UserService 的一个实现的实例,
        // 这也将是整个 JVM 共享的一个实例。
        userService = v;
    }

    ...

    public boolean isValidUser(String username, String password) {
        User user = userService.processLogin(username, password);
        return (user != null);
    }

}


一切都很基础。

我现在主要的问题就是,EJB 容器为无状态会话 Bean 创建多个实例并缓冲,到底有什么特别的好处?既然他们是无状态的,为什么不可以整个系统共享一个实例呢?难道说,多个实例的执行效率更高?这似乎颠覆了我个人的 JVM 基本概念.....

请各位指教。谢谢。
分享到:
评论
19 楼 非典型程序员 2007-02-07  
jamesby 写道
(全 JVM 唯一的)Singleton 实例肯定是不对的,准确地说应该是一个ClassLoader一个实例,当然,父ClassLoader 已经load了一个实例,则子不需要Load实例


请问,你所说的,是 Spring 的做法吗?

而且,在缺省条件下,不是整个 JVM 只有一个 ClassLoader 吗?
18 楼 非典型程序员 2007-02-07  
retreat 写道
robbin 写道
无状态SessionBean确实没有必要搞对象池,这一点Rod Johnson在《without EJB》里面有很详细的论述。


无状态Sb本身就是借助对象池来体现“无状态”,如果没有Pool,则创建出来的对象必然有状态,只是在于是否被捕获并且表现出来。我个人觉得无状态sb用内存换速度,一定程度上减少创建对象的开销,但是现在内存不值钱,而且对象池的稳定性的确让人忧虑,本人写的大多数对象池都问题多多!


我不这样认为。相反,我觉得:
1,应根据其具体需要来确定一个bean是否有状态;
2,有状态bean需要对象池,因为我们肯定会同时需要多个有状态bean,所以保持一个池可以提高效率;
3,无状态bean不需要对象池,JVM 唯一的 singleton 就可以了,并不会增加任何开销。
17 楼 非典型程序员 2007-02-07  
jianfeng008cn 写道


Singleton 并不是这样的吧 他只是在 servletcontext里面保存了这么一份 beans
每个request过来的时候,使用的是threadlocal
threadlocal 可以说是“拷贝”了 servletcontext里面的beans
可以说是一个副本
这样可以保证多个request使用的beans互相不影响(具体可以理解一下threadlocal )

threadlocal 的实现本身就类似 对象池 ,不过这个东西是语言支持的,而不是自己来实现的。
你说的synchronize 在一般bean上都不会起作用,原理也是这样 用的根本不是同一个对象(当然static得注意),除非在同一个request中存在多次调用产生 synchronize 现象。

引用
(全 JVM 唯一的)Singleton 实例
是否你的理解出偏差了呢?



我一向认为,Spring对于(缺省状态下为Singleton模式的)Bean,都是在整个 JVM 里面保持唯一一个实例的。

可能是我一直以来都搞错了吧。因为我没有真正去仔细了解Spring对 Singleton Bean 是如何提供支持的。
16 楼 非典型程序员 2007-02-07  
robbin 写道
无状态SessionBean确实没有必要搞对象池,这一点Rod Johnson在《without EJB》里面有很详细的论述。


多谢资深发言指教。还要多问一句,“无状态SessionBean确实没有必要搞对象池”,其原因是不是就是我提出的那个呢?
15 楼 jamesby 2007-02-07  
(全 JVM 唯一的)Singleton 实例肯定是不对的,准确地说应该是一个ClassLoader一个实例,当然,父ClassLoader 已经load了一个实例,则子不需要Load实例
14 楼 retreat 2007-02-06  
robbin 写道
无状态SessionBean确实没有必要搞对象池,这一点Rod Johnson在《without EJB》里面有很详细的论述。


无状态Sb本身就是借助对象池来体现“无状态”,如果没有Pool,则创建出来的对象必然有状态,只是在于是否被捕获并且表现出来。我个人觉得无状态sb用内存换速度,一定程度上减少创建对象的开销,但是现在内存不值钱,而且对象池的稳定性的确让人忧虑,本人写的大多数对象池都问题多多!
13 楼 jianfeng008cn 2007-02-06  
引用

这就是我的困惑所在。我们都知道,可以用无状态的 Singleton 来封装一些领域服务。比如说,UserService,就可以是一个 Singleton class。在实际使用时,可以用 IoC 容器的依赖注入方法得到它的一个(全 JVM 唯一的)Singleton 实例,然后再调用它里面的方法,比如说:


Singleton 并不是这样的吧 他只是在 servletcontext里面保存了这么一份 beans
每个request过来的时候,使用的是threadlocal
threadlocal 可以说是“拷贝”了 servletcontext里面的beans
可以说是一个副本
这样可以保证多个request使用的beans互相不影响(具体可以理解一下threadlocal )

threadlocal 的实现本身就类似 对象池 ,不过这个东西是语言支持的,而不是自己来实现的。
你说的synchronize 在一般bean上都不会起作用,原理也是这样 用的根本不是同一个对象(当然static得注意),除非在同一个request中存在多次调用产生 synchronize 现象。

引用
(全 JVM 唯一的)Singleton 实例
是否你的理解出偏差了呢?










12 楼 robbin 2007-02-06  
无状态SessionBean确实没有必要搞对象池,这一点Rod Johnson在《without EJB》里面有很详细的论述。
11 楼 非典型程序员 2007-02-06  
jamesby 写道
servlet是线程池实现的典型,ejb是对象池实现的典型,至于对象池有哪些好处?搞不清楚,不知道它为什么会提高性能?


其实,对于有状态 bean,我还是理所当然的认为对象池可以提高效率的,因为1,系统确确实实需要有状态 bean 的多个实例(就因为他们是有状态的);2,创建一个 bean 的开销可能很大。

但对于无状态 bean,我的疑问继续成立
10 楼 jamesby 2007-02-06  
servlet是线程池实现的典型,ejb是对象池实现的典型,至于对象池有哪些好处?搞不清楚,不知道它为什么会提高性能?
9 楼 非典型程序员 2007-02-06  
jamesby 写道
Stateless Session Bean设计池的概念应该是不同的调用端同一时间访问的一定不是同一个Bean实例!


对,我也认为,调用端会随机的访问不同的bean实例。

但我不认为这样效率会更高些。否则的话,Spring 也应该隐含的为那些 service bean 提供缓冲池,而不是简单的用 singleton 来解决问题。
8 楼 jamesby 2007-02-06  
Stateless Session Bean设计池的概念应该是不同的调用端同一时间访问的一定不是同一个Bean实例!
7 楼 jamesby 2007-02-06  
非典型程序员 写道
ahuaxuan 写道
ejb3.0最终没有能颠覆它原来的线程模型,其中缘由不是三言两语能够说清楚的,但是

引用
这似乎颠覆了我个人的 JVM 基本概念
似乎太夸张了吧。


哈哈,初级会员嘛,夸张在所难免。

个人的 JVM 基本概念,就是 JVM 会(在不同线程中)同时执行一个对象的非 synchronized 方法的多个拷贝,不需要任何额外的等待时间。

基于这个概念,我认为,使用无状态的 singleton 类来当 XyzService,是最佳的选择,不会有任何效率上的不良影响。

也因此,我不明白 EJB 为无状态 session bean 提供对象池的意义何在。当然,我以前没有做过 EJB 1, 2 的项目,所以对其历史不了解。希望资深的各位大侠不吝赐教!
一说倒还真是的,以前只知道有一个线程池的概念,但是为什么需要它不是很清楚!
6 楼 非典型程序员 2007-02-06  
ahuaxuan 写道
ejb3.0最终没有能颠覆它原来的线程模型,其中缘由不是三言两语能够说清楚的,但是

引用
这似乎颠覆了我个人的 JVM 基本概念
似乎太夸张了吧。


哈哈,初级会员嘛,夸张在所难免。

个人的 JVM 基本概念,就是 JVM 会(在不同线程中)同时执行一个对象的非 synchronized 方法的多个拷贝,不需要任何额外的等待时间。

基于这个概念,我认为,使用无状态的 singleton 类来当 XyzService,是最佳的选择,不会有任何效率上的不良影响。

也因此,我不明白 EJB 为无状态 session bean 提供对象池的意义何在。当然,我以前没有做过 EJB 1, 2 的项目,所以对其历史不了解。希望资深的各位大侠不吝赐教!
5 楼 ahuaxuan 2007-02-06  
ejb3.0最终没有能颠覆它原来的线程模型,其中缘由不是三言两语能够说清楚的,但是

引用
这似乎颠覆了我个人的 JVM 基本概念
似乎太夸张了吧。
4 楼 非典型程序员 2007-02-06  
dennis_zane 写道
对象池是对早期JVM性能上的妥协,现在的存在意义值的怀疑


哦,谢谢。这和我的困惑有点类似,就是无状态bean的对象池似乎没有必要。Singleton 就好了。
3 楼 非典型程序员 2007-02-06  
抛出异常的爱 写道
dennis_zane 写道
对象池是对早期JVM性能上的妥协,现在的存在意义值的怀疑
如果对象中需要sleep(c/s中会有用)没有池那大家只能等着


但是,sleep的含义是令 当前 thread 进入睡眠状态啊。

比如,我的一个对象 I 在一个 thread 里面调用了 userService 的方法 A,但因为某种原因,执行了方法 A 中的 Thread.sleep(xxx) 语句。与此同时,另一个对象 II 在另一个 thread 里面也试图调用 userService 的方法 A。我认为,这时候,对象 II 的调用不会被阻挡,因为 UserService 类是无状态的,方法 A 也就不是 synchronized 的。

所以,这样看来,无状态 bean 似乎确实不需要对象池来缓冲.....Singleton 就好了...
2 楼 抛出异常的爱 2007-02-06  
dennis_zane 写道
对象池是对早期JVM性能上的妥协,现在的存在意义值的怀疑
如果对象中需要sleep(c/s中会有用)没有池那大家只能等着
1 楼 dennis_zane 2007-02-06  
对象池是对早期JVM性能上的妥协,现在的存在意义值的怀疑

相关推荐

    JavaEE5学习笔记05-EJB之会话Bean总结

    - **会话Bean (Session Bean)**:分为有状态会话Bean和无状态会话Bean,主要用于封装业务逻辑。 - **实体Bean (Entity Bean)**:在EJB 2.x中用于持久化数据到数据库,但在EJB 3.0中已经被JPA(Java Persistence API...

    EJB3实战的源代码

    2. **有状态会话Bean**:与无状态会话Bean相反,有状态会话Bean能保留与特定用户会话相关的信息,适合处理需要持久会话状态的情况。 3. **实体Bean**:代表数据库中的记录,通过JPA进行持久化操作。实体Bean可以是...

    达内EJB3.0精典

    4. **会话 Bean**:无状态会话Bean(@Stateless)用于处理业务逻辑,不保存任何客户端状态,而有状态会话Bean(@Stateful)可以保持客户端会话期间的状态。此外,还有单例会话Bean(@Singleton),在整个应用程序中...

    使用eclipse通过jboss开发简单的ejb应用(jboss7.x & ejb3.x)

    3. **定义bean**:创建一个Java类,并添加适当的注解,例如`@Stateless`,以声明这是一个无状态会话bean。实现业务逻辑方法。 4. **创建接口**:对于客户端调用,创建一个接口,并将bean的方法声明为远程接口。 5....

    EJB3.0实例教程

    例如,`@Stateless`、`@Stateful`、`@Singleton`、`@MessageDriven`等注解分别表示无状态会话bean、有状态会话bean、单例bean和消息驱动bean。 **EJB3.0的主要特性** 1. **注解驱动**:EJB3.0通过注解实现组件的...

    Spring中依赖注入与容器

    在Spring框架中,依赖注入(Dependency Injection,简称DI)和控制反转(Inversion of Control,简称IoC)是核心概念,它们旨在简化组件之间的耦合,提高代码的可测试性和可维护性。IoC指的是应用程序的控制权由原本...

    Ejb 3 In Action

    - **新特性**:EJB 3.0中的新特性包括但不限于:无接口视图、容器管理实体、轻量级容器、本地会话Bean等。 - **简化编程模型**:EJB 3.0的目标之一是降低复杂性,使开发人员能够更轻松地编写和维护企业级应用程序。 ...

    EJB3.0__EJB3.0

    - **无状态Session Bean(Stateless Session Bean)**:对于多个客户端请求,服务器可能会复用相同的bean实例,不保留客户端状态。 2. **Message Driven Bean(MDB)**:用于处理消息队列(JMS)中的消息,常用于...

    EJB学习历程(转自别人)

    无状态会话Bean适用于不维护用户会话信息的场景,而有状态会话Bean则可以跟踪用户的交互状态。 - **实体Bean(Entity Beans)** 实体Bean对应于数据库中的持久化对象,通常用于表示业务对象,如客户、订单等。...

    最新EJB 3.0实例教程

    EJB可以是会话bean(Session Beans)、实体bean(Entity Beans)或消息驱动bean(Message-Driven Beans)。使用IDE如Eclipse或IntelliJ IDEA可以方便地进行这些操作。 **3. 数据库操作** EJB 3.0引入了实体bean的...

    25个经典的Spring面试问答.docx

    Spring Bean有单例(singleton)、原型(prototype)、请求(request)、会话(session)和全局会话(global session)等作用域,每个作用域对应不同的对象创建和存活策略。 10. Spring的嵌入beans是什么? 嵌入...

    spring(一)–概念和历史/hellospring/IOC创建对象的方式/Spring配置/set注入/c命名和p命名空间注入/bean的作用域

    Spring的诞生与Rod Johnson的两本著作紧密相关。第一本是《Expert One-on-One J2EE Design and Development》,书中首次提出了Spring的核心思想,这本书在2002年出版。第二本是《Expert One-on-One J2EE Development...

    Spring面试

    ### Spring框架概述与核心概念详解 #### 1. Spring框架简介 Spring框架是一个全面且功能强大的Java开发平台,旨在简化企业级应用的开发过程。它通过提供一系列的基础性支持和服务,帮助开发者解决常见的编程问题,...

    java面试题目

    - 会话bean代表了客户端的临时会话,分为Stateful(有状态)和Stateless(无状态)两种。 - Stateful Session Bean维护与特定客户端的会话状态,每个客户端请求对应一个唯一的bean实例。 - Stateless Session ...

    Spring 开发参考手册

    #### 四、依赖注入与Bean管理 - **依赖注入** (DI): 一种设计模式,用于降低组件间的耦合度,使代码更易于理解和维护。 - **属性注入** (Property Injection): 通过 setter 方法将依赖注入到 Bean 中。 - **构造...

    spring+jpa web开发例子

    5. **EJB3.0的关联**:虽然EJB 3.0不是一个主要技术,但可能会看到一些EJB的轻量级替代,比如使用`@Stateless`或`@Singleton`注解的无状态会话bean,它们在Spring容器中以类似的方式工作。 6. **部署和测试**:将...

    Spring中文帮助文档

    6.8.1. 在Spring中使用AspectJ进行domain object的依赖注入 6.8.2. Spring中其他的AspectJ切面 6.8.3. 使用Spring IoC来配置AspectJ的切面 6.8.4. 在Spring应用中使用AspectJ加载时织入(LTW) 6.9. 更多资源 7...

    Spring API

    6.8.1. 在Spring中使用AspectJ进行domain object的依赖注入 6.8.2. Spring中其他的AspectJ切面 6.8.3. 使用Spring IoC来配置AspectJ的切面 6.8.4. 在Spring应用中使用AspectJ加载时织入(LTW) 6.9. 更多资源 7...

    J2EE_API_5[1].0.rar

    此外,会话Bean也变得更轻量级,支持无状态和有状态会话Bean,并引入了名为"Singleton"的新会话Bean类型。 2. **JSF 1.2**:JavaServer Faces (JSF) 1.2是J2EE 5中的用户界面框架,提供了模型-视图-控制器(MVC)...

Global site tag (gtag.js) - Google Analytics