`
蓝色幻想
  • 浏览: 5981 次
  • 性别: Icon_minigender_2
最近访客 更多访客>>
社区版块
存档分类
最新评论

论编程中的父与子

    博客分类:
  • java
阅读更多



        编程语言中的继承就好比日常生活中的遗传和模仿。通过遗传,儿子获得了父亲的某些基因,对外表现出某种特性,通俗的说法就是我们常听到人们说“张三的眼睛......长得和他父亲很像”之类的话;通过模仿,儿子的一些行为、动作、处事方式会和父亲很像。我们在编程过程中所接触的继承实质与这很相似,这里的“某种特性”就相当于编程中的“属性”,这里的“一些行为、动作、处事方式”就相当于编程中的“成员函数”。照此说来,继承就是子类继承了父类的属性和方法。但在编程语言中继承还是有别于生活的,遗传和模仿所获得的仅仅只是一部分,而继承则是继承了父类所有的属性和方法。话虽如此,但是当子类不能调用父类的私有属性,在实例化子类对象的时候,会开辟一定的存储空间用来存储从父类继承的私有属性,但这些属性是无法被调用的,所以可以理解为子类不能继承父类的私有属性。

       当子类中重写了父类的方法时,实例化一个子类对象,调用的是子类重写后的方法。

//父类
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是所有类的父类。

      

 

 

 

  • 大小: 1.1 KB
  • 大小: 3.2 KB
  • 大小: 4.9 KB
分享到:
评论
1 楼 liu5261586 2014-08-22  
写的挺好的。。 自己的理解往往是最棒的

相关推荐

    信息论与编码报告

    - 创建一个新的父节点,其概率为这两个节点的概率之和,并将它们作为子节点。 - 将新节点放回队列。 - **生成编码规则:** - 从根节点出发,遍历到每个叶子节点,左分支标记为“0”,右分支标记为“1”。 - 叶子...

    遗传算法论文(很多篇)

    5. **交叉(Crossover)**:交叉操作模拟生物的基因重组,将两个父代个体的部分特征组合成新的子代。 6. **变异(Mutation)**:变异操作是为了保持种群的多样性,随机改变个体的部分特征,防止算法陷入局部最优。 ...

    数据结构课程设计二叉树非递归遍历遍历论文

    在课程设计中,可能需要实现这些非递归遍历算法,并进行性能分析,比较递归与非递归方法的优劣。递归方法代码简洁,但会增加调用栈的压力;非递归方法避免了栈溢出的风险,但可能需要更多的辅助数据结构。 “正文....

    论文UML图例

    - **泛化关系**:类似于面向对象编程中的继承,子用例继承父用例的所有属性和行为,同时可添加自己的特性和行为。 #### 实践建议 在撰写论文时,合理运用UML图能够显著提升论文的清晰度和专业性。具体操作时,应...

    论文研究-树型图及其在ASP.NET程序开发中的应用.pdf

    具体来说,当TreeView控件与数据库相结合时,开发者首先需要设计数据库表结构,用于存储树节点的数据,如节点名称、父节点ID等。之后,使用***(***)来访问数据库,并将查询得到的数据加载到TreeView控件中。 在...

    ijcv04_keypoints.rar_matlab之父_sift

    标题中的“ijcv04_keypoints.rar_matlab之父_sift”揭示了这个压缩包文件与计算机视觉领域的一个经典算法——尺度不变特征变换(Scale-Invariant Feature Transform, SIFT)有关,同时也暗示了该资源可能由MATLAB...

    最全面的算法竞赛的树论知识

    7. 父结点、祖先、子结点、兄弟、后代和子树:这些是描述树中结点之间关系的概念。 8. 特殊树形结构:如链(所有结点的度数不超过2的树)、菊花(存在某个结点使得其余所有结点与之相连)等。 在算法竞赛中,树的...

    信息论实验

    ### 信息论实验知识点 #### 实验背景与目标 本次实验是信息论课程中的一个重要环节,旨在通过实际...通过本实验的学习,不仅可以深入理解信息论中的编码理论,还能提升编程能力,为今后的研究和工作打下坚实的基础。

    骨骼动画 虚拟人 XML 论文

    1. **骨骼层次结构**:XML文件会定义骨骼的层级关系,即父骨骼与子骨骼的关联,形成树状结构。 2. **关节旋转和平移**:通过XML记录每个关节的旋转角度和平移距离,以控制模型运动。 3. **权重分配**:XML可以描述...

    美赛各题型常见参考代码:多种遗传算法优化论文与代码.zip

    4. **交叉操作**:也叫配对或杂交,两个父代个体通过交换部分基因片段生成新的子代,增加了种群多样性。 5. **变异操作**:随机改变个体的部分基因,引入新的变异,防止过早收敛到局部最优。 6. **终止条件**:当...

    遗传算法 论文和程序实现

    两个父代染色体的部分基因片段被交换,生成新的子代染色体。常见的交叉策略有单点交叉、多点交叉和均匀交叉等。 6. **变异操作**:变异是为了保持种群的多样性,防止早熟现象。它随机改变染色体上的某些基因,为...

    进程同步模拟设计--司机和售票员问题

    同步与异步是并发编程中的两个关键概念。同步是指多个进程或线程必须按照一定的顺序执行,一个进程必须等待另一个进程完成其操作后才能继续。而在异步环境中,进程无需等待其他进程,它们可以并行运行,通过回调、...

    2014年上半年—系统分析师—综合+案例+论文(答案+解析)_v1.0

    - **泛化关系**:当多个用例具有相似的结构和行为时,可以将这些相似性抽象为一个父用例,其他的用例作为子用例继承父用例的特征。这种关系类似于面向对象编程中的继承关系。 - **聚集关系**:聚集关系并不直接适用...

    ICS大作业论文附件.doc

    fork返回新进程的标识符,使得父进程和子进程可以独立执行。子进程随后调用execve系统调用来替换其内存空间的内容,加载hello程序的可执行文件,从而开始执行。 6. **存储管理** 在进程运行过程中,存储管理涉及到...

Global site tag (gtag.js) - Google Analytics