亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術文章
文章詳情頁

Java private修飾符失效的原因

瀏覽:21日期:2022-08-22 09:47:40

失效之Java內部類

在一個內部類里訪問外部類的private成員變量或者方法。

public class OuterClass { private String language = 'en'; private String region = 'US'; public class InnerClass { public void printOuterClassPrivateFields() { String fields = 'language=' + language + ';region=' + region; System.out.println(fields); } } public static void main(String[] args) { OuterClass outer = new OuterClass(); OuterClass.InnerClass inner = outer.new InnerClass(); inner.printOuterClassPrivateFields(); }}

查看原因

使用javap命令查看一下生成的class文件

15:30 javap -c OuterClassCompiled from 'OuterClass.java'public class OuterClass extends java.lang.Object{public OuterClass(); Code: 0: aload_0 1: invokespecial #11; //Method java/lang/Object.'<init>':()V 4: aload_0 5: ldc #13; //String en 7: putfield #15; //Field language:Ljava/lang/String; 10: aload_0 11: ldc #17; //String US 13: putfield #19; //Field region:Ljava/lang/String; 16: return public static void main(java.lang.String[]); Code: 0: new #1; //class OuterClass 3: dup 4: invokespecial #27; //Method '<init>':()V 7: astore_1 8: new #28; //class OuterClassInnerClass 11: dup 12: aload_1 13: dup 14: invokevirtual #30; //Method java/lang/Object.getClass:()Ljava/lang/Class; 17: pop 18: invokespecial #34; //Method OuterClassInnerClass.'<init>':(LOuterClass;)V 21: astore_2 22: aload_2 23: invokevirtual #37; //Method OuterClassInnerClass.printOuterClassPrivateFields:()V 26: return static java.lang.String access0(OuterClass); Code: 0: aload_0 1: getfield #15; //Field language:Ljava/lang/String; 4: areturn static java.lang.String access1(OuterClass); Code: 0: aload_0 1: getfield #19; //Field region:Ljava/lang/String; 4: areturn }

在這里有一個OuterClass方法,

static java.lang.String access0(OuterClass); Code: 0: aload_0 1: getfield #15; //Field language:Ljava/lang/String; 4: areturn static java.lang.String access1(OuterClass); Code: 0: aload_0 1: getfield #19; //Field region:Ljava/lang/String; 4: areturn }

根據(jù)注釋,可以知道access0返回outerClass的language屬性,access1返回outerClass的region屬性,并且這兩個方法都接受OuterClass的實例作為參數(shù),對這兩個方法進行反編譯。

15:37 javap -c OuterClassInnerClassCompiled from 'OuterClass.java'public class OuterClassInnerClass extends java.lang.Object{final OuterClass this0; public OuterClassInnerClass(OuterClass); Code: 0: aload_0 1: aload_1 2: putfield #10; //Field this0:LOuterClass; 5: aload_0 6: invokespecial #12; //Method java/lang/Object.'<init>':()V 9: return public void printOuterClassPrivateFields(); Code: 0: new #20; //class java/lang/StringBuilder 3: dup 4: ldc #22; //String language= 6: invokespecial #24; //Method java/lang/StringBuilder.'<init>':(Ljava/lang/String;)V 9: aload_0 10: getfield #10; //Field this0:LOuterClass; 13: invokestatic #27; //Method OuterClass.access0:(LOuterClass;)Ljava/lang/String; 16: invokevirtual #33; //Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder; 19: ldc #37; //String ;region= 21: invokevirtual #33; //Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder; 24: aload_0 25: getfield #10; //Field this0:LOuterClass; 28: invokestatic #39; //Method OuterClass.access1:(LOuterClass;)Ljava/lang/String; 31: invokevirtual #33; //Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder; 34: invokevirtual #42; //Method java/lang/StringBuilder.toString:()Ljava/lang/String; 37: astore_1 38: getstatic #46; //Field java/lang/System.out:Ljava/io/PrintStream; 41: aload_1 42: invokevirtual #52; //Method java/io/PrintStream.println:(Ljava/lang/String;)V 45: return}

下面代碼調用access$0的代碼,其目的是得到OuterClass的language 私有屬性。

13: invokestatic #27; //Method OuterClass.access$0:(LOuterClass;)Ljava/lang/String;

下面代碼調用了access$1的代碼,其目的是得到OutherClass的region 私有屬性。

28: invokestatic #39; //Method OuterClass.access$1:(LOuterClass;)Ljava/lang/String;

即,在內部類構造的時候,會有外部類的引用傳遞進來,并且作為內部類的一個屬性,所以內部類會持有一個其外部類的應用。this$0就是內部類持有的外部類引用,通過構造方法傳遞引用并賦值。

final OuterClass this0; public OuterClassInnerClass(OuterClass); Code: 0: aload_0 1: aload_1 2: putfield #10; //Field this$0:LOuterClass; 5: aload_0 6: invokespecial #12; //Method java/lang/Object.'<init>':()V 9: return

繼續(xù)失效

public class AnotherOuterClass { public static void main(String[] args) { InnerClass inner = new AnotherOuterClass().new InnerClass(); System.out.println('InnerClass Filed = ' + inner.x); } class InnerClass { private int x = 10; } }

和上面一樣,使用Javap反編譯一下

16:03 javap -c AnotherOuterClassInnerClassCompiled from 'AnotherOuterClass.java'class AnotherOuterClassInnerClass extends java.lang.Object{final AnotherOuterClass this0; AnotherOuterClassInnerClass(AnotherOuterClass); Code: 0: aload_0 1: aload_1 2: putfield #12; //Field this0:LAnotherOuterClass; 5: aload_0 6: invokespecial #14; //Method java/lang/Object.'<init>':()V 9: aload_0 10: bipush 10 12: putfield #17; //Field x:I 15: return static int access0(AnotherOuterClassInnerClass); Code: 0: aload_0 1: getfield #17; //Field x:I 4: ireturn }

編譯器自動生成了一個access$0一次來獲取x的值AnotherOuterClass.class的反編譯結果

16:08 javap -c AnotherOuterClassCompiled from 'AnotherOuterClass.java'public class AnotherOuterClass extends java.lang.Object{public AnotherOuterClass(); Code: 0: aload_0 1: invokespecial #8; //Method java/lang/Object.'<init>':()V 4: return public static void main(java.lang.String[]); Code: 0: new #16; //class AnotherOuterClassInnerClass 3: dup 4: new #1; //class AnotherOuterClass 7: dup 8: invokespecial #18; //Method '<init>':()V 11: dup 12: invokevirtual #19; //Method java/lang/Object.getClass:()Ljava/lang/Class; 15: pop 16: invokespecial #23; //Method AnotherOuterClassInnerClass.'<init>':(LAnotherOuterClass;)V 19: astore_1 20: getstatic #26; //Field java/lang/System.out:Ljava/io/PrintStream; 23: new #32; //class java/lang/StringBuilder 26: dup 27: ldc #34; //String InnerClass Filed = 29: invokespecial #36; //Method java/lang/StringBuilder.'<init>':(Ljava/lang/String;)V 32: aload_1 33: invokestatic #39; //Method AnotherOuterClassInnerClass.access0:(LAnotherOuterClassInnerClass;)I 36: invokevirtual #43; //Method java/lang/StringBuilder.append:(I)Ljava/lang/StringBuilder; 39: invokevirtual #47; //Method java/lang/StringBuilder.toString:()Ljava/lang/String; 42: invokevirtual #51; //Method java/io/PrintStream.println:(Ljava/lang/String;)V 45: return }

其中這句話,直接說明通過內部類的實例,獲取到私有屬性x的操作。

invokestatic #39; //Method AnotherOuterClassInnerClass.access0:(LAnotherOuterClass$InnerClass;)I

在官網(wǎng)文檔中是這樣說道的,如果(內部類的)成員和構造方法設定成了私有修飾符,當且僅當其外部類訪問時是允許的。

如何保證不被訪問

使用的方法相當簡單,使用匿名內部類的方法實現(xiàn)

public class PrivateToOuter { Runnable mRunnable = new Runnable(){ private int x=10; @Override public void run() { System.out.println(x); } }; public static void main(String[] args){ PrivateToOuter p = new PrivateToOuter(); //System.out.println('anonymous class private filed= '+ p.mRunnable.x); //not allowed p.mRunnable.run(); // allowed }}

以上就是Java private修飾符失效的原因的詳細內容,更多關于Java private修飾符失效的資料請關注好吧啦網(wǎng)其它相關文章!

標簽: Java
相關文章:
主站蜘蛛池模板: 国产欧美在线视频 | 色婷婷综合欧美成人 | 成人一级片在线观看 | 看黄色一级毛片 | 日韩国产一区二区 | 国产美女无遮挡免费视频 | 欧美金妇欧美乱妇视频 | 国产亚洲欧美一区二区三区 | 老头毛片 | a免费网站 | 国产精品成人va在线观看 | 久热精品视频在线观看99小说 | 亚洲欧美日韩综合在线一区二区三区 | 欧美一级特黄特色大片 | 国产黄色a级 | 日韩经典视频 | 目韩一区二区三区系列片丶 | 久久国产成人精品麻豆 | 天天影视色香欲综合网天天录日日录 | 久久久久久九九 | 欧美三级a做爰在线观看 | 青青草国产精品久久久久 | 国产性精品 | 91精品免费不卡在线观看 | 国产三级91 | 伊人中文字幕在线观看 | 亚洲欧美偷拍自拍 | 国产一极毛片 | 妞干网在线免费视频 | 欧美超高清xoxoxoxo | 国产1024一区二区你懂的 | 久久草在线视频播放 | 亚洲欧美综合网站 | 香蕉97碰碰视频在线看 | zoofilia活体videos新| 亚洲在线免费观看 | 超级97碰碰碰碰久久久久最新 | 日韩欧美一二三 | 香蕉大久久 | 国产毛片一级国语版 | a级黄色毛片视频 |