ClassForName应用于工厂模式
一、應(yīng)用工廠
接口:
package com.example.demo.service;public interface IFruitFactory {public void sell(int count) ;public void eat();}實(shí)現(xiàn)類(lèi):
package com.example.demo.service.impl;import com.example.demo.service.IFruitFactory;public class Apple implements IFruitFactory {@Overridepublic void sell(int count) {System.out.println("蘋(píng)果售價(jià):"+count);}@Overridepublic void eat() {// TODO Auto-generated method stub}} package com.example.demo.service.impl;import com.example.demo.service.IFruitFactory;public class Orange implements IFruitFactory{@Overridepublic void sell(int count) {System.out.println("橘子售價(jià):"+count);}@Overridepublic void eat() {// TODO Auto-generated method stub}}測(cè)試類(lèi):
package com.example.demo.service.impl;import com.example.demo.service.IFruitFactory;public class TestClassForName {public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException {String appleStr = "com.example.demo.service.impl.Apple";Class<?> c = Class.forName(appleStr);IFruitFactory fruit = (IFruitFactory)c.newInstance();fruit.sell(3);String orangeStr = "com.example.demo.service.impl.Orange";Class<?> c2 = Class.forName(orangeStr);IFruitFactory fruit2 = (IFruitFactory)c2.newInstance();fruit2.sell(4);}}測(cè)試結(jié)果:
蘋(píng)果售價(jià):3
橘子售價(jià):4
二.new 和Class.forName()有什么區(qū)別?
其實(shí)上面已經(jīng)說(shuō)到一些了,這里來(lái)做個(gè)總結(jié):
首先,newInstance( )是一個(gè)方法,而new是一個(gè)關(guān)鍵字;
其次,Class下的newInstance()的使用有局限,因?yàn)樗蓪?duì)象只能調(diào)用無(wú)參的構(gòu)造函數(shù),而使用 new關(guān)鍵字生成對(duì)象沒(méi)有這個(gè)限制。
簡(jiǎn)言之:
newInstance(): 弱類(lèi)型,低效率,只能調(diào)用無(wú)參構(gòu)造。
new: 強(qiáng)類(lèi)型,相對(duì)高效,能調(diào)用任何public構(gòu)造。
Class.forName(“”)返回的是類(lèi)。
Class.forName(“”).newInstance()返回的是object 。
三.為什么在加載數(shù)據(jù)庫(kù)驅(qū)動(dòng)包的時(shí)候有用的是Class.forName( ),卻沒(méi)有調(diào)用newInstance( )?
在Java開(kāi)發(fā)特別是數(shù)據(jù)庫(kù)開(kāi)發(fā)中,經(jīng)常會(huì)用到Class.forName( )這個(gè)方法。
通過(guò)查詢(xún)Java Documentation我們會(huì)發(fā)現(xiàn)使用Class.forName( )靜態(tài)方法的目的是為了動(dòng)態(tài)加載類(lèi)。
通常編碼過(guò)程中,在加載完成后,一般還要調(diào)用Class下的newInstance( )靜態(tài)方法來(lái)實(shí)例化對(duì)象以便操作。因此,單單使用Class.forName( )是動(dòng)態(tài)加載類(lèi)是沒(méi)有用的,其最終目的是為了實(shí)例化對(duì)象。
有數(shù)據(jù)庫(kù)開(kāi)發(fā)經(jīng)驗(yàn)朋友會(huì)發(fā)現(xiàn),為什么在我們加載數(shù)據(jù)庫(kù)驅(qū)動(dòng)包的時(shí)候有的卻沒(méi)有調(diào)用newInstance( )方法呢?
有的jdbc連接數(shù)據(jù)庫(kù)的寫(xiě)法里是Class.forName(xxx.xx.xx);而有一些:Class.forName(xxx.xx.xx).newInstance(),為什么會(huì)有這兩種寫(xiě)法呢?
剛才提到,Class.forName(“”);的作用是要求JVM查找并加載指定的類(lèi),首先要明白,java里面任何class都要裝載在虛擬機(jī)上才能運(yùn)行,而靜態(tài)代碼是和class綁定的,class裝載成功就表示執(zhí)行了你的靜態(tài)代碼了,而且以后不會(huì)再走這段靜態(tài)代碼了。
而我們前面也說(shuō)了,Class.forName(xxx.xx.xx)的作用就是要求JVM查找并加載指定的類(lèi),如果在類(lèi)中有靜態(tài)初始化器的話(huà),JVM必然會(huì)執(zhí)行該類(lèi)的靜態(tài)代碼段。
而在JDBC規(guī)范中明確要求這個(gè)Driver類(lèi)必須向DriverManager注冊(cè)自己,即任何一個(gè)JDBC Driver的 Driver類(lèi)的代碼都必須類(lèi)似如下:
public class MyJDBCDriver implements Driver {
static {
DriverManager.registerDriver(new MyJDBCDriver());
}
}
既然在靜態(tài)初始化器的中已經(jīng)進(jìn)行了注冊(cè),所以我們?cè)谑褂肑DBC時(shí)只需要Class.forName(XXX.XXX);就可以了。
Class.forName(xxx.xx.xx)返回的是一個(gè)類(lèi) Class.forName(xxx.xx.xx)的作用是要求JVM查找并加載指定的類(lèi), 也就是說(shuō)JVM會(huì)執(zhí)行該類(lèi)的靜態(tài)代碼段這句話(huà)即解釋了jdbc驅(qū)動(dòng)的加載本質(zhì)。總結(jié)
以上是生活随笔為你收集整理的ClassForName应用于工厂模式的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 安装ubuntu20.04,从格式化磁盘
- 下一篇: 计算机维修工程师认证,计算机维修工程师认