from :http://blog.csdn.net/b_qxzb/article/details/47922193
创建对象时的区别
1、new 对象包括2步,
1)加载类;
2)并且实例化。
2、Class的对象.getInstance(),仅仅只是实例化;
也就是说,在执行 Class的对象.getInstance() 之前一定要先加载对应的类。
- //示例代码
- /**
- * new 对象和Class的对象.getInstance()的区别?<p>
- * 答:<br>
- * 1、new 对象包括2步,<br>
- * 1)加载类;<br>
- * 2)并且实例化。<p>
- *
- * 2、Class的对象.getInstance(),仅仅只是实例化;<br>
- * 也就是说,在执行 Class的对象.getInstance() 之前一定要先加载对应的类。<br>
- * @author gzh
- *
- */
- public class Main1 {
- /**
- * @param args
- */
- public static void main(String[] args) {
- // Word word = new Word();
- // ClassLoader classLoader = word.getClass().getClassLoader();
- ClassLoader classLoader = Word.class.getClassLoader();
- System.out.println(classLoader); //注:Word.class是由sun.misc.Launcher$AppClassLoader@1372a1a加载器加载的
- try {
- Class<?> c = classLoader.loadClass(args[0]); //1、必须要先加载该类;
- Object o1 = c.newInstance(); //2、然后,才可以使用.newInstance()方法来新建实例对象。
- Object o2 = c.newInstance();
- } catch (ClassNotFoundException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (InstantiationException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (IllegalAccessException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }
具体应用上的区别
既然使用newInstance()构造对象的地方通过new关键字也可以创建对象,为什么又会使用newInstance()来创建对象呢?
假设定义了一个接口Door,开始的时候是用木门的,定义为一个类WoodenDoor,在程序里就要这样写 Door door = new WoodenDoor() 。假设后来生活条件提高,换为自动门了,定义一个类AutoDoor,这时程序就要改写为 Door door = new AutoDoor() 。虽然只是改个标识符,如果这样的语句特别多,改动还是挺大的。于是出现了工厂模式,所有Door的实例都由DoorFactory提供,这时换一种门的时候,只需要把工厂的生产模式改一下,还是要改一点代码。
而如果使用newInstance(),则可以在不改变代码的情况下,换为另外一种Door。具体方法是把Door的具体实现类的类名放到配置文件中,通过newInstance()生成实例。这样,改变另外一种Door的时候,只改配置文件就可以了。示例代码如下:
String className = 从配置文件读取Door的具体实现类的类名;
Door door = (Door) Class.forName(className).newInstance();//其中Class.forName(className)是要求jvm查找并加载指定的类
假设定义了一个接口Door,开始的时候是用木门的,定义为一个类WoodenDoor,在程序里就要这样写 Door door = new WoodenDoor() 。假设后来生活条件提高,换为自动门了,定义一个类AutoDoor,这时程序就要改写为 Door door = new AutoDoor() 。虽然只是改个标识符,如果这样的语句特别多,改动还是挺大的。于是出现了工厂模式,所有Door的实例都由DoorFactory提供,这时换一种门的时候,只需要把工厂的生产模式改一下,还是要改一点代码。
而如果使用newInstance(),则可以在不改变代码的情况下,换为另外一种Door。具体方法是把Door的具体实现类的类名放到配置文件中,通过newInstance()生成实例。这样,改变另外一种Door的时候,只改配置文件就可以了。示例代码如下:
String className = 从配置文件读取Door的具体实现类的类名;
Door door = (Door) Class.forName(className).newInstance();//其中Class.forName(className)是要求jvm查找并加载指定的类
【补充:
A a = (A)Class.forName("pacage.A").newInstance();
这和你A a = new A();
是一样的效果。
关于补充的问题
答案是肯定的,jvm会执行静态代码段,你要记住一个概念,静态代码是和class绑定的,class装载成功就表示执行了你的静态代码了。而且以后不会再走这段静态代码了。
Class.forName(xxx.xx.xx) 返回的是一个类
Class.forName(xxx.xx.xx);的作用是要求JVM查找并加载指定的类,也就是说JVM会执行该类的静态代码段
动态加载和创建Class 对象,比如想根据用户输入的字符串来创建对象
String str = 用户输入的字符串
Class t = Class.forName(str);
t.newInstance();
】
总结上面的3种情况如下,
1、直接new 对象。
2、使用接口,硬编码接口的实现类——如果要更改接口的实现类的话,则必须要修改源码,并且重新编译源码。
3、使用接口,并且从配置文件获取接口的实现类——只需要修改配置文件即可。
转载自(有补充):https://blog.csdn.net/wuqilianga/article/details/61194408