1. 程式人生 > >Java學習筆記第十天:3_內部類

Java學習筆記第十天:3_內部類

/*
	內部類概述:
		把類定義在其他類的內部,這個類就被稱為內部類。
		舉例:在類A中定義了一個類B,類B就是內部類。
	
	內部的訪問特點:
		A:內部類可以直接訪問外部類的成員,包括私有。
		B:外部類要訪問內部類的成員,必須建立物件。
	
*/
class Outer {
	private int num = 10;
	
	class Inner {
		public void show() {
			System.out.println(num);
		}
	}
	
	public void method() {
		//找不到符號
		//show();
	
		Inner i = new Inner();
		i.show();
	}
	
}

class InnerClassDemo {
	public static void main(String[] args) {
	
	}
}
/*
	內部類位置
		成員位置:在成員位置定義的類,被稱為成員內部類。	
		區域性位置:在區域性位置定義的類,被稱為區域性內部類。
		
		
	成員位置:在成員位置定義的類,被稱為成員內部類。	
		
*/
class Outer {
	private int num = 10;

	//成員位置
	/*
	class Inner {
		
	}
	*/
	

	public void method() {
		//區域性位置
		class Inner {
		
		}
	}
}

class InnerClassDemo2 {
	public static void main(String[] args) {
		
	}
}
/*
	成員內部類:
		如何直接訪問內部類的成員。
		外部類名.內部類名 物件名 = 外部類物件.內部類物件;
*/
class Outer {
	private int num = 10;
	
	class Inner {
		public void show() {
			System.out.println(num);
		}
	}
}

class InnerClassDemo3 {
	public static void main(String[] args) {
		//需求:我要訪問Inner類的show()方法
		//Inner i = new Inner();
		//i.show();
		
		//格式:外部類名.內部類名 物件名 = 外部類物件.內部類物件;
		Outer.Inner oi = new Outer().new Inner();
		oi.show();
	}
}
/*
	成員內部類的修飾符:
		private 為了保證資料的安全性
		static 為了方便訪問資料
			注意:靜態內部類訪問的外部類資料必須用靜態修飾。
	
	案例:我有一個人(人有身體,身體內有心臟。)
		
		class Body {
			private class Heart {
				public void operator() {
					System.out.println("心臟搭橋");
				}
			}
			
			public void method() {
				if(如果你是外科醫生) {
					Heart h = new Heart();
					h.operator();
				}
			}
		}
		
		按照我們剛才的講解,來使用一下
		Body.Heart bh = new Body().new Heart();
		bh.operator();
		//加了private後,就不能被訪問了,那麼,怎麼玩呢?
		Body b =  new Body();
		b.method();
*/
class Outer {
	private int num = 10;
	private static int num2 = 100;
	
	//內部類用靜態修飾是因為內部類可以看出是外部類的成員
	public static class Inner {
		public void show() {
			//System.out.println(num);
			System.out.println(num2);
		}

		public static void show2() {
			//System.out.println(num);
			System.out.println(num2);
		}		
	}
}

class InnerClassDemo4 {
	public static void main(String[] args) {
		//使用內部類
		// 限定的新靜態類
		//Outer.Inner oi = new Outer().new Inner();
		//oi.show();
		//oi.show2();
		
		//成員內部類被靜態修飾後的訪問方式是:
		//格式:外部類名.內部類名 物件名 = new 外部類名.內部類名();
		Outer.Inner oi = new Outer.Inner();
		oi.show();
		oi.show2();
		
		//show2()的另一種呼叫方式
		Outer.Inner.show2();
	}
}
/*
	區域性內部類
		A:可以直接訪問外部類的成員
		B:在區域性位置,可以建立內部類物件,通過物件呼叫內部類方法,來使用區域性內部類功能
	
	面試題:
		區域性內部類訪問區域性變數的注意事項?
		A:區域性內部類訪問區域性變數必須用final修飾
		B:為什麼呢?
			區域性變數是隨著方法的呼叫而呼叫,隨著呼叫完畢而消失。
			而堆記憶體的內容並不會立即消失。所以,我們加final修飾。
			加入final修飾後,這個變數就成了常量。既然是常量。你消失了。
			我在記憶體中儲存的是資料20,所以,我還是有資料在使用。
*/
class Outer {
	private int num  = 10;
	
