Java学习笔记之二

Java学习笔记之二
一、 面向对象基础部分
1、this、super关键字
/**
 * @author Administrator
 * 
 * @description 测试基类
 * @history
 */
public class BaseTest {
	protected String pwd; // 定义protected属性pwd
	public BaseTest(){
		// 无参数构造方法
	}
	public String sayHelloWorld(){
		return "hello world";
	}
}

/**
 * @author Administrator
 * 
 * @description this,super关键字测试类
 * @history
 */
public class MyTestDemo extends BaseTest{
	// notes:测试使用而已,代码功能模拟而已
	private String username; // 定义用户昵称私有属性private修饰
	private String password;
	public MyTestDemo(String username){ // 构造方法
		this();// this()调用无参数构造方法
		this.username = username; // this.属性访问当前username属性
		// super.属性调用父类属性
		this.password = super.pwd;
	}
	public MyTestDemo(){
		// 无参数构造方法
		super(); // 调用父类无参数构造方法
	}
	public void sayHello(){
		this.sayWord(); // this.方法调用当前对象的sayWord方法
		super.sayHelloWorld(); // super.方法调用父类sayHelloWorld方法
	}
	public String sayWord(){
		return "hello this";
	}
	// this表示当前对象
	public boolean equals(Object obj){
		MyTestDemo other = (MyTestDemo)obj; // 传入的比较对象,强制转换操作
		// 简单模拟,如果对象的username属性值相同那么就认为相同
		if(this == other){
			return true;
		}
		if(other instanceof MyTestDemo){
			if(this.username.equals(other.username)){
				return true;
			}
		}
		return false;
	}
	/**
	 *@description 
	 *@param args
	 */
	public static void main(String[] args) {
		// this关键字绑定了当前该对象,通过this.属性、this.方法
		// super关键字用在子类要调用父类的属性或者方法中,通过super.属性、super.方法
		MyTestDemo obj1 = new MyTestDemo("hello-java"); // 定义并实例化对象
		obj1.sayHello(); // 调用对象的sayHello方法
		MyTestDemo obj2 = new MyTestDemo("hello-java");
		// equals方法传入的参数类型为object,传入的obj2会自动向上转型处理
		boolean result = obj1.equals(obj2); 
		System.out.println(result); // true
	}

}

2、static、final关键字
/**
 * @author Administrator
 * 
 * @description final和static关键字学习
 * @history
 */
public class FinalAndStaticTestDemo {
	/**
	 *@description 
	 *@param args
	 */
	public static void main(String[] args) {
		// final关键字主要有三种场景
		// 1、final定义的变量为常量,比如基本数据类型的值一旦初始化了不能改变
		// 2、final定义的类为终结类,不能再被继承,比如string类型
		// 3、final定义的方法为终结方法,不能被覆写,否则编译器会报错
		final String str = "helloworld"; // final修饰变为常量不能改变
		//str = "hellojava"; // eclipse提示:The final local variable str cannot be assigned. 
		
		// 进一步思考下面两个问题
		String s1 = "hello";
		String s2 = "hello1";
		final String s = s1;
		s1 = s2;
		System.out.println(s); // hello还是hello1呢?
		
		// 引用不可变,内容可变是这个意思吗?
		StringBuffer sb = new StringBuffer();
		sb.append("hello");
		final String fs = sb.toString();
		final StringBuffer fsb = sb;
		sb.append("world");
		System.out.println(fs);// hello还是helloworld呢?
		System.out.println(fsb.toString()); // hello还是helloworld呢?
		
		// static关键字比较类似也有下面三种场景
		// static修饰的变量为静态变量,它不同于对象/实例变量,是关联在某个类上的
		// static修饰的类为静态类
		// static修饰的方法为静态方法,它也不同于对象/实例方法,是关联在某个类上的
		// static int i = 100;
		// static class HelloWorld{ //... }
		// public static void sayHeloWorld(){ //... }
	}
	// 静态变量和对象变量
	static int i = 100; // 定义静态变量i
	int j = 100; // 定义成员变量/实例变量
	public static void sayHelloWorld(){ // 定义静态方法sayhelloworld
		System.out.println("hello world");
	}
	static class HelloWorld{ // 定义内部类
		// ...
	}

}

