WEB开发网      濠电姷鏁告慨鐑藉极閸涘﹦绠鹃柍褜鍓氱换娑欐媴閸愬弶鎼愮痪鍓ф嚀閳规垿鎮╃€圭姴顥濋梺姹囧€楅崑鎾诲Φ閸曨垰绠涢柛顐f礃椤庡秹姊虹粙娆惧剳闁哥姵鍔欐俊鐢稿礋椤栨艾鍞ㄩ梺闈浤涙担鎻掍壕闁圭儤顨嗛埛鎺楁煕閺囥劌浜滄い蹇e弮閺屸€崇暆鐎n剛鏆犻柧浼欑到閵嗘帒顫濋悡搴d画缂佹鍨垮缁樻媴缁涘娈┑顔斤公缁犳捇銆佸鎰佹▌濠电姭鍋撳ù锝囩《閺€浠嬫煟濡鍤嬬€规悶鍎辫灃闁绘ê寮堕崯鐐电磼閸屾氨效鐎规洘绮忛ˇ瀵哥棯閹佸仮鐎殿喖鐖煎畷鐓庘槈濡警鐎崇紓鍌欑劍椤ㄥ棗鐣濋幖浣歌摕闁绘棃顥撻弳瀣煟濡も偓閻楀棗鈻撳Δ鍛拺閻犲洠鈧櫕鐏€闂佸搫鎳愭慨鎾偩閻ゎ垬浜归柟鐑樼箖閺呮繈姊洪棃娑氬婵☆偅鐟╅、娆掔疀閺冨倻鐦堥梺姹囧灲濞佳勭閿曞倹鐓曢柕濞垮劤閸╋絾顨ラ悙鏉戝妤犵偞锕㈤、娆撴嚃閳哄骞㈤梻鍌欐祰椤鐣峰Ο鑲╃煋妞ゆ棁锟ユ禍褰掓煙閻戞ɑ灏ù婊冪秺濮婅櫣绱掑Ο铏逛桓闂佹寧娲嶉弲娑滅亱闂佸憡娲﹂崹閬嶅煕閹达附鐓欓柤娴嬫櫅娴犳粌鈹戦垾鐐藉仮闁诡喗顨呴埥澶愬箳閹惧褰囩紓鍌欑贰閸犳牠顢栭崨鎼晣闁稿繒鍘х欢鐐翠繆椤栨粎甯涙繛鍛喘濮婄粯鎷呴悷閭﹀殝缂備浇顕ч崐鍨嚕缂佹ḿ绡€闁搞儯鍔嶅▍鍥⒑缁嬫寧婀扮紒瀣崌瀹曘垽鎮介崨濠勫幗闁瑰吋鐣崹濠氬煀閺囥垺鐓ユ慨妯垮煐閻撶喖鐓崶銉ュ姢缂佸宕电槐鎺旂磼濡偐鐣虹紓浣虹帛缁诲牆鐣峰鈧俊姝岊槺缂佽鲸绻堝缁樻媴缁涘娈愰梺鎼炲妺閸楀啿鐣烽鐐茬骇闁瑰濮靛▓楣冩⒑缂佹ɑ鈷掗柍宄扮墦瀵偊宕掗悙瀵稿幈闂佹娊鏁崑鎾绘煛閸涱喚鎳呮俊鍙夊姇铻i悶娑掑墲閺傗偓闂備胶绮崝鏇炍熸繝鍥у惞闁绘柨鐨濋弨鑺ャ亜閺冨洦顥夐柛鏂诲€濋幗鍫曟倷閻戞ḿ鍘遍梺鍝勬储閸斿本鏅堕鐐寸厱婵炲棗绻掔粻濠氭煛鐏炵晫效鐎规洦鍋婂畷鐔碱敆閳ь剙鈻嶉敐鍥╃=濞达絾褰冩禍鐐節閵忥絾纭炬い鎴濇川缁粯銈i崘鈺冨幍闁诲孩绋掑玻璺ㄧ不濮椻偓閺屻劌鈽夊Ο澶癸絾銇勯妸锝呭姦闁诡喗鐟╅、鏃堝礋椤撴繄绀勯梻鍌欐祰椤曟牠宕伴弽顐ょ濠电姴鍊婚弳锕傛煙椤栫偛浜版俊鑼额嚙閳规垿鍩勯崘銊хシ濡炪値鍘鹃崗妯侯嚕鐠囨祴妲堥柕蹇曞閳哄懏鐓忓璺虹墕閸旀挳鏌涢弬娆炬Ш缂佽鲸鎸婚幏鍛矙鎼存挸浜鹃柛婵勫劤閻挾鎲搁悧鍫濈瑨闁哄绶氶弻鐔煎礈瑜忕敮娑㈡煛閸涱喗鍊愰柡灞诲姂閹倝宕掑☉姗嗕紦 ---闂傚倸鍊搁崐鎼佸磹閻戣姤鍊块柨鏃堟暜閸嬫挾绮☉妯哄箻婵炲樊浜滈悡娑㈡煕濞戝崬骞樻い鏂挎濮婅櫣鎹勯妸銉︾彚闂佺懓鍤栭幏锟�
开发学院WEB开发Jsp Java类加载器 阅读

