在Java Servlet中我们能有构造函数么?或者,如果已经有一个init()方法来初始化Servlet,那在Servlet中为什么需要构造函数?或者说,构造函数和init()之间的区别是什么呢?
以上问题都与Servlet实现class中,构造函数和init()的角色有关。
首先,Servlet特别之处在于它们的生命周期由web容器管理,如Tomcat和Jetty。它们负责创建Servlet实例,以及在它们没有足够资源或需要支持更多Servlets实例的时候destroy它们。
-
我们能在Servlet中创建constructor么?
Servlet实现类可以有constructor,但是它们应该使用init()方法来初始化Servlet,因为以下两个原因:- java中,你不能在接口里面声明构造函数,也就是说你不能对任何实现Servlet接口的类强制这个要求。
- Servlet需要容器创建的ServletConfig对象来初始化,因为ServletConfig有指向ServletContext object的引用。
Servlet是一个定义在javax.servlet包中的接口,HttpServlet是一个类,并且像java中其它类一样,它可以有构造函数,但是你不能在接口中声明构造函数。如果你没有显示提供一个构造函数,编译器会在Servlet实现类中添加一个默认无参构造函数。另外一个原因是你不应该使用构造函数来初始化Servlet,因为Servlet并不直接被Java代码实例化,而是由容器创建它们的实例,并且把它们保持在pool中。由于web server上的容器使用Java反射来创建Servlet实例,必须要有无参构造函数。如果你提供了一个有参数的构造函数并且没有提供无参构造函数,web容器不能创建你的Servlet的实例。如果有一个有参数的构造函数,Java编译器不能添加默认无参构造函数。这就是不建议在Servlet class中提供构造函数的原因。
为什么我们需要使用init()方法来初始化Servlet?
-
Constructor和init方法的区别
在真实的应用中,你最好使用init()来做初始化,因为init()方法接收一个ServletConfig参数,它能包含web.xml中对应Servlet的任意初始化参数。由于web.xml提供了有用的信息给web容器,比如实例化的Servlet名字。ServletConfig instance被用来提供初始化参数给Servlets。你能够基于ServletConfig对象中提供的配置来设置你的Servlet。比如你能够提供具体的环境设置:临时目录的路径、数据库连接参数以及其他配置参数。你能够在一个环境中使用web.xml中不同设置来简单地deploy你的web应用。请记住,init()不像构造函数是链式的,构造函数的super类构造函数会在子类构造函数执行之前执行,也叫做构造器链。
容器使用web.xml来获取Servlet的名字来初始化,并且使用Java反射API(主要是class.newInstance())来创建Servlet实例,也就意味着Servlet必须需要一个默认无参构造函数。我们已经看到为什么Servlet不能有用户定义的构造函数,主要是因为Servlets作为接口不能保证它和web容器创建Servlet实例以及能够访问Context和Config对象,开发者不能访问到它们。总是优先使用init()方法来初始化Servlet,因为ServletConfig对象被用来给init()方法提供参数。
参考资料
javarevisited