3、引用传递介绍和应用
/**
 * @author Administrator
 * 
 * @description 引用传递的简单学习
 * @history
 */
class DemoTest{
	String ide = "eclipse"; // 为了方便访问暂不封装
}
public class MyDemoTest{
	/**
	 *@description 
	 *@param args
	 */
	public static void main(String[] args){
		// 对象属性值访问和变更
		DemoTest dt1 = new DemoTest();
		System.out.println(dt1.ide); // eclipse
		dt1.ide = "netbeans"; // 改变ide属性值
		System.out.println(dt1.ide); // netbeans
		
		function(dt1); // 调用方法function
		System.out.println(dt1.ide); // jcreator
		
		// String类赋值和变更
		String ide = "eclipse";
		function(ide);
		System.out.println(ide); // eclipse而不是netbeans
	}
	private static void function(String ide) {
		ide = "netbeans"; // 赋值为netbeans
	}
	// 测试方法function
	private static void function(DemoTest dt2) {
		dt2.ide = "jcreator"; // 改变ide属性值
	}
}

4、静态块、代码块、同步块
/**
 * @author Administrator
 * 
 * @description 静态块、代码块、同步块学习测试类
 * @history
 */
class BlockDemo {
	static {
		// 在类中定义的static{ //...}为静态代码块
		// 静态代码块的程序只会被初始化一次
		System.out.println("BlockDemo->static");
	}
	{
		// 在类中定义的{ //...}为构造代码块
		// 构造代码块和对象一起初始化的
		System.out.println("BlockDemo->{}");
	}
	public BlockDemo() {
		System.out.println("BlockDemo->BlockDemo");
	}
	// 构造方法、构造代码块和静态代码块初始化顺序呢?
	// 静态块、构造块、构造方法 // 表面初步分析,具体有待研究
}
public class MyDemo {
	/**
	 * @description
	 * @param args
	 */
	public static void main(String[] args) {
		// 静态块、代码块、同步块
		{
			// 在方法中用{ //... }定义的为普通代码块
			int x = 100;
			System.out.println(x); // 100 相当于局部变量
		}
		int x = 200;
		System.out.println(x); // 200

		new BlockDemo();
		new BlockDemo();
		// BlockDemo->static static代码块只会被初始化一次
		// BlockDemo->{}
		// BlockDemo->BlockDemo
		// BlockDemo->{}
		// BlockDemo->BlockDemo
	}
}

5、构造方法、匿名对象、单例定义以及实现
/**
 * @author Administrator
 * 
 * @description 构造方法、匿名对象学习测试类
 * @history
 */
public class MyDemoTest {
	private String username; // 定义私有属性username
	
	// 构造方法的作用是初始化操作
	// 构造方法是一种特殊的方法,构造方法也有重载
	public MyDemoTest(){
		// 无参数构造方法
	}
	public MyDemoTest(String username) { // 定义构造方法
		this.username = username; // 对属性初始化赋值操作
	}
	/**
	 *@description 
	 *@param args
	 */
	public static void main(String[] args) {
		// 实例化一个匿名对象
		// 匿名对象不在栈中开辟空间赋地址值
		new MyDemoTest("eclipse"); 
	}
}

/**
 * @author Administrator
 * 
 * @description 单例学习测试类
 * @history
 */
class Singleton {
	private Singleton() {
		// 将构造方法私有化,外部不能直接调用
		System.out.println("private Singleton(){//...}");
	}
	// 饿汉式获取单例代码
	private static Singleton instance = new Singleton();
	// 对外提供一个getInstance方法获取单例
	public static Singleton getIntance() {
		return instance;
	}
}
class Singleton2{
	private Singleton2(){
		// 构造方法私有化,外部不能直接调用
		System.out.println("private Singleton2(){//...}");
	}
	// 懒汉式获取单例代码
	private static Singleton2 instance = null;
	private static Object lock = new Object();
	