Java类加载器

 2009-11-04 20:57:02 来源:WEB开发网 闂傚倸鍊搁崐鎼佸磹妞嬪孩顐芥慨姗嗗厳缂傛氨鎲稿鍫罕闂備礁婀遍搹搴ㄥ窗閺嶎偆涓嶆い鏍仦閻撱儵鏌i弴鐐测偓鍦偓姘炬嫹闂傚倸鍊搁崐鎼佸磹妞嬪海鐭嗗〒姘e亾妤犵偛顦甸弫鎾绘偐閹绘帞鈧參姊哄Ч鍥х仼闁诲繑鑹鹃悾鐑藉蓟閵夛妇鍘甸梺瑙勵問閸犳牠銆傛總鍛婄厱閹艰揪绱曟牎闂侀潧娲ょ€氫即鐛幒妤€绠f繝闈涘暙娴滈箖鏌i姀鈶跺湱澹曟繝姘厵闁绘劦鍓氶悘杈ㄤ繆閹绘帞澧涚紒缁樼洴瀹曞崬螖閸愬啠鍓濈换娑樼暆婵犱胶鏁栫紓浣介哺閹瑰洤鐣烽幒鎴僵闁瑰吀鐒﹂悗鎼佹⒒娴g儤鍤€闁搞倖鐗犻獮蹇涙晸閿燂拷濠电姷鏁告慨鐑藉极閸涘﹥鍙忔い鎾卞灩缁狀垶鏌涢幇闈涙灈鐎瑰憡绻冮妵鍕箻鐎靛摜鐣奸梺纭咁潐濞茬喎顫忕紒妯肩懝闁逞屽墮宀h儻顦查悡銈夋煏閸繃鍋繛宸簻鎯熼梺瀹犳〃閼冲爼宕濋敃鈧—鍐Χ閸℃鐟愰梺鐓庡暱閻栧ジ宕烘繝鍥у嵆闁靛骏绱曢崢顏堟⒑閹肩偛鍔楅柡鍛⊕缁傛帟顦寸紒杈ㄥ笚濞煎繘鍩℃担閿嬵潟闂備浇妗ㄩ悞锕傚箲閸ヮ剙鏋侀柟鍓х帛閺呮悂鏌ㄩ悤鍌涘闂傚倸鍊搁崐鎼佸磹妞嬪孩顐芥慨姗嗗厳缂傛氨鎲稿鍫罕闂備礁婀遍搹搴ㄥ窗閺嶎偆涓嶆い鏍仦閻撱儵鏌i弴鐐测偓鍦偓姘炬嫹  闂傚倸鍊搁崐鎼佸磹閻戣姤鍤勯柤鍝ユ暩娴犳氨绱撻崒娆掑厡缂侇噮鍨堕妴鍐川閺夋垹鍘洪悗骞垮劚椤︻垶宕¢幎鑺ョ厪闊洦娲栨牎闂佽瀵掗崜鐔奉潖閾忓湱纾兼俊顖氭惈椤矂姊洪崷顓涙嫛闁稿妫濋幆鈧い蹇撴祩濡嫰姊洪崫鍕拱婵炲弶岣块幑銏犫攽婵犲嫮鏉搁梺鍝勬川婵兘鎮伴妷鈺傗拻濞达絽鎼敮璺侯熆閻熷府鏀荤紒鍌氱Т楗即宕煎锝呬壕闁哄啫鐗嗙粈鍐┿亜韫囧海顦﹀ù婊堢畺閺屻劌鈹戦崱娑扁偓妤€顭胯閸犳牠婀侀梺缁樕戦悷銉р偓姘煎枤缁粯銈i崘鈺冨幈濡炪倖鍔戦崐鏇㈠几鎼淬劍鐓熼煫鍥ь儏閸旀粓鏌曢崶褍顏€殿喗娼欒灒闁告繂瀚濠碉紕鍋戦崐鎴﹀垂濞差亝鍋¢柍鍝勬噹缁犳牠鏌嶉埡浣告殲闁稿海鍠栭弻鏇㈠炊瑜嶇花濠氭煙閸戙倖瀚�
