關(guān)于protected修飾符詳解-源于Cloneable接口
Cloneable接口是一個空接口,僅用于標(biāo)記對象,Cloneable接口里面是沒有clone()方法,clone()方法是Object類里面的方法,默認(rèn)實(shí)現(xiàn)是一個Native方法。
今天在做對象拷貝的時候發(fā)現(xiàn)一個問題
某類實(shí)現(xiàn)Cloneable接口后,還要重新實(shí)現(xiàn)Object類中的clone()方法:
感覺有點(diǎn)奇怪,這個類里也沒做什么特殊的操作啊,就是調(diào)用一下父類的clone方法:
這太麻煩了,我們都知道protected的權(quán)限范圍:
也就是說子類是可以訪問protected修飾的方法的。
接下來按照我們的思路寫代碼實(shí)現(xiàn)
1.去掉User類中的clone方法:
接下來直接調(diào)用Object類中的clone方法:
這時發(fā)現(xiàn),報錯啦?。?! 子類實(shí)例竟然不能調(diào)用父類的clone方法?。?!我們會有疑問:Object類是所有類的父類,那么為什么子類不能訪問父類protected修飾的方法呢?
其實(shí)是因?yàn)椋骸芭c基類不在同一個包中的子類,只能訪問自身從基類繼承而來的受保護(hù)成員,而不能訪問基類實(shí)例本身的受保護(hù)成員”。這句話是什么意思?不要急,接下來我們用代碼詳細(xì)解釋。
所以這里是因?yàn)?,User類和Object類不同包,導(dǎo)致protected方法訪問不了,接下來我們驗(yàn)證一下:
沒有錯誤,可以直接調(diào)用父類protected方法getA:
此時Student類和Person類在同一個包下,假如不在同一個包中呢:
此時報錯了:
我們發(fā)現(xiàn)此時子類不能調(diào)用父類的protected方法了。
為什么應(yīng)用了Cloneable接口的類
通常還必須重寫一個public的clone()方法
這里有兩個原因:
(1) 如果不重寫,由于Object.clone()是proteced屬性,所以這個clone()方法將無法在外部被調(diào)用,更精確地說,無法在目標(biāo)類之外的任何地方調(diào)用。這樣就使得克隆失去了用武之地。
(2) Object.clone()畢竟只是提供了淺層拷貝,對于基本類型的字段,可以說它成功克隆了。但對于對象型字段,它并沒有實(shí)現(xiàn)克隆的功能,僅僅做了一個賦值。試運(yùn)行一下下面的代碼就會更清楚了:
public class Student implements Cloneable { private int id; private String name; public StringBuffer sb = new StringBuffer(""); public Student() { this.id = 744; this.name = "FL"; } public Student(int id, String name) { this.id = id; this.name = name; } public boolean equals(Object obj) { return this.id == ((Student) obj).id; } public String toString() { return "Student id : " + id + " Name " + name; } public static void main(String[] args) throws CloneNotSupportedException { Student s1 = new Student(101, "WangQiang"); Student s2 = (Student) s1.clone(); System.out.println(s1 == s2); System.out.println(s1); System.out.println(s2); s1.sb.append("s1's string"); System.out.println("s2.sb's value = " + s2.sb.toString()); System.out.println(s1.sb==s2.sb); } }
總結(jié)
對于protected的成員或方法,要分子類和基類是否在同一個包中。與基類不在同一個包中的子類,只能訪問自身從基類繼承而來的受保護(hù)成員和方法,而不能訪問基類實(shí)例本身的受保護(hù)成員和方法。在相同包時,protected和public是一樣的。
以上為個人經(jīng)驗(yàn),希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
SpringCloud升級2020.0.x版之OpenFeign簡介與使用實(shí)現(xiàn)思路
在微服務(wù)系統(tǒng)中,我們經(jīng)常會進(jìn)行 RPC 調(diào)用。在 Spring Cloud 體系中,RPC 調(diào)用一般就是 HTTP 協(xié)議的調(diào)用。對于每次調(diào)用,都要經(jīng)過一系列詳細(xì)步驟,接下來通過本文給大家介紹SpringCloud OpenFeign簡介與使用,感興趣的朋友一起看看吧2021-10-10public?static?void?main(String[]?args)使用解讀
這篇文章主要介紹了public?static?void?main(String[]?args)的使用,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01IDEA2020導(dǎo)入非maven項(xiàng)目并部署tomcat的方法
這篇文章主要介紹了IDEA 2020 導(dǎo)入非maven項(xiàng)目并部署tomcat的方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07java 實(shí)現(xiàn)切割文件和合并文件的功能
這篇文章主要介紹了java 實(shí)現(xiàn)切割文件和合并文件的功能的相關(guān)資料,這里實(shí)現(xiàn)文件的切割的實(shí)現(xiàn)代碼和文件合并的實(shí)現(xiàn)代碼,需要的朋友可以參考下2017-07-07Mybatis下動態(tài)sql中##和$$的區(qū)別講解
今天小編就為大家分享一篇關(guān)于Mybatis下動態(tài)sql中##和$$的區(qū)別講解,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2019-03-03quartz的簡單使用、SpringBoot使用和自定義數(shù)據(jù)源集成方式
這篇文章主要介紹了quartz的簡單使用、SpringBoot使用和自定義數(shù)據(jù)源集成方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教<BR>2024-01-01netty-grpc一次DirectByteBuffer內(nèi)存泄露問題
這篇文章主要介紹了netty-grpc一次DirectByteBuffer內(nèi)存泄露問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-12-12Java基于MySQL實(shí)現(xiàn)學(xué)生管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了Java基于MySQL實(shí)現(xiàn)學(xué)生管理系統(tǒng),具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-01-01