黑马程序员——Java基础--多态、内部类、异常

------ Java培训、Android培训、iOS培训、.Net培训、期待与您交流! -------

多态、内部类、异常

一、多态
1、多态:可以理解为事物存在的多种体现形态。
例:动物中猫、狗
猫这个对象对应的类型是猫类型:猫 y = new 猫();
同时猫也是动物中的一种,也可以把猫称为动物。
动物 y = new 猫();
动物是猫和狗具体事物中抽取出来的父类型。
父类型引用指向了子类对象。
2、多态的体现
  父类的引用指向了自己的子类对象。
  父类的引用也可以接收自己的子类对象。
3、多态的前提
  必须是类与类之间有关系,要么是继承,要么是实现。通常还有一个前提就是:存在覆盖。
4、多态的好处
  多态的出现大大的提高了程序的扩展性。
  弊端:
提高了扩展性,但是只能使用父类的引用访问父类中的成员。 
5、多态的应用
/*
需求:
动物
猫、狗
*/
abstract class Animal
{
	abstract void eat();	
}
class Cat extends Animal
{
	public void eat()
	{
		System.out.println("吃鱼");
	}
	public void catchMouse()
	{
		System.out.println("抓老鼠");
	}
}
class Dog extends Animal 
{
	public void eat()
	{
		System.out.println("吃骨头");
	}
	public void KanJia()
	{
		System.out.println("看家");
	}	
}

class Pig extends Animal
{
	public void eat()
	{
		System.out.println("饲料");
	}
	public void gongdi()
	{
		System.out.println("拱地");
	}
}
//------------------------------------------------------------
class DuoTaiDemo1
{
	public static void main(String[] args)
	{
		Animal a = new Cat();//类型提升。向上转型
		a.eat();
		//如果想要调用猫的特有方法时,如何操作?
		//强制将父类的引用,转成子类类型,向下转型。
		Cat c = (Cat)a;
		c.catchMouse();
/*
	千万不要出现这样的动作,就是将父类的对象转成子类类型。
<span style="white-space:pre">	</span>我们能转换的是父类应用指向了自己的子类对象时,该应用可以被提升,也可以被强制转换
<span style="white-space:pre">	</span>多态自始至终都是子类对象做着变化。
		Animal a = new Animal();
		Cat c = (Cat)a;
<span style="white-space:pre">		</span>这是错误的
*/
		function(new Cat());
		function(new Dog());		
	}
	public static void function(Animal a)//Animal a = new Cat();
	{
		a.eat();
		//instanceof判断是否是cat的类型
		if(a instanceof Cat)
		{
			Cat c = (Cat)a;
			c.catchMouse();
		}
		else if(a instanceof Dog)
		{
			Dog c = (Dog)a;
			c.KanJia();
		}
	}
}
练习一:
/*
基础班学生:
	学习,睡觉
高级版学生:
	学习,睡觉

可以将这两类事物进行抽取。

*/
abstract class Student
{
	public abstract void study();
	public void sleep()
	{
		System.out.println("躺着睡");
	}
}
//单独封装操作学生的对象
class DoStudent
{
	public void doSome(Student stu)
	{
		stu.study();
		stu.sleep();
	}
}
class BaseStudent extends Student
{
	public void study()
	{
		System.out.println("base study");
	}
	public void sleep()
	{
		System.out.println("坐着睡");
	}
}
class AdvStudent extends Student
{
	public void study()
	{
		System.out.println("adv study");
	}
}

