Java中的static--靜態(tài)變量你了解嗎
引言
static 是 java 語(yǔ)言中的關(guān)鍵字,意思是“靜態(tài)的”,它可以用來修飾變量、方法、代碼塊等,修飾的變量叫做靜態(tài)變量,修飾的方法叫做靜態(tài)方法,修飾的代碼塊叫做靜態(tài)代碼塊。
初學(xué)stat
引言
static 是 java 語(yǔ)言中的關(guān)鍵字,意思是“靜態(tài)的”,它可以用來修飾變量、方法、代碼塊等,修飾的變量叫做靜態(tài)變量,修飾的方法叫做靜態(tài)方法,修飾的代碼塊叫做靜態(tài)代碼塊。
初學(xué)static的最大的問題就是:static有什么用?什么時(shí)候用?這里先說一下static的作用:
static的作用就是用于區(qū)分成員變量、方法、內(nèi)部類、初始化塊這四種成員到底屬于類本身還是屬于實(shí)例
這篇文章先介紹靜態(tài)變量,后面幾種會(huì)再下面的篇章中分別介紹
靜態(tài)變量是成員變量的一種,靜態(tài)變量存儲(chǔ)在方法區(qū)中,則它在類加載時(shí)就會(huì)進(jìn)行初始化,所以靜態(tài)變量訪問的時(shí)候不需要?jiǎng)?chuàng)建實(shí)例(對(duì)象),直接可以通過類名來訪問;
什么時(shí)候使用靜態(tài)變量?
先來看一個(gè)代碼:
public class StaticTest01 {
public static void main(String[] args) {
Chinese c1 = new Chinese("張三", "333333333333", "中國(guó)");
System.out.print("姓名:" + c1.name);
System.out.print(" 身份證號(hào):" + c1.idNum);
System.out.println(" 國(guó)籍:" + c1.nation);
Chinese c2 = new Chinese("李四", "444444444444", "中國(guó)");
System.out.print("姓名:" + c2.name);
System.out.print(" 身份證號(hào):" + c2.idNum);
System.out.println(" 國(guó)籍:" + c2.nation);
}
}
class Chinese {
// 姓名; 實(shí)例變量
String name;
//身份證號(hào); 實(shí)例變量
String idNum;
//國(guó)籍; 實(shí)例變量
String nation;
//無參構(gòu)造器
public Chinese() {}
//有參構(gòu)造器
public Chinese(String myName, String myIdNum, String myNation) {
this.name = myName;
this.idNum = myIdNum;
this.nation = myNation;
}
}
輸出結(jié)果:
姓名:張三 身份證號(hào):333333333333 國(guó)籍:中國(guó)
姓名:李四 身份證號(hào):444444444444 國(guó)籍:中國(guó)
可以看出來,這個(gè)Chinese類抽象的是中國(guó)人,所以這里面”國(guó)籍“實(shí)例變量就可以聲明為靜態(tài)變量,讓國(guó)籍固定為chinese;
訪問時(shí)只需要用Chinese類來訪問就可以了;
代碼如下:
public class StaticTest02 {
public static void main(String[] args) {
Chinese c1 = new Chinese("張三", "333333333333");
System.out.print("姓名:" + c1.name);
System.out.print(" 身份證號(hào):" + c1.idNum);
// 靜態(tài)變量不需要實(shí)例化,只需要用類來訪問就可以了
System.out.println(" 國(guó)籍:" + Chinese.nation);
Chinese c2 = new Chinese("李四", "444444444444");
System.out.print("姓名:" + c2.name);
System.out.print(" 身份證號(hào):" + c2.idNum);
System.out.println(" 國(guó)籍:" + Chinese.nation);
}
}
class Chinese {
// 姓名; 實(shí)例變量
String name;
//身份證號(hào); 實(shí)例變量
String idNum;
//國(guó)籍; 靜態(tài)變量
static String nation = "中國(guó)";
//無參構(gòu)造器
public Chinese() {}
//有參構(gòu)造器
public Chinese(String myName, String myIdNum) {
this.name = myName;
this.idNum = myIdNum;
}
}
輸出結(jié)果:
姓名:張三 身份證號(hào):333333333333 國(guó)籍:中國(guó)
姓名:李四 身份證號(hào):444444444444 國(guó)籍:中國(guó)
可以看到輸出結(jié)果都是一樣的,這時(shí)候應(yīng)該會(huì)有一個(gè)疑問:既然輸出結(jié)果都一樣,那么用靜態(tài)變量有什么用呢?
這就涉及到底層的內(nèi)存了,簡(jiǎn)單說一下,實(shí)例變量都存放在堆中,每一個(gè)實(shí)例變量都會(huì)占有一定的內(nèi)存空間,像”國(guó)籍“這個(gè)實(shí)例變量同樣每聲明一個(gè)對(duì)象就會(huì)多占用一塊空間,而靜態(tài)變量存放在方法區(qū)中,而且只需要聲明一次,這個(gè)類就可以一直用;
所以,如果這個(gè)類的所有對(duì)象的屬性值都是一樣的,不建議定義為實(shí)例變量,會(huì)浪費(fèi)內(nèi)存空間;
建議定義為靜態(tài)變量,在方法區(qū)中只保留一份,節(jié)省內(nèi)存的開銷;
小總結(jié)
對(duì)于靜態(tài)變量,
對(duì)象級(jí)別的變量不需要static
類級(jí)別的變量需要static
對(duì)象級(jí)別:一個(gè)對(duì)象對(duì)應(yīng)一份數(shù)據(jù),各個(gè)對(duì)象都可以不同
類級(jí)別: 所有對(duì)象對(duì)應(yīng)一份數(shù)據(jù),各個(gè)對(duì)象都一樣
引用訪問靜態(tài)變量
靜態(tài)變量可以使用引用去訪問,但實(shí)際上在執(zhí)行過程中,引用所指向的對(duì)象并沒有參與,如果是空引用訪問實(shí)例變量,程序一定會(huì)發(fā)生空指針異常,但是如果用空引用訪問靜態(tài)變量,程序并沒有發(fā)生空指針異常,所以通過引用訪問靜態(tài)變量實(shí)際上還是直接通過類去訪問的
測(cè)試代碼:
public class StaticTest03 {
public static void main(String[] args) {
Chinese c = new Chinese("張三", "3333333333");
System.out.print("姓名:" + c.name);
System.out.print(" 身份證號(hào):" + c.idNum);
//這里使用引用調(diào)用了靜態(tài)變量nation
System.out.println(" 國(guó)籍:" + c.nation);
//將引用c置空
c = null;
/*
這兩個(gè)實(shí)例變量當(dāng)引用c置空后就它所指向的實(shí)例就無法從堆內(nèi)存中找到了
System.out.print("姓名:" + c.name);
System.out.print(" 身份證號(hào):" + c.idNum);
*/
//而因?yàn)閚ation是靜態(tài)變量,存放在方法區(qū)中,還是可以找到;
System.out.println(" 國(guó)籍:" + c.nation);
}
}
class Chinese {
// 姓名; 實(shí)例變量
String name;
//身份證號(hào); 實(shí)例變量
String idNum;
//國(guó)籍; 靜態(tài)變量
static String nation = "中國(guó)";
// 無參構(gòu)造器
public Chinese() {}
// 有參構(gòu)造器
public Chinese(String myName, String myIdNum) {
this.name = myName;
this.idNum = myIdNum;
}
}
小總結(jié)
通過代碼測(cè)試,可以知道通過引用也可以訪問靜態(tài)變量,但是知道就行,不要這樣做,因?yàn)椴捎靡梅绞皆L問的時(shí)候,程序員會(huì)認(rèn)為你訪問的是實(shí)例變量,就會(huì)給程序員產(chǎn)生困惑;
總結(jié)
對(duì)于靜態(tài)變量,我們需要掌握它的用法、特點(diǎn),要記住,只要與實(shí)例有關(guān)系的,必須先創(chuàng)建對(duì)象,然后通過引用的方式去訪問,如果空引用訪問實(shí)例相關(guān)的成員,必然會(huì)出現(xiàn)空指針異常;所有靜態(tài)相關(guān)的,直接使用類名去訪問,雖然靜態(tài)相關(guān)的成員也能使用引用去訪問,但是不要去使用,養(yǎng)成良好的代碼習(xí)慣很重要!
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
Spring Boot 單元測(cè)試和集成測(cè)試實(shí)現(xiàn)詳解
這篇文章主要介紹了Spring Boot 單元測(cè)試和集成測(cè)試實(shí)現(xiàn)詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-09-09
如何解決Field name doesn‘t have a defau
這篇文章主要介紹了如何解決Field name doesn‘t have a default value報(bào)錯(cuò)問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-02-02
arthas?jprofiler做復(fù)雜鏈路的調(diào)用分析
這篇文章主要為大家介紹了arthas?jprofiler做復(fù)雜鏈路的調(diào)用分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06
SpringMVC中利用@InitBinder來對(duì)頁(yè)面數(shù)據(jù)進(jìn)行解析綁定的方法
本篇文章主要介紹了SpringMVC中利用@InitBinder來對(duì)頁(yè)面數(shù)據(jù)進(jìn)行解析綁定的方法,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2018-03-03
web 容器的設(shè)計(jì)如何實(shí)現(xiàn)
這篇文章主要介紹了web 容器的設(shè)計(jì)如何實(shí)現(xiàn)的相關(guān)資料,本文旨在介紹如何設(shè)計(jì)一個(gè)web容器,只探討實(shí)現(xiàn)的思路,并不涉及過多的具體實(shí)現(xiàn)。把它分解劃分成若干模塊和組件,每個(gè)組件模塊負(fù)責(zé)不同的功能,需要的朋友可以參考下2016-12-12