	public void method() {
		//int num2 = 20;
		//final int num2 = 20;
		class Inner {
			public void show() {
				System.out.println(num);
				//從內部類中訪問本地變數num2; 需要被宣告為最終型別
				System.out.println(num2);//20
			}
		}
		
		//System.out.println(num2);
		
		Inner i = new Inner();
		i.show();
	}
}

class InnerClassDemo5 {
	public static void main(String[] args) {
		Outer o = new Outer();
		o.method();
	}
}
/*
	匿名內部類
		就是內部類的簡化寫法。

	前提:存在一個類或者介面
		這裡的類可以是具體類也可以是抽象類。
	
	格式:
		new 類名或者介面名(){
			重寫方法;
		}
		
	本質是什麼呢?
		是一個繼承了該類或者實現了該介面的子類匿名物件。
*/
interface Inter {
	public abstract void show();
	public abstract void show2();
}

class Outer {
	public void method() {
		//一個方法的時候
		/*
		new Inter() {
			public void show() {
				System.out.println("show");
			}
		}.show();
		*/
		
		//二個方法的時候
		/*
		new Inter() {
			public void show() {
				System.out.println("show");
			}
			
			public void show2() {
				System.out.println("show2");
			}
		}.show();
		
		new Inter() {
			public void show() {
				System.out.println("show");
			}
			
			public void show2() {
				System.out.println("show2");
			}
		}.show2();
		*/
		
		//如果我是很多個方法,就很麻煩了
		//那麼,我們有沒有改進的方案呢?
		Inter i = new Inter() { //多型
			public void show() {
				System.out.println("show");
			}
			
			public void show2() {
				System.out.println("show2");
			}
		};
		
		i.show();
		i.show2();
	}
}

class InnerClassDemo6 {
	public static void main(String[] args) {
		Outer o = new Outer();
		o.method();
	}
}
/*
	面試題:
		要求請填空分別輸出30,20,10。
		
	注意:
		1:內部類和外部類沒有繼承關係。
		2:通過外部類名限定this物件
			Outer.this
*/
class Outer {
	public int num = 10;
	class Inner {
		public int num = 20;
		public void show() {
			int num = 30;
			System.out.println(num);
			System.out.println(this.num);
			//System.out.println(new Outer().num);
			System.out.println(Outer.this.num);
		}
	}
}
class InnerClassTest {
	public static void main(String[] args) {
		Outer.Inner oi = new Outer().new Inner();
		oi.show();
	}	
}
/*
	匿名內部類在開發中的使用
*/
interface Person {
	public abstract void study();
}

class PersonDemo {
	//介面名作為形式引數
	//其實這裡需要的不是介面,而是該介面的實現類的物件
	public void method(Person p) {
		p.study();
	}
}

//實現類
class Student implements Person {
	public void study() {
		System.out.println("好好學習,天天向上");
	}
}

class InnerClassTest2 {
	public static void main(String[] args) {
		//測試
		PersonDemo pd = new PersonDemo();
		Person p = new Student();
		pd.method(p);
		System.out.println("--------------------");
		
		//匿名內部類在開發中的使用
		//匿名內部類的本質是繼承類或者實現了介面的子類匿名物件
		pd.method(new Person(){
			public void study() {
				System.out.println("好好學習,天天向上");
			}
		});
	}
}
/*
	匿名內部類面試題:
		按照要求,補齊程式碼
			interface Inter { void show(); }
			class Outer { //補齊程式碼 }
			class OuterDemo {
				public static void main(String[] args) {
					  Outer.method().show();
				  }
			}
			要求在控制檯輸出”HelloWorld”
*/
interface Inter { 
	void show(); 
	//public abstract
}

class Outer { 
	//補齊程式碼
	public static Inter method() {
		//子類物件 -- 子類匿名物件
		return new Inter() {
			public void show() {
				System.out.println("HelloWorld");
			}
		};
	}
}

class OuterDemo {
	public static void main(String[] args) {
		Outer.method().show();
		/*
			1:Outer.method()可以看出method()應該是Outer中的一個靜態方法。
			2:Outer.method().show()可以看出method()方法的返回值是一個物件。
				又由於介面Inter中有一個show()方法,所以我認為method()方法的返回值型別是一個介面。
		*/
	}
}