使用URLClassLoader类载入类实例

 

Tomcat当中在接受到要调用的Servlet请求后,需要载入相应的Servlet类,然后创建Servlet类实例,从而调用Servlet类实例的service()方法

下面实例的场景,request和response分别是一个javax.servlet.ServletRequest实例和一个javax.servlet.ServletResponse实例,request通过getUri()方法从ServletRequest当中获取uri

uri格式为/servlet/servletName

其中servletName就是所请求servlet资源的类名,针对这个资源名,调用URLClassLoader类载入相应的类实例

 

public void process(Request request, Response response)
	{

		String uri = request.getUri();
		String servletName = uri.substring(uri.lastIndexOf("/") + 1);
		URLClassLoader loader = null;

		try
		{
			// create a URLClassLoader
			URL[] urls = new URL[1];
			URLStreamHandler streamHandler = null;
			File classPath = new File(Constants.WEB_ROOT);
			// the forming of repository is taken from the createClassLoader
			// method in
			// org.apache.catalina.startup.ClassLoaderFactory
			String repository = (new URL("file", null, classPath.getCanonicalPath() + File.separator)).toString();
			// the code for forming the URL is taken from the addRepository
			// method in
			// org.apache.catalina.loader.StandardClassLoader class.
			urls[0] = new URL(null, repository, streamHandler);
			loader = new URLClassLoader(urls);
		} catch (IOException e)
		{
			System.out.println(e.toString());
		}
		Class myClass = null;
		try
		{
			myClass = loader.loadClass(servletName);
		} catch (ClassNotFoundException e)
		{
			System.out.println(e.toString());
		}

		Servlet servlet = null;

		try
		{
			servlet = (Servlet) myClass.newInstance();
			servlet.service((ServletRequest) request, (ServletResponse) response);
		} catch (Exception e)
		{
			System.out.println(e.toString());
		} catch (Throwable e)
		{
			System.out.println(e.toString());
		}

	}

使用java.net.URLClassLoader类来完成,该类是java.lang.ClassLoader类的一个直接子类,一旦创建了URLClassLoader类的实例之后,就可以使用它的loadClass()方法来载入Servlet类,实例化URLClassLoader类有三种方法,也就是有三个构造方法

 

构造方法摘要
URLClassLoader(URL[] urls)
使用默认的委托父 ClassLoader 为指定的 URL 构造一个新 URLClassLoader。
URLClassLoader(URL[] urls, ClassLoader parent)
为给定的 URL 构造新 URLClassLoader。
URLClassLoader(URL[] urls, ClassLoader parent, URLStreamHandlerFactory factory)
为指定的 URL、父类加载器和 URLStreamHandlerFactory 创建新 URLClassLoader。
URLClassLoader


参数:
urls - 从其位置加载类和资源的 URL 
抛出: 
SecurityException - 如果安全管理器存在并且其 checkCreateClassLoader 方法不允许创建类加载器。
另请参见:
SecurityManager.checkCreateClassLoader()


public URLClassLoader(URL[] urls)使用默认的委托父 ClassLoader 为指定的 URL 构造一个新 URLClassLoader。\n
首先在父类加载器中搜索 URL,然后按照为类和资源指定的顺序搜索 URL。这里假定任何以 '/' 结束的 URL 都是指向目录的。\n
如果不是以该字符结束,则认为该 URL 指向一个将根据需要下载和打开的 JAR 文件。
如果有安全管理器,该方法首先调用安全管理器的 checkCreateClassLoader 方法以确保允许创建类加载器。

 

参数中urls是一个java.net.URL  对象数组,当载入一个类时每个URL对象都指明了类载入器要到哪里查找类。

若一个URL以“/”结尾,则表明它指向的是个目录,否则,URL默认指向一个JAR文件,根据需要载入器会下载并打开这个JAR文件。

 

在这个例子当中,类载入器只需要查找一个位置,即工作目录下的webroot目录,因此,需要先创建一个只有一个URL对象的数组。

URL类提供了一系列的构造函数,因此有很多方法可以创建URL对象。

  
URL
public URL(URL context,
           String spec,
           URLStreamHandler handler)
    throws MalformedURLException通过在指定的上下文中用指定的处理程序对给定的 spec 进行解析来创建 URL。如果处理程序为 null,则使用两参数构造方法进行解析。 

参数:
context - 要在其中解析规范的上下文。
spec - 将作为 URL 解析的 String。
handler - URL 的流处理程序。 
抛出: 
MalformedURLException - 如果未指定任何协议,或者找到了未知协议。 
SecurityException - 如果安全管理器存在并且其 checkPermission 方法不允许指定流处理程序。
另请参见:
URL(java.lang.String, java.lang.String, int, java.lang.String), URLStreamHandler, URLStreamHandler.parseURL(java.net.URL, java.lang.String, int, int)

可以将第二个参数指定为一个目录,指定第一个和第三个参数为null,这样就可以使用构造函数了,但是我们看到在URL当中,还有一个构造函数,它也接受三个参数,

 

URL
public URL(String protocol,String host,String file) throws MalformedURLException

根据指定的 protocol 名称、host 名称和 file 名称创建 URL。使用指定协议的默认端口。
此方法等同于调用带四个参数的构造方法,四个参数为 protocol、host、-1 和 file。 此构造方法不执行对输入的验证。


参数:
protocol - 要使用的协议名称。
host - 主机名称。
file - 主机上的文件。
抛出:
MalformedURLException - 如果指定了未知协议。
另请参见:
URL(java.lang.String, java.lang.String, int, java.lang.String)

所以为了编译器的区别,指明第三个参数的类型

URLStreamHandler streamHandler = null;

new URL(null,aString,streamHandler);

第二个参数中的字符串指明了仓库的路径,也就是查找servlet类的目录。可以使用下面的代码生成仓库

String repository = (new URL("file", null, classPath.getCanonicalPath() + File.separator)).toString();
posted @ 2015-11-15 19:17  无心流泪  阅读(10673)  评论(0编辑  收藏  举报