JDBC drivers register themselves in the JVM-wide singleton
DriverManager
which is shared by allweb apps. If you have the same (as in class name) JDBC driver register twice from two different web apps, this might cause your problem. This is even more problematic if your web apps use different versions of the same JDBC driver.
Also, putting JDBC drivers into Tomcat's lib folder will help prevent memory leaks when you redeploy your web app without restarting Tomcat, e.g. if you just put a new WAR file into Tomcat's webapps folder:
The class
DriverManager
gets loaded by the bootstrap classloader and thereby "lives" globally in the JVM, while Tomcat loads all web apps in their own classloaders. So if a JDBC driver from a web app's WEB-INF/lib folder registers itself in DriverManager
, it pins that web app's classloader in memory (and thereby all the classes of that web app), preventing its garbage collection.
If instead both
DriverManager
and JDBC drivers come from non-web app classloaders, you can freely redeploy your web apps without any web app classes pinning themselves in classes loaded from other classloaders.
Current versions of Tomcat (probably 6.x and definitely 7.x) will log warnings on undeployment of a web app if a memory leak is detected, among other things by JDBC drivers.
We can simply ignore it, or you can create a ServletContextListener that deregisters the drivers at app shutdown:
We can simply ignore it, or you can create a ServletContextListener that deregisters the drivers at app shutdown:
import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; import java.sql.Driver; import java.sql.DriverManager; import java.sql.SQLException; import java.util.Enumeration; public class MyContextListener implements ServletContextListener { @Override public void contextDestroyed(ServletContextEvent arg0) { System.out.println("App shutdown ..."); System.out.println("Deregistering SQL-Drivers ..."); Enumeration<Driver> drivers = DriverManager.getDrivers(); while (drivers.hasMoreElements()) { Driver driver = drivers.nextElement(); try { DriverManager.deregisterDriver(driver); System.out.println(driver.getClass().getName()); } catch (SQLException e) { System.err.println("Error deregistering driver " + driver.getClass().getName()); } } } @Override public void contextInitialized(ServletContextEvent arg0) { } }
No comments:
Post a Comment