class DuoTaiDemo3
{
	public static void main(String[] args)
	{
		DoStudent ds = new DoStudent();
		ds.doSome(new BaseStudent());
		ds.doSome(new AdvStudent());


/*		BaseStudent bs = new BaseStudent();
		bs.study();
		bs.sleep();
		AdvStudent as = new AdvStudent();
		as.study();
		as.sleep();
*/
	}
}
6、在多态中成员函数的特点:
在编译时期:参阅引用型变量所属的类中是否有调用的方法,如果有,编译通过,如果没有编译失败
在运行时期:参阅对象所属的类中是否有调用的方法。
简单总结就是:成员函数在多态调用时,编译看左边,运行看右边。
<*>面试环节--
在多态中,成员变量的特点:
无论编译和运行,都参考左边(引用应变量所属的类)。
在多态中,静态成员函数的特点:
无论编译和运行,都参考左边。
示例:
class Fu
{
	int num = 5;
	void method1()
	{
		System.out.println("fu method_1");
	}
	void method2()
	{
		System.out.println("fu method_2");
	}
	static void method4()
	{
		System.out.println("fu method_4");
	}
}
class Zi extends Fu
{
	int num = 8;
	void method1()
	{
		System.out.println("zi method_1");
	}
	void method3()
	{
		System.out.println("zi method_3");
	}
	static void method4()
	{
		System.out.println("zi method_4");
	}
}
class DuoTaiDemo4
{
	public static void main(String[] args)
	{
	/*	Fu f = new Zi();//多态编译时期
		System.out.println(f.num);
		Zi z = new Zi();
		System.out.println(z.num);
		f.method1();
		f.method2();
		//f.method3();
	*/
		Fu f = new Zi();
		f.method4();

/*		Zi z = new Zi();
		z.method1();
		z.method2();
		z.method3();
*/
	}
}
练习:
练习:
/*
需求:
电脑运行实例,
电脑运行基于主板。
*/

interface PCI
{
	public void open();
	public void close();
	
}
class MainBoard
{
	public void run()
	{
		System.out.println("mainboard run");
	}
	public void usePCI(PCI p)//PCI p = new NetCard()//接口型引用指向自己的子//类对象
	{
		if(p!=null)
		{
			p.open();
			p.close();
		}
	}

}
//复写接口全部方法
class NetCard implements PCI
{
	public void open()
	{
		System.out.println("netcard open");
	}
	public void close()
	{
		System.out.println("netcard close");
	}
}
class SoundCard implements PCI
{
	public void open()
	{
		System.out.println("Soundcard open");
	}
	public void close()
	{
		System.out.println("Soundcard close");
	}
}


/*
class MainBoard
{
	public void run()
	{
		System.out.println("mainboard run");
	}
	public void usePCI(PCI p)
	{
		if(p!=null)
		{
		p.open();
		p.close();
		}
	}
}

class NetCard
{
	public void open()
	{
		System.out.println("netcard open");
	}
	public void close()
	{
		System.out.println("netcard close");
	}
}
*/
class DuoTaiDemo5
{
	public static void main(String[] args)
	{
		MainBoard mb = new MainBoard();
		mb.run();
		mb.usePCI(null);
		mb.usePCI(new NetCard());
		mb.usePCI(new SoundCard());
	}
}
二、Object
Object:是所有对象的直接或者间接父类,就是传说中的上帝。该类中定义的肯定是所有的对象都具备的功能。
Object类中已经提供了对对象是否相同的比较方法。如果自定义类中也有比较相同的功能,没有必要重新定义。
只要沿袭父类中的功能,建立自己特有的比较内容即可,这就是覆盖。
练习:
class Demo //extends Object
{
	private int num;
	Demo(int num)
	{
		this.num = num;
	}
	public boolean equals(Object obj)
	{
		if(!(obj instanceof Demo))
			return false;
		Demo d = (Demo)obj;//转型
		return this.num == d.num;
	}
	public String toString()
	{
		return "demo:"+num;
	}

/*	public boolean compare(Demo d)
	{
		return this.num==d.num;
	}
*/
}
class Person
{

}

class ObjectDemo
{
	public static void main(String[] args)
	{
		Demo d1 = new Demo(4);
		
		System.out.println(d1.toString());
		//Demo d2 = new Demo(6);
<span style="white-space:pre">		</span>//可以直接调用父类方法
		System.out.println(Integer.toHexString(d1.hashCode()));
		//Person p = new Person();

		//System.out.println(d1.equals(d2));
	}
}
三、内部类
将一个类定义在另一个类里面,对里面那个类就称为内部类(内置类,嵌套类)
1、内部类的访问规则:
a.内部类可以直接访问外部类中的成员,包括私有。
  之所以可以直接访问外部类中的成员,是因为内部类中持有了一个外部类的引用,
  格式 外部类名.this
b.外部类要访问内部类,必须建立内部类对象。
访问格式:
a.当内部类定义在外部类的成员位置上,而且非私有,可以在外部其他类中。可以直接建立内部类对象。
格式:
外部类名.内部类名 变量名 = 外部类对象.内部类对象
Outer.Inner in = new Outer().new Inner();
b.当内部类在成员位置上,就可以被成员修饰符所修饰。
  比如,private:将内部类在外部类中进行封装。
   static:内部类就具备了static的特性。只能直接访问外部类中的static成员,出现了访问局限。