核心提示: 如果你想对java加载类的机制有比较深入的理解的话,请你花点时间好好阅读一下这篇东西.这里从三个方面讲述java classloader1,将从web应用出发模拟一下web应用的classloader(这节是重点)2,通过测试2实现的classloader来加深理解3,翻译一下classloader的相关技术文章(来
  如果你想对java加载类的机制有比较深入的理解的话,请你花点时间好好阅读一下这篇东西.

  这里从三个方面讲述java classloader

  1,将从web应用出发模拟一下web应用的classloader(这节是重点)

  2,通过测试2实现的classloader来加深理解

  3,翻译一下classloader的相关技术文章(来自Core Java第二卷的Chapter 9. Security),由于我的英文水平有限,难免有不合适,请大家指出.

看看怎么实现一个自己的ClassLoader(我们以web应用的classloader为例来讲解 )

大家都很熟悉tomcat,比如我们要写一个servlet运行的话.

1,我们通常会在tomcat的webapps目录下建一个自己的web目录(比如myweb),然后让自己的myweb至少具有以下目录结构.

└─WEB-INF
  ├─classes
  └─lib

2,直接把class文件拷贝到classes目录下,或者把自己做的servet打成jar放到lib下

3,启动tomcat就能访问servlet了.

再说一点:不少人了解struts,也做过基于struts的应用.其实struts对于我们来说只是几个jar而已,我们要在自己的应用中用struts,

只不过就象上面一样把struts的各个lib拷贝到web应用的WEB-INF/lib下就OK了.

再综合前面的基础部分,其实web应用是有自己的classloader的,他专门负责加载WEB-INF/lib和WEB- INF/classes的类。下面我们就来模拟一下web应用的classloader,相信看了下面的代码你就会对classloader的理解又上一个台阶.

view plaincopy to clipboardPRint?
import java.io.File;  
import java.io.FilenameFilter;  
import java.lang.reflect.Method;  
import java.net.MalformedURLException;  
import java.net.URL;  
import java.net.URLClassLoader;  
/* 
* web应用的classloader 
* 带你步入classloader的天堂 
*/ 
public class WebAppClassLoader{  
  URLClassLoader myClassLoader = null;    
  /* 
   * 用法:new WebAppClassLoader(webRoot) 
   *  比如:new WebAppClassLoader("c:/web") 
   */ 
  public WebAppClassLoader(String root) {  
    URL[] urls = null;  
    try {  
      urls = getJarURLs(root);  
    } catch (MalformedURLException e) {  
      System.out.println(e.getMessage());  
    }  
    myClassLoader = new URLClassLoader(urls);  
  }  
  /* 
   * 获得web应用下的所有class的path的URL. 
   * 1,WEB-INF\classes目录 
   * 2,WEB-INF\lib下的所有jar文件 
   */ 
  private URL[] getJarURLs(String rootDirStr) throws MalformedURLException {  
    if (!rootDirStr.endsWith(File.separator)) {  
      rootDirStr += File.separator;  
    }  
      
    // classesDir就是web应用中的"WEB-INF\classes"目录  
    File classesDir = new File(rootDirStr + File.separator + "WEB-INF" 
        + File.separator + "classes");  
      
    // classesDir就是web应用中的"WEB-INF\lib"目录  
    File libDir = new File(rootDirStr + File.separator + "WEB-INF" 
        + File.separator + "lib");  
    // 找出目录下所有的jar文件  
    File[] jarFiles = null;  
    if (libDir.isDirectory()) {  
      jarFiles = libDir.listFiles(new FilenameFilter() {  
        public boolean accept(File dir, String name) {  
          // 注意"dir"参数指的是jar文件的父目录,"name"才是jar文件的  
          if (dir.isDirectory() // jar文件的父目录必须是一个文件夹  
              && (name.endsWith(".jar") || name.endsWith(".zip"))) // 注意zip文件也是可以的哦  
            return true;  
          return false;  
        }  
      });  
    }  
    int jarCount = null == jarFiles?0:jarFiles.length;  
    URL[] urls = new URL[1 + jarCount];  
    urls[0] = classesDir.toURI().toURL();//WEB-INF\classes  
    for (int i = 0; i < jarCount; i++) {//WEB-INF\lib下的所有jar文件  
      urls[i + 1] = jarFiles[i].toURI().toURL();  
    }  
      
    return urls;  
  }  
    
  /* 
   *加载class,直接调用 myClassLoader的loadClass(className)方法 
   */ 
  public Class<?> loadClass(String className) throws ClassNotFoundException{  
    return myClassLoader.loadClass(className);  
  }  
    
  /** 
   * 测试WebAppClassLoader 
   * @param args 
   */ 
  public static void main(String[] mainArgs) {  
    try {  
      //请根据实际情况指定目录和类名  
      WebAppClassLoader classLoader = new WebAppClassLoader("c:/web/");  
      Class<?> c = classLoader.loadClass("classloader.ButtonTest");  
      //用反射调用main方法  
      String[] args = new String[] {};  
      Method m = c.getMethod("main", args.getClass());  
      m.invoke(null, (Object) args);  
    } catch (Exception e) {  
      // handle exception  
      e.printStackTrace();  
    }  
  }  
} 
import java.io.File;
import java.io.FilenameFilter;
import java.lang.reflect.Method;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;
/*
* web应用的classloader
* 带你步入classloader的天堂
*/
public class WebAppClassLoader{
URLClassLoader myClassLoader = null;
/*
 * 用法:new WebAppClassLoader(webRoot)
 *  比如:new WebAppClassLoader("c:/web")
 */
public WebAppClassLoader(String root) {
 URL[] urls = null;
 try {
  urls = getJarURLs(root);
 } catch (MalformedURLException e) {
  System.out.println(e.getMessage());
 }
 myClassLoader = new URLClassLoader(urls);
}
/*
 * 获得web应用下的所有class的path的URL.
 * 1,WEB-INF\classes目录
 * 2,WEB-INF\lib下的所有jar文件
 */
private URL[] getJarURLs(String rootDirStr) throws MalformedURLException {
 if (!rootDirStr.endsWith(File.separator)) {
  rootDirStr += File.separator;
 }
 
 // classesDir就是web应用中的"WEB-INF\classes"目录
 File classesDir = new File(rootDirStr + File.separator + "WEB-INF"
  + File.separator + "classes");
 
 // classesDir就是web应用中的"WEB-INF\lib"目录
 File libDir = new File(rootDirStr + File.separator + "WEB-INF"
  + File.separator + "lib");
 // 找出目录下所有的jar文件
 File[] jarFiles = null;
 if (libDir.isDirectory()) {
  jarFiles = libDir.listFiles(new FilenameFilter() {
  public boolean accept(File dir, String name) {
   // 注意"dir"参数指的是jar文件的父目录,"name"才是jar文件的
   if (dir.isDirectory() // jar文件的父目录必须是一个文件夹
    && (name.endsWith(".jar") || name.endsWith(".zip"))) // 注意zip文件也是可以的哦
   return true;
   return false;
  }
  });
 }
 int jarCount = null == jarFiles?0:jarFiles.length;
 URL[] urls = new URL[1 + jarCount];
 urls[0] = classesDir.toURI().toURL();//WEB-INF\classes
 for (int i = 0; i < jarCount; i++) {//WEB-INF\lib下的所有jar文件
  urls[i + 1] = jarFiles[i].toURI().toURL();
 }
 
 return urls;
}

/*
 *加载class,直接调用 myClassLoader的loadClass(className)方法
 */
public Class<?> loadClass(String className) throws ClassNotFoundException{
 return myClassLoader.loadClass(className);
}

/**
 * 测试WebAppClassLoader
 * @param args
 */
public static void main(String[] mainArgs) {
 try {
  //请根据实际情况指定目录和类名
  WebAppClassLoader classLoader = new WebAppClassLoader("c:/web/");
  Class<?> c = classLoader.loadClass("classloader.ButtonTest");
  //用反射调用main方法
  String[] args = new String[] {};
  Method m = c.getMethod("main", args.getClass());
  m.invoke(null, (Object) args);
 } catch (Exception e) {
  // handle exception
  e.printStackTrace();
 }
}
}


测试自己的ClassLoader

1,在c盘建一个web目录(当然了,你可以把上面类的main改一下 使他适合自己的需求),然后子目录如下

└─WEB-INF
  ├─classes
  └─lib

2,新写一个带main方法的类:classloader.ButtonTest(当然了,你可以把上面类的main改一下使他适合自己的需求),然后打包成jar文件放到web/WEB-INF/lib下(当然了,你也可以不打包,直接把你的类放到web/WEB-INF/classes下),如果你很懒,就copy 我下面的代码吧,顺便还可以了解了解AWT和SWING;##这个ButtonTest的main方法也可以不象下面这么复杂,就一个 System.out.print也可以的哦

view plaincopy to clipboardprint?
package classloader;  
import java.awt.*;  
import java.awt.event.*;  
import javax.swing.*;  
public class ButtonTest  
{  
  public static void main(String[] args)  
  {  
   ButtonFrame frame = new ButtonFrame();  
   frame.setDefaultCloSEOperation(JFrame.EXIT_ON_CLOSE);  
   frame.setVisible(true);  
  }  
}  
/** 
  A frame with a button panel 
*/ 
class ButtonFrame extends JFrame  
{  
  public ButtonFrame()  
  {  
   setTitle("ButtonTest");  
   setSize(DEFAULT_WIDTH, DEFAULT_HEIGHT);  
   // add panel to frame  
   ButtonPanel panel = new ButtonPanel();  
   add(panel);  
  }  
  public static final int DEFAULT_WIDTH = 300;  
  public static final int DEFAULT_HEIGHT = 200;  
}  
/** 
  A panel with three buttons. 
*/ 
class ButtonPanel extends JPanel  
{  
  public ButtonPanel()  
  {  
   // create buttons  
   JButton yellowButton = new JButton("Yellow");  
   JButton blueButton = new JButton("Blue");  
   JButton redButton = new JButton("Red");  
   // add buttons to panel  
   add(yellowButton);  
   add(blueButton);  
   add(redButton);  
   // create button actions  
   ColorAction yellowAction = new ColorAction(Color.YELLOW);  
   ColorAction blueAction = new ColorAction(Color.BLUE);  
   ColorAction redAction = new ColorAction(Color.RED);  
   // associate actions with buttons  
   yellowButton.addActionListener(yellowAction);  
   blueButton.addActionListener(blueAction);  
   redButton.addActionListener(redAction);  
  }  
  /** 
   An action listener that sets the panel's background color. 
  */ 
  private class ColorAction implements ActionListener  
  {  
   public ColorAction(Color c)  
   {  
     backgroundColor = c;  
   }  
   public void actionPerformed(ActionEvent event)  
   {  
     setBackground(backgroundColor);  
   }  
   private Color backgroundColor;  
  }  
} 
package classloader;
import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
public class ButtonTest
{
  public static void main(String[] args)
  {
   ButtonFrame frame = new ButtonFrame();
   frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
   frame.setVisible(true);
  }
}
/**
  A frame with a button panel
*/
class ButtonFrame extends JFrame
{
  public ButtonFrame()
  {
   setTitle("ButtonTest");
   setSize(DEFAULT_WIDTH, DEFAULT_HEIGHT);
   // add panel to frame
   ButtonPanel panel = new ButtonPanel();
   add(panel);
  }
  public static final int DEFAULT_WIDTH = 300;
  public static final int DEFAULT_HEIGHT = 200;
}
/**
  A panel with three buttons.
*/
class ButtonPanel extends JPanel
{
  public ButtonPanel()
  {
   // create buttons
   JButton yellowButton = new JButton("Yellow");
   JButton blueButton = new JButton("Blue");
   JButton redButton = new JButton("Red");
   // add buttons to panel
   add(yellowButton);
   add(blueButton);
   add(redButton);
   // create button actions
   ColorAction yellowAction = new ColorAction(Color.YELLOW);
   ColorAction blueAction = new ColorAction(Color.BLUE);
   ColorAction redAction = new ColorAction(Color.RED);
   // associate actions with buttons
   yellowButton.addActionListener(yellowAction);
   blueButton.addActionListener(blueAction);
   redButton.addActionListener(redAction);
  }
  /**
   An action listener that sets the panel's background color.
  */
  private class ColorAction implements ActionListener
  {
   public ColorAction(Color c)
   {
     backgroundColor = c;
   }
   public void actionPerformed(ActionEvent event)
   {
     setBackground(backgroundColor);
   }
   private Color backgroundColor;
  }
}


3,运行上面的WebAppClassLoader,看看你自己写的,放在web/WEB-INF/lib下的有没有被调用,被调用的话就恭喜你了。

请注意:不要把你新写的带main方法的类放在WebAppClassLoader所在的工程目录下,如果你这样做的话起不到测试效果,因为这样的话类不需要自己写的WebAppClassLoader也能被load着的。


以下是翻译的关于ClassLoader介绍 

ClassLoader
 JAVA编译器把源代码转换成一个假想机器(就是我们所说的虚拟机)的语言.虚拟机指令被保存在class后缀的文件里.
 每一个类文件包含类和接口的定义以及实现代码.这些类文件必须被一个程序解释,这个程序能够把虚拟机的指令翻译成
 宿主机的机器语言.

 注意:虚拟机只加载执行一个程序所需要的类文件.举个例子,比如执行MyProgram.class,虚拟机运行的步骤如下:
 1,虚拟机有一个加载类文件的机制,比如,从硬盘读取文件或者就网络获得;虚拟机用这个机制加载MyProgram的类文件
 2,如果MyProgram有一个实例变量或者是超类,那么实例变量和超类的类文件也被加载.
 (加载一个类所依赖的所有类的过程叫做resolving the class-->自己理解吧)
 3,然后虚拟机执行MyProgram的main方法(因为是静态方法,所以不需要new MyProgram的实例)
 4,如果main 方法或者main方法调用的方法需要其他的类的话,这些类也被加载.

 类加载机制不是仅仅用一个类加载器,任何一个java程序至少有以下三个类加载器(为了不影响大家的理解,这里我就不翻译这三个类加载器的名称了)
 The bootstrap class loader:加载系统类(有代表性的,jdk的rt.jar里的类),他是虚拟机的必要组成部分,并且一般是用C实现的.
 也有类加载器对象(就是指具体的一个类加载器)不关联bootstrap class loader,比如String.class.getClassLoader()返回null.

 The extension class loader:加载jre/lib/ext目录下的class,你可以把你的jar文件放到这个目录,extension class loader
 将会加载到jar里面的类,即使你不设置classpath.(一些人建议使用这个机制以让你不受classpath的烦扰,不过注意以下的警告)
 
 The system class loader (有时也叫应用程序加载器):加载应用程序类.
 他主要加载classpath目录和jar/zip文件里的class,通过设置CLASSPATH环境变量或者是运行java的时候用[-classpath]选项指定classpath
 
 在SUN的java实现里,the extension and system class loaders都是用java实现的,他们都是URLClassLoader类的实例.
 
  警告:如果你把jar文件放到jre/lib/ext目录下,并且你的jar文件中的类需要加载一个不是system or extension的类的话,
  你将遇到麻烦.扩展类加载器不使用类路径.如果你想把类放到jre/lib/ext下进行管理的话,请牢记这一点.
  ==>怎么理解这一点:也就是说如果你把自己的x.jar放到jre/lib/ext下的话,如果你自己的x.jar里的class要用到不在
  x.jar里也不在jre/lib/ext的类的话,会导致类加载不了.不难想象吧,因为你x.jar里的类是由extension class loader
  加载的,他不会加载classpath路径下的类.
 
  警告:把jar文件放到jre/lib/ext目录下,还有第二个缺陷:有时侯,程序员忘记了他很久以前放在这个目录下的类文件.
  当class loader似乎忽略了类路径(其实没有,因为类加载总是先让父的类加载器加载类,只有父的类加载器加载不了的话
  才由自己来加载,"extension class loader是system class loader的父,因此..."),
  而加载了放在扩展目录下的遗忘已久的类的时候,他们就会迷惑不解.
 
 class loader有父子关系,bootstrap class loader以外的每一个class loader都有一个父的类加载器.
 类加载器会给父的加载器一个机会加载任何给定的类,如果父加载器加载失败的话自身才去加载.
 举个例子,当系统class loader被要求加载一个系统类的时候(比如,java.util.ArrayList),
 那么,首先需要extension class loader加载,而extension class loader又先让bootstrap class loader,
 最终由bootstrap class loader查找并且加载了rt.jar,其他任何类加载器不需要再搜索.

  注意:当实现一个类加载器的时候,你应该总是授权父加载器去加载类.否则,将会有一些潜在的安全隐患:自定义的类加载器
  可能避开重要的安全检查,意外地加载了系统类.

 Applets, servlets, and RMI stubs是用户自定义的类加载器加载的.你甚至可以根据自己的需要写自己的类加载器.
 这种方式允许你在传字节码给虚拟机之前实现特殊的安全检查.比如,你可以写一个类加载器拒绝加载没有用"paid for"
 表示的类.下一节将展示这么去实现.

 大多数时间,你不需要担心类加载器.很多类因为被其他类引用而被加载,这个过程对你来说是透明的.

 如果你在程序里调用Class.forName来加载一个类,那么一个新的类被调用Class.forName的类的加载器加载.通常,
 这不会有什么问题.然而,在下面的情况下将会失败:
  1.你的lib自己实现了一个有Class.forName方法的类的时候
  2.你的lib里的类的方法被一个不同的类加载器加载的应用程序类所调用的时候(这一先需要好好理解)
  3,被加载的类对于应用程序的类加载器来说是不可见的时候(也就是说classpath下没有相关的类)
  
  这种情况下.库类需要搜索应用程序类加载器(代码如下):
  Thread t = Thread.currentThread();
  ClassLoader loader = t.getContextClassLoader();
  Class cl = loader.loadClass(className);

Using Class Loaders as Namespaces
 任何一个java程序员都知道包名是用来消除名字冲突的.在标准类库里有两个叫Date的类(java.util.Date and java.sql.Date).
 简单的名字(这里指的是你在程序里直接写Date)只是程序员方便,并且需要包含import语句.在一个运行的程序中,
 所有的class都包含他们的包名.

 这也许让你吃惊,然而,在同一个虚拟机里面你可以有两个类名和包名都相同的类.一个类是通过他的全名和类加载器来标识的.
 This technique is useful for loading code from multiple sources.比如,浏览器为每个web页面使用单独的
 applet class loader.这允许虚拟机分开来自不同web页面的类,不管他们是怎么命名的.

  NOTE:这项技术还有其他的用途,比如Sevlet和EJB的"热部署",请参照:http://java.sun.com/developer/TechTips/2000/tt1027.html
 
 
Writing Your Own Class Loader
 The loadClass method of the ClassLoader superclass takes care of the delegation to
 the parent and calls findClass only if the class hasn't already been loaded and
 if the parent class loader was unable to load the class.
 定义自己的类加载器只需要继承ClassLoader类并且重写findClass(String className)方法.
 ClassLoader父类的loadClass方法负责授权给父的类加载器 并且只有在还没有加载并且父的类加载器不能加载的时候
 才调用findClass方法.
 
 NOTE:在早期版本的JDK中,程序员必须重写loadClass方法.现在不再建议这种做法.
 
 实现findClass方法必须做下面两件事情,
 1,从文件或者其他地方加载类的字节码
 2,为了把字节码提交给虚拟机,需要调用ClassLoader类的defineClass方法,
 
 以下是实现一个加载加密类文件的类加载器.

view plaincopy to clipboardprint?
/** 
* This class loader loads encrypted class files. 
*/ 
class CryptoClassLoader extends ClassLoader {  
  /** 
   * Constructs a crypto class loader. 
   *  
   * @param k 
   *      the decryption key 
   */ 
  public CryptoClassLoader(int k) {  
    key = k;  
  }  
  protected Class findClass(String name) throws ClassNotFoundException {  
    byte[] classBytes = null;  
    try {  
      classBytes = loadClassBytes(name);  
    } catch (IOException e) {  
      throw new ClassNotFoundException(name);  
    }  
    Class cl = defineClass(name, classBytes, 0, classBytes.length);  
    if (cl == null)  
      throw new ClassNotFoundException(name);  
    return cl;  
  }  
  /** 
   * Loads and decrypt the class file bytes. 
   *  
   * @param name 
   *      the class name 
   * @return an array with the class file bytes 
   */ 
  private byte[] loadClassBytes(String name) throws IOException {  
    String cname = name.replace('.', '/') + ".caesar";  
    FileInputStream in = null;  
    in = new FileInputStream(cname);  
    try {  
      ByteArrayOutputStream buffer = new ByteArrayOutputStream();  
      int ch;  
      while ((ch = in.read()) != -1) {  
        byte b = (byte) (ch - key);  
        buffer.write(b);  
      }  
      in.close();  
      return buffer.toByteArray();  
    } finally {  
      in.close();  
    }  
  }  
  private int key;  
} 
/**
* This class loader loads encrypted class files.
*/
class CryptoClassLoader extends ClassLoader {
/**
 * Constructs a crypto class loader.
 *
 * @param k
 *      the decryption key
 */
public CryptoClassLoader(int k) {
 key = k;
}
protected Class findClass(String name) throws ClassNotFoundException {
 byte[] classBytes = null;
 try {
  classBytes = loadClassBytes(name);
 } catch (IOException e) {
  throw new ClassNotFoundException(name);
 }
 Class cl = defineClass(name, classBytes, 0, classBytes.length);
 if (cl == null)
  throw new ClassNotFoundException(name);
 return cl;
}
/**
 * Loads and decrypt the class file bytes.
 *
 * @param name
 *      the class name
 * @return an array with the class file bytes
 */
private byte[] loadClassBytes(String name) throws IOException {
 String cname = name.replace('.', '/') + ".caesar";
 FileInputStream in = null;
 in = new FileInputStream(cname);
 try {
  ByteArrayOutputStream buffer = new ByteArrayOutputStream();
  int ch;
  while ((ch = in.read()) != -1) {
  byte b = (byte) (ch - key);
  buffer.write(b);
  }
  in.close();
  return buffer.toByteArray();
 } finally {
  in.close();
 }
}
private int key;
} 

Tags:Java 加载

编辑录入:爽爽 [复制链接] [打 印]
赞助商链接