抽象类
* 当多个类中出现相同功能,但是功能主题不同,
* 这时可以进行向上抽取。这时抽取功能定义,而不抽取功能主题。
*
* 抽象:看不懂、笼统的
* 抽象类的特点:
* 1、抽象方法必须放在抽象类中
* 2、抽象方法和抽象类都必须被abstract关键字修饰
* 3、抽象类不可以用new创建对象。因为调用抽象方法没意义。
* 4、抽象类中的方法要被使用,必须由子类复写起所有的抽象方法后,建立子类对象调用。
* 如果子类只能覆盖了部分抽象方法,那么该子类是一个抽象类。
*
* 抽象类和一般类没有太大的不同。
* 该如何描述事物,就如何描述事物,只不过,该事物出现了一些看不懂的东西。
* 这些不确定的部分,也是该事物的功能,需要明确出现。但是无法定义主体。
* 通过抽象方法来表示。
*
* 抽象类比一般类多个了抽象函数。就是在类中可以定义抽象方法。
* 抽象类不可以实例化。
*
* 特殊:抽象类中可以不定义抽象方法,这样做仅仅是不让该类建立对象。在java类库中会有所体现
* 继承:
* 1、提高代码的复用性
* 2、让类与类之间产生了关系。有了这个关系,才有了多态的特性。
*
* 注意:
* 千万不要为了获取其他类的功能,简化代码而继承。
* 必须是类与类之间有所属关系才可以继承。所属关系is a。
*
* java语言中:java只支持单继承,不支持多继承[不是很严谨的说法](c++支持)。
* eg:b继承a,就不能继承b或其他类
*
* 因为多继承容易带来安全隐患:当多个父类中定义了相同内容,
* 当功能内容不同时,子类对象不确定要运行哪一个.
* 但是java保留这种机制。并用另一种形式来完成表示。多实现。
*
* java支持多层继承。也就是一个继承体系。
* c 继承b继承 a
* a是最共性的部分
*如何使用一个继承体系中的功能呢?
*想要使用体系,先查阅体系父类的描述,
*因为父类中定义的是该体系中的共性功能。
*那么这个体系已经可以使用了。
*那么在具体调用时,要创建最子类的对象,
*为什么呢?
*一是因为有可能父类不能创建对象(如抽象类),
*二是创建子类对象可以使用更多的共功能,包括基本的也包括特有的。
*
*简单一句话:查阅父类功能,创建子类对象使用功能。
*
*聚集:hase a(用的情况比继承多一些)
*聚合:球员和球队的关系。球队少了一个球员没事
*组合:事物的联系程度更紧密。手是人的一部分,人没手不行
*用的情况比继承多一些
/*
* 子父类出现后,类成员的特点:
*
* 类中成员:
* 1、变量。
* 2、函数。
* 3、构造函数
*
* 1、变量
* 如果子父类中出现非私有的同名成员变量时,
* 子类要访问本类中的变量,用this
* 子类要访问父类中的同名变量,用super。
*
* super的使用和this的使用几乎一致。
* this代表的是本类对象的引用。
* super代表的是父类对象的引用。
*/
class Father
{
// private int num=4;//不能被继承
int num=4;
}
class Sun extends Father
{
int num=5;
void show()
{
System.out.println(""+this.num);//没有定义num=5时结果为4,相当于子类自己定义的num值为4,
//定义num=5时值为5(因为覆盖,也就是重写)
System.out.println(super.num);//父类的num
}
}
class ExtendsDemo2
{
// public void main(String[] args) {
// Sun sun=new Sun();
// sun.show();
// System.out.print(sun.num+"...");
public static void main(String[] args) {
Sun sun=new Sun();
sun.show();
System.out.print(sun.num+"...");
}
}
* 2,当子类出现和父类一模一样的函数时,
* 当子类对象调用该函数,会运行子类函数的内容。
* 如同父类的函数被覆盖一样
*
* 这种情况是函数的另一个特性:重写(覆盖)
*
* 当子类继承父类,沿袭了父类的功能,到子类中,
* 但是子类虽具备该功能,但是功能的内容却和父类不一致,
* 这时,没有必要定义新功能,而是使用覆盖特性,保留父类的功能定义,并重写功能内容。
*
* 覆盖:
* 1、子类覆盖父类,必须保证子类权限大于等于父类权限,才可以覆盖,否则编译失败。
* 2、静态只能静态覆盖。
* eg:父类的方法定义成私有的,子类中方法与父类相同,不是被覆盖,因为父类的方法为私有,子类不知道有此方法
* 记住:
* 重载:只看同名函数的参数列表。
* 重写:子父类方法要一模一样。
* 子类不可以重载父类中的方法,因为在子类中重载后执行哪个方法不确定,编译错误
* 3,子父类中的构造函数。
* 在对子类对象进行初始化时,父类的构造函数也会运行,
* 那是因为子类的构造函数默认第一行有一条隐式的语句super();
* super():会访问父类中空参数的构造函数。而且子类中所有的构造函数默认第一行都是super();
*
* 为什么子类一定要访问父类中的构造函数
* 因为父类中的数据子类可以直接获取,所以子类对象在建立时需要查父类是如何对这些数据初始化的,
* 所以子类在对象初始化时,要先访问一下父类中的构造函数。
* 如果要访问父类中指定的构造函数,可以通过手动定义super语句的方式来指定
*
* 注意:super语句一定定义在子类构造函数的第一行
*
* 子类的实例化过程
* 结论:
* 子类的所有的构造函数,默认都会访问父类中空参数的构造函数。
* 因为子类中每一个构造函数内的第一行都有一句隐式super();
* 当父类中没有空参数的构造函数时,子类必须手动通过super或者this语句形式来指定要访问的构造函数。
* 不能同时放在一行
* * this() super() 都必须放在第一行,都是初始化动作
class Father4
{
int num;
Father4()
{
num=60;
System.out.println("father");
}
Father4(int a)
{
System.out.println("fathera");
}
}
class Sun4 extends Father4
{
Sun4()
{
//super();//默认加的隐式语句
// super(4);//指定方式;如果没有写出Father4(),不写super(4)这句则出错,因为默认的是super()
System.out.println("sun");
}
Sun4(int x)
{
this();//访问 Sun4()
//super();//默认加的隐式语句
System.out.println("sun"+x);
}
}
public class ExtendsDemo4 {
public static void main(String[] args) {
Sun4 sun=new Sun4();
System.out.println(sun.num);
Sun4 sun1=new Sun4(5);
}
}
* final:最终。作为一个修饰符,
* 1,可以修饰类,函数,变量。
* 2,被final修饰的类不可以被继承.为了避免被继承,被子类复写功能。
* 3,被final修饰的方法不可以被复写。
* 4,被final修饰的变量是一个常量只能赋值一次,既可以修饰成员变量,又可以修饰局部变量。
* 当在描述事物时,一些数据的出现值是固定的,那么这时为了增强阅读性,都给这些值起个名字。方便阅读。
* 而这个值不需要改变,所以加上final修饰。作为常量:常量的书写规范:所有字符都大写,如果由多个单词组成。
* 单词间通过_连接
* 5,内部类定义在类中的局部位置上时,只能访问该局部被被final修饰的局部变量。
类修饰符
public
final