Java編程復(fù)用類代碼詳解
本文研究的主要是Java編程中的復(fù)用類,那么到底復(fù)用類是什么東西,又有什么用法,下面具體介紹。
看了老羅羅升陽的專訪,情不自禁地佩服,很年輕,我之前以為和羅永浩一個級別的年齡,也是見過的不是初高中編程的一位大牛之一,專訪之后,發(fā)現(xiàn)老羅也是一步一個腳印的人。別說什么難做,做不了,你根本就沒去嘗試,也沒有去堅持。
If you can't fly then run,if you can't run then walk, if you can't walk then crawl,but
whatever you do,you have to keep moving forward——Martin Luther King.
復(fù)用類這標題剛開始很難懂,后面專門去看了書的英文原版,其實標題是reusing classes,重新使用類,其實復(fù)用就是“利用現(xiàn)成的東西”的意思,其實實現(xiàn)的兩種方法就是java中經(jīng)常聽到的——組合和繼承。
(1)組合
has-a的作用。
public class TV {
Show show;
public String toString(){
return "showgirl";
}
}
class Show{
}
提一下toString方法,當你需要String而你是一個對象的時候,編譯器會調(diào)用對象的toString方法。
TV里有Show,現(xiàn)在的show沒有初始化,為null,不能調(diào)用show的方法。
組合的作用強大,以面向?qū)ο罂?,假如你在造一個Car類,那么你可以用組合輕易的將Glass,Light,Engine等等的Car這些部件組合起來。
(2)繼承
is-a
package com.myown.iaiti;
public class Father {
public int i;
void get(){
System.out.println("father");
}
}
package son;
import com.myown.iaiti.*;
public class Son extends Father{
Father f = new Father();
int j = f.i;
Son son = new Son();
son.get();
}
public void get(){
super.get();
System.out.println("son");
}
}
這里有個包訪問權(quán)限的問題,假如沒有加public的時候,默認是包內(nèi)成員訪問,不同包訪問,即Son中的Father成員訪問get方法是不可見的。而public的話是可見的,所以i訪問得到。
private部分是不能繼承,屬于父類私有,而public的部分,將繼承,需要修改的方法,可以進行重寫。要添加的屬性可以單獨添加。
而且繼承的方法,如果原本的father的public方法重寫之后沒將public加上,會有Cannot reduce the visibility of the inherited method from Father,也就是不能減少父類中繼承方法的可見性。super指的是父類,即Father。
還有一點是,其實java中所有的類都隱式地繼承了Object類。Object是父類,其他類是子類
老外喜歡講為基類。子類也叫導(dǎo)出類或者派生類。
(3)代理
設(shè)計模式里面有個比較難懂的——代理模式,作者講的很有趣,代理是組合和繼承的中庸之道。
package son;
class Father{
public void get(){
System.out.println("father");
}
}
public class Son extends Father{
public static void main(String[] args) {
Father f = new Father();
f.get();
}
}
class FatherProxy{
private Father f = new Father();
public void get(){
f.get();
}
}
像直接把Father當做成員,那么father的方法就暴露給這個類了,那我們可以使用FatherProxy這樣的代理類,我自己定義好get方法是怎么拿的,我自己知道是調(diào)用father的get方法,但是使用我這個代理的人不知道,我只告訴他你要用就用代理的get的方法就可以了。封裝性就體現(xiàn)出來了。上面只是隨便敲的一個簡單例子。
(4)重寫和重載
class Father{
public void get(String s){
System.out.println("father");
}
public void get(boolean b){
System.out.println("boolean");
}
}
public class Son extends Father{
@Override
public void get(String s){
System.out.println("father");
}
// @Override //會有錯誤提示 因為父類沒有該方法,不是重寫
public void get(int i ){
System.out.println("sonint");
}
public static void main(String[] args) {
Son s = new Son();
s.get("d");
s.get(false);
s.get(1);
}
}
重寫是重新覆蓋父類的方法,如果沒有重寫或者重載,那么子類調(diào)用一個子類沒有的方法時,其實是調(diào)用父類。
重載是同樣的方法名,但參數(shù)名稱不同,為了防止你錯誤的進行重載可以加上@Override標簽,那樣會提示你并沒有重寫方法。
(5)protected
在前面一篇提前寫了,因為之前沒講繼承的東西。
可以簡單將protected看成父類給兒子繼承的遺產(chǎn),其他非繼承類不能訪問。
(6)final關(guān)鍵字
加上final關(guān)鍵字的基本類型,表示這個變量初始化后不會改變。類似c的define,你希望一個變量在這個程序里就是這個值不需要改變。就可以用final。
public class Son{
int age = 2;
public static void main(String[] args) {
final int i = 1;
// i = 2; 值不能再改變
final Son son = new Son();
// son = new Son();
//The final local variable son cannot be assigned.
//It must be blank and not using a compound assignment
//final修飾的局部變量son不能被分配,必須為空或者不要再次分配
son.age = 4;
//雖然引用恒定不變,但是,對象本身卻可以改變。
}
void change(final int c){
// c= this.age; 無法賦予新值 因為值只有在方法傳參決定 對象引用和這個類似
//age ++; 無法改變
}
}
static本來是靜態(tài)初始化,和final一起用就是占據(jù)了一塊不能改變的存儲空間。
static final即編譯期常量,常量名按照c的常量命名傳統(tǒng),全部用大寫字母,單詞之間用下劃線分開。
static final VALUE_ONE = 1;
final修飾方法時
public class Print {
final void cannotprint(){
System.out.println(1);
}
}
public class PrintSon extends Print{
//void cannotprint(){}
//無法重寫 因為被final修飾了
public static void main(String[] args) {
PrintSon ps = new PrintSon();
ps.cannotprint();
}
}
可以看成父類要求子類必須繼承的不可修改財產(chǎn)(祖?zhèn)鳎rivate隱式地指定為final,因為private根本就不給你繼承。這比給你繼承但不能修改還更私有。
順便將權(quán)限理清。
- public,公共財產(chǎn),不止是子類,其他類也可以用。
- final,祖?zhèn)髡鋵殻艚o子類,但不允許修改。
- private,父類私有財產(chǎn),不會給子類繼承。
- protected,父類專門留給子類的財產(chǎn),其他人不能用它。
當final修飾的是類的時候,是為了讓這個類不會被繼承。
(7)繼承和初始化
這里的順序問題是一個很有趣的問題??蠢印?/p>
class GrandFather{
private static int i = print();
private static int print(){
System.out.println("g");
return 1;
}
}
class Father extends GrandFather{
private static int i = print();
private static int print(){
System.out.println("f");
return 1;
}
}
public class Son extends Father{
private static int i = print();
private static int print(){
System.out.println("s");
return 1;
}
public static void main(String[] args) {
System.out.println("first");
}
}
打印的結(jié)果是first嗎?錯了。
雖然執(zhí)行的是main方法,但是看到son這個需要靜態(tài)初始化的i沒有,結(jié)果是s,first嗎?
這還有初始化的問題,son是繼承father,那么編譯器會加載father,并初始化i,那father繼承g(shù)randfather,那么編譯器會去加載grandfather,類似遞歸。
那最后最先初始化的是grandfather的i。
所以最后的結(jié)果是:g,f,s,first。
總結(jié)
以上就是本文關(guān)于Java編程復(fù)用類代碼詳解的全部內(nèi)容,希望對大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站其他相關(guān)專題,如有不足之處,歡迎留言指出。感謝朋友們對本站的支持!
相關(guān)文章
spring boot下mybatis配置雙數(shù)據(jù)源的實例
這篇文章主要介紹了spring boot下mybatis配置雙數(shù)據(jù)源的實例,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-09-09
scala當中的文件操作和網(wǎng)絡(luò)請求的實現(xiàn)方法
這篇文章主要介紹了scala當中的文件操作和網(wǎng)絡(luò)請求的實現(xiàn)方法,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-06-06
spring boot 下對JSON返回值去除null和空字段操作
這篇文章主要介紹了spring boot 下對JSON返回值去除null和空字段操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-10-10
Java報錯Java.text.ParseException的解決方法匯總
在Java開發(fā)的復(fù)雜世界中,錯誤處理是開發(fā)者必須面對的關(guān)鍵挑戰(zhàn)之一,其中,Java.text.ParseException就像一個隱藏在代碼叢林中的陷阱,常常讓開發(fā)者們陷入困惑,本文給大家介紹了Java報錯Java.text.ParseException的解決方法,需要的朋友可以參考下2024-10-10
基于java查找并打印輸出字符串中字符出現(xiàn)次數(shù)
這篇文章主要介紹了基于java查找并打印輸出字符串中字符出現(xiàn)次數(shù),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-11-11
java正則表達式的應(yīng)用 java讀取文件并獲取電話號碼
這篇文章主要介紹了java正則表達式的應(yīng)用,應(yīng)用的內(nèi)容是java讀取文件并獲取電話號碼,感興趣的小伙伴們可以參考一下2015-11-11