	// 获取单例方法,效率比较低
	// 每次进入都要进行锁判断,而实际情况我们是第一次null比较特殊
	/*public static Singleton2 getIntance(){
		synchronized (lock) {
			if (instance == null) { // 如果为null进行实例化对象操作
				instance = new Singleton2();
			}
		}
		return instance;
	}*/
	// 上面同步代码的另外一种写法如下,效率比较低
	// public static synchronized getInstance(){ //...}
	
	// 对同步代码进行改写
	public static Singleton2 getIntance(){
		if(instance == null){
			synchronized(lock){
				if(instance == null){ // 进行双重检验操作
					instance = new Singleton2();
				}
			}
			// 如果换为下面这种形式那么就可能不是产生一个实例对象了
			/*synchronized(lock){
				instance = new Singleton2();
			}*/
		}
		return instance;
	}
}
public class SingletonTest {
	/**
	 *@description 
	 *@param args
	 */
	public static void main(String[] args) {
		Singleton.getIntance(); // 饿汉式获取单例
		Singleton2.getIntance(); // 懒汉式获取单例
	}
}

6、内部类、匿名内部类应用
class Outer{ // 定义外部类
	private String info = "helloworld" ; // 定义外部类属性
	private static String info2 = "helloeclipse"; // 定义静态变量
	class Inner{  // 定义内部类
		public void print(){	
			// 内部类的好处之一:直接访问外部类属性
			System.out.println(info) ; // 直接访问外部类属性
		}
	}
	static class StaticInner{ // 通过static定义的内部类为外部类
		public void print(){
			System.out.println(info2);
		}
	}
	public void fun(){
		new Inner().print() ; // 通过内部类的实例化对象调用方法
	}
}
/**
 * @author Administrator
 * 
 * @description 内部类以及匿名内部类学习测试类
 * @history
 */
public class InnerClassDemo{
	/**
	 *@description 
	 *@param args
	 */
	public static void main(String[] args){
		new Outer().fun() ; // 调用外部类的fun()方法
		// 另外一种实例化方式
		Outer out = new Outer();
		Outer.Inner in = out.new Inner();
		in.print();
		
		// 通过外部类.内部类实例化内部类对象
		//StaticInner sin = new Outer.StaticInner(); //需要导入外部类所在的包
		//sin.print();
	}
}

二、 面向对象高级部分
1、继承的进一步讨论
class Base{
	// 定义基类
	void print(){
		System.out.println("Base->print()");
	}
	public void print1(){
		// 父类方法的访问权限,子类覆写该方法的时候不能降低权限
	}
}
class ZiLei extends Base{
	// 定义子类覆写print方法
	void print(){
		// 子类访问父类中的方法,前面super关键字内容
		// super.print();
		System.out.println("Super->print()");
	}
	/*void print1(){
		// Cannot reduce the visibility of the inherited method from Base
	}*/
	// 覆写override和重载overload区别
	// 覆写是父类子类之间的关系,重载是同一个类之间的关系
}
/**
 * @author Administrator
 * 
 * @description 继承中的方法覆写
 * @history
 */
public class OverrideDemo{
	public static void main(String args[]){
		new ZiLei().print(); // 调用的内容是覆写后的方法实现
		// 另外java中只能够单继承,不同于C++,但是可以多重继承
		/*class A{
			// ...
		}
		class B extends A{  //class C extends A,B{ // ...}编译错误
			// ...
		}
		class C extends B{
			// ...
		}*/
	}
}

2、接口和抽象类进一步讨论
/**
 * @author Administrator
 * 
 * @description 接口测试类
 * @history
 */
interface Interface{
	public static final String INFO = "helloeclipse";
	public void print(); // 定义方法
	public abstract void printInfo();
}
class I implements Interface{
	public void print() {
		// 方法实现
		System.out.println(INFO); // helloeclipse
	}
	public void printInfo() {
		// ...
	}
}
public class InterfaceTestDemo {
	/**
	 *@description 
	 *@param args
	 */
	public static void main(String[] args) {
		// 接口interface
		// 接口interface和抽象类abstract一样不能直接实例化必须通过子类来实现
		new Interface(){
			public void print() {
				// 匿名内部类,方法实现
				System.out.println(INFO); // helloeclipse
			}
			public void printInfo() {
				// ...
			}
		};
		// 接口和抽象类对比,接口更像一种规范、抽象类更像一种模板
		// 抽象类定义一些公共的实现,以及一些未实现的方法交给子类具体实现特定的功能-模板方法
	}
}

