Java高薪基礎(chǔ)加強(qiáng)
一枚舉:
為什么要有枚舉:
問題:要定義星期幾或性別的變量。該怎么定義?
枚舉就是要讓某個(gè)類型的變量的取值只能為若干個(gè)固定值中的一個(gè),否則,編譯器就會報(bào)錯。枚舉可以讓編譯器在編譯時(shí)就可以控制源程序中填寫的非法值,普通變量在方式在開發(fā)階段無法實(shí)現(xiàn)這一目標(biāo)。
用普通類如何實(shí)現(xiàn)枚舉功能,定義一個(gè)weekday的類來模擬美劇功能:
私有的構(gòu)造方法
每一個(gè)元素分別用一個(gè)共有的靜態(tài)成員變量表示
可以有若干共有方法或抽象方法,例如:要提供nextDay方法必須是抽象的。
采用抽象方法定義nextDay就將大量的if?else語句轉(zhuǎn)移成了一個(gè)個(gè)獨(dú)立的類。
如果想在一個(gè)類中編寫完各個(gè)枚舉類和測試調(diào)用類,那么可以將枚舉類定義成調(diào)用類的內(nèi)部類。
枚舉的基本應(yīng)用:
擴(kuò)展:枚舉類的values,valueOf,name,toString,ordinal等方法。
總結(jié):枚舉是一種特殊的類,其中的每個(gè)元素都是該類的一個(gè)實(shí)例對象,
public class EnumTest {
?public static void main(String[] args) {
? WeekDay weekDay2 = WeekDay.FRI;
? System.out.println(weekDay2);
? System.out.println(weekDay2.hashCode());
? System.out.println(weekDay2.name());
? System.out.println(weekDay2.ordinal());
? System.out.println(WeekDay.valueOf("SUN").toString());
? System.out.println(WeekDay.values().length);
? //System.out.println(WeekDay.valueOf(, "SUN"));
?}
?
?public enum WeekDay
?{
? SUN,MON,TUE,WED,THI,FRI,SAT;
? private WeekDay(){System.out.println("frist");}
? private WeekDay(int i){System.out.println("second");}
?}
}
enum?TrafficLamp{ /** ?*?匿名對象內(nèi)部類! ?*?枚舉最簡單的創(chuàng)建對象方法:RED,YELLOW,GREEN ?*?對于普通的類而言,創(chuàng)建對象:Person?p?=?new?Person(); ?*? ?*/ RED(30){ @Override public?TrafficLamp?nextLamp()?{ return?YELLOW; } }, YELLOW(40){ @Override public?TrafficLamp?nextLamp()?{ return?GREEN; } }, GREEN(45){ @Override public?TrafficLamp?nextLamp()?{ return?RED; } }; public?abstract?TrafficLamp?nextLamp(); private?int?time; private?TrafficLamp(int?time){ this.time?=?time; } public?int?getTime()?{ return?time; } public?void?setTime(int?time)?{ this.time?=?time; } } public?class?EnumDemo4?{ public?static?void?main(String[]?args)?{ System.out.println(TrafficLamp.RED.nextLamp()); System.out.println(TrafficLamp.RED.nextLamp().getTime()); } }
//=================================================
二、內(nèi)省JavaBean
??? JavaBean是一種特殊的Java類,主要用于傳輸數(shù)據(jù)信息,這種java類中的方法主要用于訪問私有的字段,且方法名符合某種命名規(guī)則。
如果要在兩個(gè)模塊之間傳遞多個(gè)信息,可以將這些信息封裝到一個(gè)JavaBean中,這種JavaBean的實(shí)例對象通常稱之為值對象。這些信息在類中用私有字段來存儲,如果讀取或設(shè)置這些字段的值,可以通過setter和getter方法來確定。
????命名規(guī)則:setId()的屬性名->id
????????????????????? isLast()->last
???????????????????? setCPU->CPU
???????????????????? getUPS->UPS
???? 總之,一個(gè)類被當(dāng)作JavaBean使用時(shí),JavaBean的屬性是根據(jù)方法名推斷出來的,它根本看不到j(luò)ava類內(nèi)部的成員變量。
2,對JavaBean的簡單內(nèi)省操作
????ReflectPoint pt1 = new ReflectPont(3,5);
????String propertyName? = "x";
??? x->X->getX->MethodGetX
??? PropertyDescriptor pd = new PropertyDescriptor(propertyName,pt1.getClass());//JavaBean的屬性描述類
??? Method methodGetX = pd.getReadMethod();
??? Objec reVal = methodGetX.invoke(pt1);
??? System.out.println(retVal);
??? Method methodSetX = pd.getWriteMethod();
???? methodSetX.invoke(pt1,7);
??? System.out.printn(pt1.getX());//通過普通Java類的方法獲得
3,對JavaBean的復(fù)雜內(nèi)省操作
? 采用遍歷BeanInfo的所有屬性方式來查找和設(shè)置某個(gè)RefectPoint對象的x屬性。在程序中把一個(gè)類當(dāng)作JavaBean來看,就是調(diào)用IntroSpector.getBeanInfo方法,得到的
?? BeanInfo對象封裝了把這個(gè)類當(dāng)作JavaBean看的結(jié)果信息。
??? BeanInfo beanInfo = Introspector.getBeanInfo(pt1.getClass());
??? PropertyDescriptor [] pds = beanInfo.getPropertyDescriptors();
??? Object retVal = null;
?? for(PropertyDescriptor pd : pds?){
????? if(pd.getName().equals(propertyName)){
??????????? Method methodGetX = pd.getReadMethod();
??????????? retVal = methodGetX.invoke(pt1);
?????????? break;
??? }
?}
?return retVal;
4,使用BeanUtils工具包操作JavaBean
??? 先導(dǎo)入beanutils包,在導(dǎo)入logging包,BeanUtils包把上述操作進(jìn)行封裝
?????? BeanUtils.setProperty(pt1,"x","9");
?????? BeanUtils.setProperty(pt1,"birthday.time","111");//birthday為Data類型,可以級聯(lián)操作,操作對象的對象。birthday有setTime方法。
?????? ProprtyUtils.setProperty(pt1,"x",8);
????? 注意:BeanUtils類get屬性時(shí)返回的結(jié)果是字符串,set屬性時(shí)可以接受任意類型的對象,通常使用字符串。
???????????????? 用PropertyUtils類get屬性時(shí)返回的結(jié)果為該屬性本來的類型,set屬性時(shí)只接受該屬性本來的類型。
//========================================================
三、了解和入門注解的應(yīng)用
???? 注解相當(dāng)于一種標(biāo)記,在程序中加了注解就等于為程序打上了某種標(biāo)記,沒加則等于沒有某種標(biāo)記,以后,javac編譯器,開發(fā)工具和其他程序可以用反射來了解你的
?????類及各種元素上有無何種標(biāo)記,看你有什么標(biāo)記,就去干相應(yīng)的事。標(biāo)記可以加在包,類,字段,方法,方法的參數(shù)以及局部變量上。
??? 看java.lang包,可看到j(luò)dk中提供的最基本的annotation.
??? @Override重寫注解
??? @SuppressWarnings? 編譯器警告注解
??? @Deprecated 作廢的注解
??? 注解就相當(dāng)于一個(gè)你的源程序中要調(diào)用的一個(gè)類,要在源程序中應(yīng)用某個(gè)注解,得先準(zhǔn)備好了這個(gè)注解類。就像你要調(diào)用某個(gè)類,得先有開發(fā)好這個(gè)類。
? ?
??? 應(yīng)用:定義一個(gè)最簡單的注解:public @interface MyAnnotation{}
??????????????? 把它加在某個(gè)類上:@MyAnnotation public class AnnotationTest{}
??????????????? 用反射進(jìn)行測試AnnotationTest的定義上是否有@MyAnnotation
????????????? @MyAnnotation
????????????? public class AnnotationTest{
???????????????????? public static void main(String [] args) throws Exception{
????????????????????????? if(AnnotationTest.class.isAnnotationPresent(MyAnnotation.class)){
?????????????????????? MyAnnotation? annotation = (MyAnnotation)AnnotationTest.class.getAnnotation(MyAnnotation.class)
??????????????????????????? System.out.println(anntation);
??????????????? ?}
???????????? }
?? }
???? @Retention元注解有三種取值:RetetionPolicy.SOURCE、RetetionPolicy.CLASS、RetetionPolicy.RUNTIME;
????????? 分別對應(yīng):java源文件,class文件,內(nèi)存中的字節(jié)碼
?????@Override、@SuppressWarnings是源文件期,@Deprecated是內(nèi)存中的字節(jié)碼
????
???? @Target元注解
?????????Target的默認(rèn)值為任何元素,可以用數(shù)組方式設(shè)置{ElementType.METHOD,ElementType.TYPE}//ElementType為枚舉類,TYPE代表類的類型
????? 定義基本類型的屬性和應(yīng)用屬性:
????? 在注解類中增加String color();
??????? @MyAnnotation(color="red")
??????? 用反射方式獲得注解對應(yīng)的實(shí)例對象后,再通過該對象調(diào)動屬性對象的方法。
?????? 為屬性指定缺省值:String color() default "yellow";
?????? value屬性:
???????String? value() default "xxx";??? 如果注釋中有一個(gè)名稱為value的屬性,且只想設(shè)置value屬性,那么可以省略value=部分,例:@MyAnnotation("dd").
?//========================================================
四、類加載器
?? 類加載器:加載類的工具
在java中用到一個(gè)類,類加載器把文件字節(jié)碼問價(jià)從硬盤加載到內(nèi)存
? ? 類加載器也是JAVA類,因?yàn)槠渌鹙ava類的類加載器本身也是被類加載器加載,顯然必須有第一個(gè)類加載器不是java類,這是BootStrap。
? ? ?Java虛擬機(jī)中的所有類裝載器采用具有父子關(guān)系的樹形結(jié)構(gòu)進(jìn)行組織,在實(shí)例化每個(gè)類裝載器對象時(shí),需要為其指定一個(gè)父級類加載器
對象或者默認(rèn)采用系統(tǒng)類裝載器為其父級類加載。
? ? ? ? ? ??
???????
類加載器的委托機(jī)制:
當(dāng)java虛擬機(jī)要加載一個(gè)類時(shí),到底派出哪個(gè)類加載器去加載呢?
? ? 首先當(dāng)前線程的類加載器去加載線程中的第一個(gè)類。
? ? 如果類A中引用了類B,Java虛擬機(jī)將使用加載類A的類裝載器來加載類B。
? ? 還可以直接調(diào)用ClassLoader.loadClass()方法來指定某個(gè)類加載器去加載。
? ? 每個(gè)類加載器加載類時(shí),又先委托給其上級類加載器。
? ? ?當(dāng)所有祖宗類加載器沒有加載到類,回到發(fā)起者類加載器,還加載不了,則拋ClassNotFoundExcepiton,不是再去找發(fā)起者類加載器的兒子,因?yàn)闆]有g(shù)etChild方法,即使有,那么多個(gè)兒子,去找哪一個(gè)呢?
?????? 每個(gè)ClassLoader本身只能分別加載特定位置和目錄中的類,但他們可以委托其他的類裝載器去加載類,這就是類加載器的委托模式。
????????類裝載器一級級委托到BootStrap類加載器,當(dāng)BootStrap無法加載當(dāng)前所要加載的類時(shí),然后才一級級回退到子孫類裝載器去進(jìn)行真正
??????? 的加載。當(dāng)回退到最初的類裝載器時(shí),如果它自己也不能完成類的裝載,那就應(yīng)報(bào)告ClassNotFoundExcepiton異常。
??????面試題:能不能自己寫個(gè)類叫java.lang.System,為了不讓我們寫System類,類加載采用委托機(jī)制,這樣可以保證爸爸們優(yōu)先,也就是
??????? 總是使用爸爸們能找到的類,這樣總是使用java系統(tǒng)提供的System.
? //========================================================
五、代理
程序中的代理:要為已存在的多個(gè)具有相同接口的目標(biāo)類的各個(gè)方法增加一些系統(tǒng)功能,例如:異常處理、日志、計(jì)算方法的運(yùn)行時(shí)間、事務(wù)管理、等等。 代理類要調(diào)用目標(biāo)類的功能。 目標(biāo)類??doSomeThing(){ ????????業(yè)務(wù)功能代碼 ????????????} 代理類??doSomeThing(){ ??????????//前置系統(tǒng)功能代碼 ???????????目標(biāo)對象.doSomeThing() ??????????//后置系統(tǒng)功能代碼
? ? ? ? ?}? ?
??????
動態(tài)代理:
動態(tài)代理的工作原理: 1)Client(客戶端)調(diào)用代理,代理的構(gòu)造方法接受一個(gè)InvocationHandler,client調(diào)用代理的各個(gè)方法,代理的各個(gè)方法請求轉(zhuǎn)發(fā)給剛才通過構(gòu)造方法傳入的handler對象,又把各請求分發(fā)給目標(biāo)的相應(yīng)的方法。就是將handler封裝起來,其中this引用了當(dāng)前的放(發(fā)來什么請求就接受哪個(gè)方法)。 猜想分析動態(tài)生成的類的內(nèi)部代碼: 1、動態(tài)生成的類實(shí)現(xiàn)了Collection接口(可以實(shí)現(xiàn)若干接口),生成的類有Collection接口中的所有方法和一個(gè)如下接受InvocationHandler參數(shù)的構(gòu)造方法。 2、構(gòu)造方法接受一個(gè)InvocationHandler對象,接受對象了要干什么用呢?該方法內(nèi)部的代碼會是怎樣的呢? 實(shí)現(xiàn)Collection接口的動態(tài)類中的各個(gè)方法的代碼又是怎樣的呢?InvocationHandler接口中定義的invoke方法接受的三個(gè)參數(shù)又是什么意思?圖解說明如下: 分析為什么動態(tài)類的實(shí)例對象的getClass()方法返回了正確結(jié)果呢? 為何動態(tài)類的實(shí)例對象的getClass()方法返回了正確結(jié)果,而沒調(diào)用invoke方法: 因?yàn)榇眍悘腛bject上繼承了許多方法,其中只對三個(gè)方法(hashCode、equals和toString)進(jìn)行開發(fā),委托給handler去自行處理,對于它身上其他方法不會交給代理類去實(shí)現(xiàn),所以對于getClass()方法,還是由Object本身實(shí)現(xiàn)的。即proxy3.getClass(),該是什么結(jié)果還是什么結(jié)果,并不會交給invoke方法處理。 自定義代理類的步驟: 方式1: ?????????????獲得動態(tài)代理類的字節(jié)碼文件 Class?claz=Proxy.getProxyClass(Collection.class.getClassLoader(),?Collection.class); //通過反射獲得該動態(tài)代理類的構(gòu)造方法 Constructor?c=claz.getConstructor(InvocationHandler.class); ?????//編寫一個(gè)InvocationHandler類 class?myInvocationHandler?implements?InvocationHandler{ @Override public?Object?invoke(Object?proxy,?Method?method,?Object[]?args) throws?Throwable?{ //?TODO?Auto-generated?method?stub return?null; } } //創(chuàng)建實(shí)例,接受一個(gè)invocationhandler對象 c.newInstance(new?myInvocationHandler()); 方式2,newProxyInstance這個(gè)方法需要三個(gè)參數(shù),可以直接創(chuàng)建target的代理對象?? ????????Object?proxy3?=?Proxy.newProxyInstance(?? ????????????????target.getClass().getClassLoader(),?? ????????????????/*new?Class[]{Collection.class},*/?? ?????????????????? ????????????????//獲取target上的接口?? ????????????????target.getClass().getInterfaces(),?? ????????????????new?InvocationHandler(){?? ?????????????????? ????????????????????public?Object?invoke(Object?proxy,?Method?method,?Object[]?args)?? ????????????????????????????throws?Throwable?{?? ?? ????????????????????????/*long?beginTime?=?System.currentTimeMillis();?? ????????????????????????Object?retVal?=?method.invoke(target,?args);?? ????????????????????????long?endTime?=?System.currentTimeMillis();?? ????????????????????????System.out.println(method.getName()?+?"?running?time?of?"?+?(endTime?-?beginTime));?? ????????????????????????return?retVal;*/?? ?????????????????????????? ????????????????????????//把上邊的代碼封裝到一個(gè)類中,讓后調(diào)用該類的方法,就實(shí)現(xiàn)了方法的封裝?? ????????????????????????advice.beforeMethod(method);?? ????????????????????????Object?retVal?=?method.invoke(target,?args);?? ????????????????????????advice.afterMethod(method);?? ????????????????????????return?retVal;???????????????????????? ?????????????????????????? ????????????????????}?? ????????????????}?? ????????????????);?? ????????return?proxy3;?? ????}?? ?? }