编程语言中的继承就好比日常生活中的遗传和模仿。通过遗传,儿子获得了父亲的某些基因,对外表现出某种特性,通俗的说法就是我们常听到人们说“张三的眼睛......长得和他父亲很像”之类的话;通过模仿,儿子的一些行为、动作、处事方式会和父亲很像。我们在编程过程中所接触的继承实质与这很相似,这里的“某种特性”就相当于编程中的“属性”,这里的“一些行为、动作、处事方式”就相当于编程中的“成员函数”。照此说来,继承就是子类继承了父类的属性和方法。但在编程语言中继承还是有别于生活的,遗传和模仿所获得的仅仅只是一部分,而继承则是继承了父类所有的属性和方法。话虽如此,但是当子类不能调用父类的私有属性,在实例化子类对象的时候,会开辟一定的存储空间用来存储从父类继承的私有属性,但这些属性是无法被调用的,所以可以理解为子类不能继承父类的私有属性。
当子类中重写了父类的方法时,实例化一个子类对象,调用的是子类重写后的方法。
//父类 public class People { private String name; public void Action(){ System.out.println(name+"的行动"); } public String getName() { return name; } public void setName(String name) { this.name = name; } } //子类 public class Child extends People{ private String name; private int age; public void Action(){ System.out.println(name+"的行动"); } } //测试类 public class Test { public static void main(String[] args) { People people=new People(); people.setName("Mike"); people.Action(); Child child=new Child(); child.setName("Bob"); child.Action();
结果:
发生继承时,子类继承的其实严格来说应该是父辈,而不是直系父亲,每个子类都有且只有一个父类,就像生活中每个人都只有一个亲生父亲一样。当子类发生继承时,被继承的类不一定是该子类的父亲,只能说是父辈,父辈只是用来显示告诉子类其父亲有哪些方法和属性,当子类没有发生重写,即没有自己独有的属性和方法时,它可以利用父辈的方法和属性,这时改变的是父辈的方法和属性,并且影响自己的属性和方法(可以理解为子类和父类共有一套方法和属性),而当子类重写了父类的方法,或者定义了自己的属性时,利用父辈的方法,改变的是自己的属性和方法。
特别值得注意的一点是:因为子类构造器都会默认调用父类的无参构造器,所以每一个子类对象的产生 必然伴随着一个父类对象的诞生。每一个类都会有一个默认的构造器,子类的构造器其实是这样的:
public 子类类名(){
super();
}
如果父类中重载了有参的构造器,则在创建子类对象时会报错,如下:
public class People { private String name; private int IDnum; private int age; //构造器 public People(String name){ this.name=name; } } public class Child extends People{ private String name; private int age; public void Action(){ System.out.println(name+"的行动"); } }
则会报错如下:
意思是父类没有定义无参构造器
解决办法有两种:
1.在父类中把默认的无参构造器显示化的写出来,即
public People(){
}
2.在子类中把无参构造器重写一下,在该构造器内调用父类有参的构造器,即
public Child(){
super(“张三”);
}
自动转型:
可以将子类自动转成父类,即将子类看作是父类,但是我们不能将父类看作是子类。
格式:
父类类名 对象名=new 子类类名();
形象点来说,在一堆小轿车中随机找出一辆,它既是小轿车,同时又是车。在生活中,我们可以把小轿车看作是车,但车却不一定都是小轿车,还会有其他类型的车,例如:客车、货车......
public class People { private String name; public void play(){ System.out.println(name+"的行动"); } public void Action(){ System.out.println(name+"正在行动"); } public String getName() { return name; } public void setName(String name) { this.name = name; } } public class Adult extends People{ private String name; public void play(){ System.out.println(name+"的行动"); } public void rest(){ System.out.println(name+"在休息"); } public String getName() { return name; } public void setName(String name) { this.name = name; } } public class Test { public static void main(String[] args) { People people=new People(); people.setName("李四"); People adult=new Adult(); adult.setName("张三"); } }
当在Test类的主函数中执行 adult.play(); [子类重写了父类的方法] 时,结果如下:
张三的行动
当执行:adult.Action(); [父类特有的方法] 时,结果如下:
null在行动
当测试类是这样写时:
public class Test { public static void main(String[] args) { People people=new People(); people.setName("李四"); People adult=new Adult(); adult.Action(); adult.play(); } }
结果如下:
null正在行动
null的行动
当执行:adult.rest();时,会出现下面的报错
错误显示的是rest()方法在父类中没有定义,由此可知自动转型产生的对象不能调用子类中所特有的方法。
上面这些测试,结果表明:自动转型产生的对象不能够调用子类所特有的方法;当子类重新定义了自己的属性并重写父类的方法时,自动转型产生的对象调用的是子类所重写的方法。
其实还有一种比较常用的自动转型,比较隐蔽,容易忽视
在创建界面时,向窗体上添加各种组件的add(comb)方法原代码为
public Component add(Component comp) {
addImpl(comp, null, -1);
return comp;
}
它的形参是Component类型的,但是add方法它却可以添加JButton、JCheckBox、JComboBox等类型的对象,其实JButton、JCheckBox、JComboBox都是Component的子类,这里就是把子类自动转型成父类了。
此外,在写数组队列的添加对象功能时,形参是Object类型的,这里也应用到了自动转型,Object是所有类的父类。
相关推荐
- 创建一个新的父节点,其概率为这两个节点的概率之和,并将它们作为子节点。 - 将新节点放回队列。 - **生成编码规则:** - 从根节点出发,遍历到每个叶子节点,左分支标记为“0”,右分支标记为“1”。 - 叶子...
5. **交叉(Crossover)**:交叉操作模拟生物的基因重组,将两个父代个体的部分特征组合成新的子代。 6. **变异(Mutation)**:变异操作是为了保持种群的多样性,随机改变个体的部分特征,防止算法陷入局部最优。 ...
在课程设计中,可能需要实现这些非递归遍历算法,并进行性能分析,比较递归与非递归方法的优劣。递归方法代码简洁,但会增加调用栈的压力;非递归方法避免了栈溢出的风险,但可能需要更多的辅助数据结构。 “正文....
- **泛化关系**:类似于面向对象编程中的继承,子用例继承父用例的所有属性和行为,同时可添加自己的特性和行为。 #### 实践建议 在撰写论文时,合理运用UML图能够显著提升论文的清晰度和专业性。具体操作时,应...
具体来说,当TreeView控件与数据库相结合时,开发者首先需要设计数据库表结构,用于存储树节点的数据,如节点名称、父节点ID等。之后,使用***(***)来访问数据库,并将查询得到的数据加载到TreeView控件中。 在...
标题中的“ijcv04_keypoints.rar_matlab之父_sift”揭示了这个压缩包文件与计算机视觉领域的一个经典算法——尺度不变特征变换(Scale-Invariant Feature Transform, SIFT)有关,同时也暗示了该资源可能由MATLAB...
7. 父结点、祖先、子结点、兄弟、后代和子树:这些是描述树中结点之间关系的概念。 8. 特殊树形结构:如链(所有结点的度数不超过2的树)、菊花(存在某个结点使得其余所有结点与之相连)等。 在算法竞赛中,树的...
### 信息论实验知识点 #### 实验背景与目标 本次实验是信息论课程中的一个重要环节,旨在通过实际...通过本实验的学习,不仅可以深入理解信息论中的编码理论,还能提升编程能力,为今后的研究和工作打下坚实的基础。
1. **骨骼层次结构**:XML文件会定义骨骼的层级关系,即父骨骼与子骨骼的关联,形成树状结构。 2. **关节旋转和平移**:通过XML记录每个关节的旋转角度和平移距离,以控制模型运动。 3. **权重分配**:XML可以描述...
4. **交叉操作**:也叫配对或杂交,两个父代个体通过交换部分基因片段生成新的子代,增加了种群多样性。 5. **变异操作**:随机改变个体的部分基因,引入新的变异,防止过早收敛到局部最优。 6. **终止条件**:当...
两个父代染色体的部分基因片段被交换,生成新的子代染色体。常见的交叉策略有单点交叉、多点交叉和均匀交叉等。 6. **变异操作**:变异是为了保持种群的多样性,防止早熟现象。它随机改变染色体上的某些基因,为...
同步与异步是并发编程中的两个关键概念。同步是指多个进程或线程必须按照一定的顺序执行,一个进程必须等待另一个进程完成其操作后才能继续。而在异步环境中,进程无需等待其他进程,它们可以并行运行,通过回调、...
- **泛化关系**:当多个用例具有相似的结构和行为时,可以将这些相似性抽象为一个父用例,其他的用例作为子用例继承父用例的特征。这种关系类似于面向对象编程中的继承关系。 - **聚集关系**:聚集关系并不直接适用...
fork返回新进程的标识符,使得父进程和子进程可以独立执行。子进程随后调用execve系统调用来替换其内存空间的内容,加载hello程序的可执行文件,从而开始执行。 6. **存储管理** 在进程运行过程中,存储管理涉及到...