c.在外部其他类中,如何直接访问static内部类的非静态成员呢?
new Outer.Inner().function();
注意:当内部类中定义了静态成员,该内部类必须是static的。
      当外部类中的静态方法访问内部类时,内部类也必须是static的
当描述事物时,事物的内部还有事物,该事物用内部类来描述。因为内部事务在使用外部事物的内容。
lass Outer
{
	private  static int x = 3;

	static class Inner//静态内部类
	{
		//int x = 4;
		static void function()
		{
			//int x = 6;
			System.out.println("inner:"+x);
				
		}
	}
	static class Inner2
	{
		void show()
		{
			System.out.println("inner2 show");
		}
	}
	 
	public static void method()
	{
		Inner.function();
		new Inner2().show();


		/*System.out.println(x);
		Inner in = new Inner();
		in.function();*/
	}
}
class InnerClassDemo
{
	public static void main(String[] args)
	{
		Outer.method();
		//Outer.Inner.function();
		//new Outer.Inner().function();
		//Outer out = new Outer();
		//out.method();

		//直接访问内部类中的成员。
		//Outer.Inner in = new Outer().new Inner();
		//in.function
	}
}

/*
内部类定义在局部时,
1.不可以被成员修饰符修饰。
2.可以直接访问外部类中的成员,因为还持有外部类中的引用。
  但是不可以访问他所在的局部中的变量,只能访问被final修饰的局部变量。  
*/
class Outer
{
	int x = 3;
	void method(final int a)
	{  
		final int y = 4;
		class Inner
		{
			void function()
			{
				System.out.println(y);
			}
		}
		new Inner().function();
	}
}


