servlet 在常规的web项目里面为什么是单例,寻找源码地址

1.httpservlet接口及servlet的结构 都了解了!
2.创建流程我也知道,说是解析web.xml 通过反射来创建对象
那么为什么不是多次创建呢 而是创建一次呢?
这块的源码哪里可以找到呢!

纠结两天了!

1个回答

首先HttpServlet是非线程安全,在web 容器中只创建一个实例,所以多个线程同时访问servlet的时候,Servlet是线程不安全的
那为什么还默认Servlet 创建一次即单例模式呢,首先线程安全问题是一个相对的问题,如果你继承的HttpServlet中没有成员变量,
那么就是线程安全的,所以只需要创建一次即可,跟Spring实例化Bean是一样的道理,只有没有成员变量就不会有线程安全
举例说明
如下,如果DemoHttpServelet是单例的且有成员变量count时,这样是线程不安全的,多个线程访问doGet方法对count++是线程不安全

public class DemoHttpServlet extends HttpServlet {

    private Long count = 0L;

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        count++;
        super.doGet(req, resp);
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        super.doPost(req, resp);
    }
}

那如何改造成线程安全?
方式一:就是多次创建DemoHttpServlet
方式二:DemoHttpServlet实现SingleThreadModel接口(java apic doc 已建议废弃此接口方式()
方式三:把Long count 换成线程安全的类,如:AtomicLongh或LongAdder()java8才有
最佳实践是方式三

你可以去看HttpSerrvelt源码,你可以看到里面的基本类型或者String类型都是final的,final是线相对程安全(基本类型或String类型可以认为是线程安全的,final修饰引用类型是视情况而定了)

希望对你有所帮助,建议看下effective java和java并发编程的艺术(阿里大神写的,非常深、非常棒的一本书,网上的并发的文章大多数都取经此书,初学者慎入,不然会让你怀疑人生,)

u013355306
gan到吐血而亡 有些理解了,主要是我们的action or controller 是不会存在成员变量 所以我们只需要单例就可以满足需求!
9 个月之前 回复
Csdn user default icon
上传中...
上传图片
插入图片
抄袭、复制答案,以达到刷声望分或其他目的的行为,在CSDN问答是严格禁止的,一经发现立刻封号。是时候展现真正的技术了!
立即提问