/**
 * @author Administrator
 * 
 * @description 抽象类学习测试类
 * @history
 */
abstract class Abstract{ // 定义抽象类
	public final static String INFO = "helloeclipse"; // 定义全局变量/常量
	abstract public void print(); // 定义抽象方法
	// 抽象类不能直接实例化,抽象类的子类如果不是抽象类那么一定要实现所有的抽象方法
}
class AbstractZiLei extends Abstract{ // 定义抽象子类
	public void print() { // 实现子类方法
		System.out.println(INFO);
	} 
}
public class AbstractTestDemo {
	/**
	 *@description 
	 *@param args
	 */
	public static void main(String[] args) {
		// 接口和抽象类interface,abstract
		// 抽象类:含有抽象方法的类叫做抽象类用abstract修饰的
		new Abstract(){
			public void print() {
				System.out.println(INFO); // 匿名抽象类实现
			}
		};
		// 抽象类能不能被final修饰呢?final定义的类为终结类,抽象类是需要子类来实现的
		// final abstract class Abstract{ //... } // 编译报错
	}
}

3、object类和包装类进一步讨论
/**
 * @author Administrator
 * 
 * @description Object类学习测试类
 * @history
 */
public class ObjectTestDemo {
	/**
	 *@description 
	 *@param args
	 * @throws ClassNotFoundException 
	 */
	public static void main(String[] args) throws ClassNotFoundException {
		// Object类学习测试代码
		// 1、toString方法
		// 2、equals和hashCode方法
		// 3、getClass方法
		ObjectTestDemo otd = new ObjectTestDemo();
		System.out.println(otd);
		System.out.println(otd.toString());// 现实调用toString方法
	    /*
	     * Object类中toString方法
	     * public String toString() {
	    	return getClass().getName() + "@" + Integer.toHexString(hashCode());
	    }
	    * Object类中equal方法
	    * public boolean equals(Object obj) {
	    	return (this == obj);
	    }*/
		// 定义昵称和年龄属性,如果昵称和年龄相同则equal方法返回true
		ObjectTestDemo otd1 = new ObjectTestDemo();
		otd1.nickname = "eclipse";
		otd1.age = 20;
		ObjectTestDemo otd2 = new ObjectTestDemo();
		otd2.nickname = "eclipse";
		otd2.age = 20;
		System.out.println(otd1==otd2); // false
		System.out.println(otd1.equals(otd2)); // 如果不覆写的话返回的一直是false
		
		// hashCode方法和equals方法的关系,没有绝对的关系
		// 具体参看源代码的注释说明,在实际用中尽量同时覆写这两个方法,保持一致性
		
		// public final native Class<?> getClass();
		// getClass方法是一个本地方法,调用底层代码,返回当前对象对应的一个类对象实例
		// 类对象实例,理解为内存中对应的那份字节码对象,java反射内容初步学习
		// 通过字节码对象构造出一个个的对象实例
		Class claz = otd.getClass();
		Class claz1 = ObjectTestDemo.class;
		Class claz2 = Class.forName("ObjectTestDemo"); // 类的完整路径
		System.out.println(otd.getClass()); // class ObjectTestDemo
		// VM内存中只会产生一份字节码对象
		System.out.println(claz == claz1); // true
		
	}
	// 方法模拟,添加两个属性昵称和年龄
	private String nickname;
	private int age;
	
	// 注意参数类型不要写成了ObjectTestDemo
	public boolean equals(Object obj) {
		ObjectTestDemo otd = (ObjectTestDemo) obj; // 向上转型,强制转换
		if (this == otd) {
			return true;
		}
		if (otd instanceof ObjectTestDemo) {
			if (this.nickname.equals(otd.nickname) && this.age == otd.age) {
				return true;
			}
		}
		return false;
	}
	public String toString(){
		return "helloworld"; //覆写toString方法,返回helloworld
	}
}

你可能感兴趣的:(java,面向对象,学习笔记)