class InnerClassDemo3
{
	public static void main(String[] args)
	{
		new Outer().method(5);
	}
}
2、匿名内部类;
a.匿名内部类其实就是内部类的简写格式。
b.定义匿名内部类的前提:内部类必须是继承一个类或者实现接口。
c.匿名内部类的格式:new 父类或者接口(){定义子类的内容}
d.其实匿名内部类就是一个匿名子类对象。而且这个对象有点胖,可以理解为带内容的对象。
e.匿名内部类中定义的方法最好不要超过3个。
abstract class AbsDemo
{
	abstract void show();
}
class Outer
{
	int x =3;
	/*class Inner extends AbsDemo
	{
		void show()
		{
			System.out.println("show :"+x);
		}
	}*/
	public void function()
	{
		//new Inner().show();

	
		
		AbsDemo d = new AbsDemo()//多态,。,。,。
		{
			int num = 9;
			void show()
			{
				System.out.println("x="+num);
			}
			void abc()
			{
				System.out.println("haha ");
			}
		};
		d.show();
		//d.abc();//编译失败;
	
	}

}
class InnerClassDemo4
{
	public static void main(String[] args)
	{
		new Outer().function();
	}
}
四、异常
1、异常:就是程序中运行时出现不正常情况。
异常由来:问题也是现实生活中一个具体的事物,也可以通过java的类的形式进行描述,并封装成对象。
其实就是java对不正常情况进行描述后的对象体现。
对于问题的划分:两种:一种是严重的问题,一种非严重的问题。
a.对于严重的,java通过Error类进行描述。如:运行的类不存在或者内存溢出分等
对于Error一般不编写针对性的代码对其进行处理。
b.对于非严重的,java通过Exception类进行描述。
对于Exception可以使用针对性的处理方式进行处理。
无论Error或者Exception都具有一些共性内容。比如:不正常情况的信息,引发原因等。
Throwable
|--Error
|--Exception
2、异常的处理
java提供了特有的语句进行处理。
try
{
需要被检测的代码
}
catch(异常类 变量)
{
处理异常代码:(处理方式)
}
finally
{
一定会执行的语句;
}
3、对捕获到的异常对象进行常见方法操作。
a.String getMessage();获取异常信息
b.在函数上声明异常。便于提高安全性,让调用着进行处理,不处理编译失败。
4、对多异常的处理。
a.声明异常时,建议声明更为具体的异常,这样处理的可以更具体。
b.对方声明几个异常,就对应有几个catch块。不要定义多余的catch块,如果多个catch块中的异常出现继承 关系,父类异常catch块放在最下面。
建议在进行catch处理时,catch中一定要定义具体处理方式。不要简单定义一句e.printStackTrace();也不要 简单的诛邪一条输出语句。
class Demo
{
	int div(int a,int b)throws ArithmeticException,ArrayIndexOutOfBoundsException
<span style="white-space:pre">	</span>//在功能上通过throws的关键字声明了该功能有可能会出现问题
	{
		int[] arr = new int[a];
		System.out.println(arr[4]);
		return a/b;
	}
}
class ExceptionDemo1
{
	public static void main(String[] args)
	{
		Demo d = new Demo();
		try
		{
			int x = d.div(4,0);
			System.out.println("x="+x);
		}
		catch(Exception e)//Exception e= new ArithmeticException e()
		{
			System.out.println("haha+"+e.toString());

			e.printStackTrace();//异常名称、异常信息、异常出现的位置
//其实jvm默认的异常处理机制,就是调用e.printStackTrace方法。打印异常的堆栈的跟踪信息
		}

/*	//声明几个异常,就catch几个异常
		catch(ArithmeticException e)
		{
			System.out.println(e.toString());
			System.out.println("除零了");
		}
		catch(ArrayIndexOutOfBoundsException e)
		{
			System.out.println(e.toString());
			System.out.println("角标越界了");
		}
*/		
		System.out.println("over");
	}

}
5、自定义异常:
因为项目中会出现特有的问题。而这些问题并未被java所描述并封装对象。所以对于这些特有的问题可以按 照java的对问题封装的思想。将特有的问题,进行自定义的异常封装——自定义异常。
需求:在本程序中,对于除数是-1,也视为错误的是无法进行运算的。那么就需要对这个问题进行自定义的 描述。
当在函数内部出现了throw抛出异常对象,那么就必须要给对应的处理动作。要么在内部try catch 处理。要 么在函数上声明让调用者处理。
一般情况下,函数内出现异常,函数上需要声明。
发现打印的结果中只有异常的名称,却没有异常的信息。
因为自定义的异常并没有定义信息。
如何自定义异常信息呢?
因为父类中已经把异常信息的操作都完成了。所以子类只要在构造时,将异常信息传递给父类通过super语 句,那么就可以直接通过getMessage方法获取自定义的异常信息。
自定义异常:
必须是自定义类继承Exception。
继承Exception原因:
异常体系有一个特点:因为异常类和异常对象都被抛出。他们都具备可抛性,这个可抛性是Throwable这个 体系的独有特点。只有这个体系中的类和对象才可以被throws和throw操作。
throws和throw的区别:
throws使用在函数上。
throw只用在函数内。
throws后面跟的异常类,可以跟多个,用逗号隔开。
throw后跟的是异常对象。
class FuShuException extends Exception
{
	private int value;
	FuShuException(String msg,int value)
	{
		super(msg);
		this.value = value;
	}
	public int getValue()
	{
		return value;
	}


/*	private String msg;
	FuShuException(String msg)
	{
		this.msg = msg;
	}
	public String getMessage()
	{
		return msg;
	}
*/
}

class Demo
{
	int div(int a,int b)throws FuShuException
	{
		if(b<0)
		throw new FuShuException("出现了除数是负数的情况/by fushu",b);
//手动通过throw关键字抛出一个自定义异常现象

		return a/b;
	}
}
class ExceptionDemo2
{
	public static void main(String[] args)
	{
		Demo d = new Demo();
		try
		{
		int x = d.div(4,-1);
		System.out.println("x="+x);
		}
		catch(FuShuException e)
		{
			System.out.println(e.toString());
			//System.out.println("除数出现负数了");
			System.out.println("错误的负数是:"+e.getValue());
		}
	
		System.out.println("over");
	}

}
6、RuntimeException
Exception中有一个特殊的子类异常RuntimeException运行时异常。
如果在函数内容抛出该异常,函数上可以不用声明,编译一样通过。
如果在函数上声明了该异常,调用者可以不用进行处理,编译一样通过。
之所以不用在函数声明,是因为不需要让调用者处理。当该类异常发生,希望程序停止,因为在运行时, 出现了无法继续运算的情况,希望停止程序后,对代码进行修正。
自定义异常时:如果该异常的发生,无法在继续进行运算。就让自定义异常继承RuntimeException。
对于异常分两种:
a.编译时被检测的异常
b.编译时不被检测的异常(运行时异常,RuntimeException以及其子类)
class FuShuException extends RuntimeException
{
	FuShuException(String msg)
	{
		super(msg);
	}
}

