浅谈servlet
刚开始接触servlet的时候,其实不是太理解servlet的,后来经过慢慢摸爬滚打式的的学习,有了一点自己的理解。
servlet的产生还要从Java和HTTP说起
- Java的servletAPI是能够让Java更好地支持web应用。HTTP协议是无状态的(也就是浏览器的每一个请求都会被服务器当作是接收到的唯一请求处理,当用户再次发出请求时,服务器并不知道是同一个用户),而servletAPI提供了会话机制(session),可以关联同一用户的请求;同时,HTTP协议是基于文本的,也就是在一个HTTP请求中,所有的数据都是以文本的形式表示。而Java又是强类型语言,用Java去处理必然会将这些数据一一对应到Java的数据类型中,过程十分繁琐。servletAPI将HTTP公开给Java平台,意味着开发人员可以直接根据通信要求更加直观地编写HTTP服务器代码。
- servletAPI的核心对象:request、response、servlet。servlet是一个单例的对象,servlet的用途是接收请求,处理之后返回响应;request对象封装了各种请求细节(表单、字符串参数);response对象包含了响应首部、生成响应文本的输出流等。
- servlet容器。servlet容器内可以包含多个web应用,一个web应用里可以包含多个servlet,(在早期的servlet规范中,servlet的映射地址是写在web.xml中,在servlet3.0之后就用java注解映射地址) 当客户端发出请求,首先会被web容器处理,而在servlet规范中,servlet容器通常监听8080端口上的所有请求(如tomcat),当请求发出后,容器会解析命名空间对应到web应用中,再对应到web应用中的servlet的URL。
servlet生命周期
在每个web中实例化一次,生命周期是由web容器决定的,在Servlet的整个生命周期内,init()方法只被调用一次,web容器创建一个Servlet实例并且调用Servlet的init()方法进行初始化,当容器关闭时,调用destroy()方法释放资源。
servlet中的线程安全问题
每一个jsp文件最后都会编译成servlet,然后按照其生命周期执行。servlet实质还是一个Java类,最终编译成CLASS文件。当客户端发起一个请求,服务端把jsp文件编译成CLASS文件,并且创建一个该类的实例,(由于servlet是单例的,该实例在整个jvm中只有一份)然后创建一个线程去处理这个请求, 当客户端有多个用户同时发起请求,服务端则会创建多个线程去处理,每个客户端对应一个线程。
- servlet中的实例变量: 实例变量是在堆中分配的,并被属于该实例的所有线程共享,所以不是线程安全的。
- 内置对象中的OUT,REQUEST,RESPONSE,SESSION,CONFIG,PAGE,PAGECONXT是线程安全的(因为每个线程对应的request,respone对象都是不一样的,不存在共享问题), APPLICATION在整个系统内被使用,所以不是线程安全的。
- 局部变量: 局部变量在堆栈中分配,因为每个线程都有它自己的堆栈空间,所以是线程安全的.。
- 静态类: 静态类不用被实例化,就可直接使用,也不是线程安全。
servlet容器如何同时处理多个请求
Servlet采用多线程来处理多个请求同时访问。servlet依赖于一个线程池来服务请求。线程池实际上是一系列的工作者线程集合。Servlet使用一个调度线程来管理工作者线程。当容器收到一个Servlet请求,调度线程从线程池中选出一个线程,将请求传递给该线程,然后由该线程来执行Servlet的service方法。当这个线程正在执行的时候,容器收到另外一个请求,调度线程同样从线程池中选出另一个线程来服务新的请求,容器并不关心这个请求是否访问的是同一个Servlet.当容器同时收到对同一个Servlet的多个请求的时候,那么这个Servlet的service()方法将在多线程中并发执行。Servlet容器默认采用单实例多线程的方式来处理请求,这样减少产生Servlet实例的开销,提升了对请求的响应时间,对于Tomcat可以在server.xml中通过<Connector>元素设置线程池中线程的数目。