class Demo
{
	int div(int a,int b)
	{
		if(b<0)
			throw new FuShuException("除数是负数");
		if(b==0)
			throw new ArithmeticException("被零除了");
		return a/b;
	}
}
class ExceptionDemo3
{
	public static void main(String[] args)
	{
		Demo d = new Demo();
		int x = d.div(4,-3);
		System.out.println("x="+x);
		
		System.out.println("over");
	}

}
finally代码块:定义一定执行的代码。
通常用于关闭资源。

*/

class FuShuException extends Exception
{
	FuShuException(String msg)
	{
		super(msg);
	}
}

class Demo
{
	int div(int a,int b)throws FuShuException
	{
		if(b<0)
			throw new FuShuException("除数是负数");
		
		return a/b;
	}
}
class ExceptionDemo4
{
	public static void main(String[] args)
	{
		Demo d = new Demo();
		try
		{
			int x = d.div(4,-1);
			System.out.println("x="+x);
		}
		catch(FuShuException e)
		{
			System.out.println(e.toString());
			return;
		}
		finally
		{
			System.out.println("finally");
	//finally里放的是一定会执行的代码。
		}
	
		System.out.println("over");
	}

}
/*
public void method()
{
	连接数据库;

	数据操作;//throw new SQLEException();数据库异常

	关闭数据库;//该动作,无论数据操作是否成功,一定要关闭资源。

	try
	{
		链接数据库;
		数据操作;//throw new SQLEException

	}
	catch
	{
		会对数据库进行异常处理;
		
	}
	finally
	{
		关闭数据库;
	}

}
7、异常在子父类覆盖中的体现
a.子类在覆盖父类时,如果父类的方法抛出异常,那么子类的覆盖方法,只能抛出父类的异常或者该异常的 子类.
b.如果父类方法抛出多个异常,那么子类在覆盖该方法时,只能抛出父类异常的子集。
c.如果父类或者接口的方法中没有异常抛出,那么子类在覆盖方法时,也不可以抛出异常。如果子类方法发 生了异常,就必须要进行try处理,绝对不能抛。
class AException extends Exception
{
	
}
class BException extends AException
{
	
}
class CException extends Exception
{
	
}
/*
Exception
	|--AException
		|--BException
	|--CException
*/

class Fu
{
	void show()throws AException
	{

	}
	
}
class Test
{
	void function(Fu f)
	{
		try
		{
			f.show();
		}
		catch(AException e)
		{
	
		}		
	}
}


class Zi extends Fu
{
	void show()throws BException
	{
		Test t= new Test;
		t.function(new Fu());
	}
}
练习:
/*
有一个圆形和长方形。

都可以获取面积,对于面积如果出现非法的数值,视为是获取面积出现的问题。
问题通过异常来表示。

先要对着个程序进行基本设计。

*/
class NoValueException extends RuntimeException
{
	NoValueException(String message)
	{
		super(message);
	}	
}
interface shape
{
	void getArea();
}

class Rec implements shape
{
	private int len,wid;
	Rec(int len,int wid)//throws NoValueException
	{
		if(len<=0 || wid <=0)
			throw new NoValueException("出现非法值");
		this.len = len;
		this.wid = wid;
	}
	public void getArea()
	{
		System.out.println(len*wid);
	}
}
`
class Circle implements shape
{
	private int radius;
	public static final double PI = 3.14;
	Circle(int radius)
	{
		if(radius<=0)
			throw new NoValueException("非法");
		this.radius = radius;
	}
	public void getArea()
	{
		System.out.println(radius*radius*PI);
	}
}

class ExceptionTest
{
	public static void main(String[] args)
	{
		
		Rec r = new Rec(3,4);
		Circle c = new Circle(-8);
		r.getArea();
		c.getArea();
		System.out.println("over");
		
	}
}
五、包(Package)
对类文件进行分类管理。
给类提供多层命名空间
写在程序文件的第一行。
类名的全称是:包名.类名
包也是一中封装形式。
包与包之间进行访问,被访问的包中的类以及类中的成员,需要public修饰。
不同包中的子类可以访问父类中被protected权限修饰的成员。
包与包之间可以使用的权限是有两种,public protected。
权限对比:
public protected default private
同一个类中 OK     OK   OK       OK
同一个包中 OK       OK     OK
子类 OK  OK
不同包中 OK




  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值