精华内容
下载资源
问答
  • 子类对象指向父类引用,先执行父类,又有如下的优先级关系, 静态代码块>普通代码块>构造方法 所以先执行父类静态代码快,子类静态代码块,父类构造函数,子类构造函数。 ab = new B(); 又新建对象,重新执行构造...
    public static void main(String[] args) {
        A ab = new B();
          ab = new B();
        }
    }
    class A{
        static{
            System.out.print("1");
        }
        public A(){
            System.out.print("2");
        }
    }
    class B extends A{
        static{
            System.out.print("a");
        }
        public B(){
            System.out.print("b");
        }
    

    输出结果是

    1a2b2b
    

    先分析这句:
    A ab = new B();
    子类对象指向父类引用,先执行父类,又有如下的优先级关系,
    静态代码块>普通代码块>构造方法
    所以先执行父类静态代码快,子类静态代码块,父类构造函数,子类构造函数。
    ab = new B();
    又新建对象,重新执行构造方法 父类>子类
    static代码块只执行一次原因:
    static代码块只在类加载时执行,类是用类加载器来读取的,类加载器是带有一个缓存区的,
    它会把读取到的类缓存起来,所以在一次虚拟机运行期间,一个类只会被加载一次,这样的话静态代码块只会运行一次。

    展开全文
  • 永远记住:不管父类子类之间如何转换,永远是父类引用指向子类对象,子类引用指向父类对象,需要将父类对象强制转换为子类! 举例: public class Animal{} //父类 class Dog extends Animal{} //子类 public class...

    永远记住:不管父类子类之间如何转换,永远是父类引用指向子类对象,子类引用指向父类对象,需要将父类对象强制转换为子类!

    举例:

    public class Animal{}	//父类
    class Dog extends Animal{}	//子类
    
    public class Test{
    	public static void main(String[] args){
    		Animal a1=new Animal();	//父类引用指向父类对象,√
    		Animal a2=new Dog();	//父类引用指向子类对象,这是多态,√
    		Dog d1=new Dog();		//子类引用指向子类对象,√
    		Dog d2=new Animal();	//子类引用指向父类对象,×
    		a1=d1;					//父类引用指向子类对象,√
    		d1=a1;					//子类引用指向父类对象,×
    		d1=(Dog)a1;				//父类对象a1强制转换为子类对象,然后子类引用指向子类对象,√
    	}
    }
    

    朴实无华的分割框,下面是知识点框架梳理,上面时精辟要点!

    在这里插入图片描述

    展开全文
  • 父类中的private变量和方法,父类本身的对象是无法直接调用的,自然指向子类对象父类引用也是无法访问父类中的private变量和方法。注意点2 子类扩展父类的变量和方法,该引用是无法调用的。public class Point { ...

    注意点1
    父类中的private变量和方法,父类本身的对象是无法直接调用的,自然指向子类对象的父类引用也是无法访问父类中的private变量和方法。

    注意点2
    子类扩展父类的变量和方法,该引用是无法调用的。

    public class Point {
        private int x;
        private final int y;
    
        public Point(int x,int y) {
            this.x= x;
            this.y= y;
        }
    }

    Point是父类,在实现一个它的子类。

    public class ColorPoint extends Point{
        public int color;
    
        public ColorPoint(int x, int y,int color) {
            super(x, y);
            // TODO Auto-generated constructor stub
            this.color = color;
        }
    
        public int getColor() {
            return color;
        }   
    
    }

    ColorPoint中的color和getColor()方法是扩展父类Point的。于是,其中

    Point cp = new ColorPoint(3,4,5);

    cp是无法访问getColor()和color的。

    注意点3
    指向子类对象的父类引用只会调用在子类的方法。如果子类覆盖了父类的方法,该引用只会取得被覆盖了的方法。对于子类继承过来并没有覆盖的方法,则已经是子类所拥有,该引用直接调用。

    public class Point {
        public int x;
        public int y;
    
        public Point(int x,int y) {
            this.x= x;
            this.y= y;
        }
    
        public void method() {
            System.out.println("父类");
        }
    
        public void method1() {
            System.out.println("父类方法没有被覆盖");
        }
    
    }

    父类method()方法将被子类覆盖。

    public class ColorPoint extends Point{
        public int color;
        public int x;
    
        public ColorPoint(int x, int y,int color) {
            super(x, y);
            // TODO Auto-generated constructor stub
            this.color = color;
        }
    
    
        @Override
        public void method() {
            System.out.println("子类覆盖父类方法");
        }
    
    }

    最后进行打印输出

    Point cp1 = new ColorPoint(4,8,9);
    cp1.method();
    cp1.method1();
    
    //输出结果
    //子类覆盖父类方法class com.kdk.java.ColorPoint
    //父类方法没有被覆盖class com.kdk.java.ColorPoint

    注意点4
    在向上转型中,如果子类具有和父类定义一样的变量时,该变量时无法被继承的。要获得该定义相同的变量在子类对象中的值,则需通过get()方法,而通过.的方式取得是该定义相同的变量在父类中的值。

    public class Point {
        public int x;
        public int y;
    
        public Point(int x,int y) {
            this.x= x;
            this.y= y;
        }
    
        public int getY() {
            return y;
        }
    }
    
    public class ColorPoint extends Point{
        public int color;
        public int x = 8;
        public int y = 9;
    
        public ColorPoint(int x, int y,int color) {
            super(x, y);
            // TODO Auto-generated constructor stub
            this.color = color;
        }
    
        public int getY() {
            return y;
        }
    }
    
    
    public class Main {
        public static void main(String[] args) { 
            Point cp = new ColorPoint(3,4,5);
    
            System.out.println(cp.x+"");//输出3
            System.out.println(cp.y);//输出4
            System.out.println(cp.getY()+"");//输出9
        }
    }
    
    

    未完待续….

    展开全文
  • 我们都知道,面向对象程序设计中的类有三大特性:继承,封装,多态,这个也是介绍类的时候,必须提到的话题,那么今天就来看一下OC中类的三大特性: 一、封装 封装就是对类中的一些字段,方法进行保护,不被外界...

     

    我们都知道,面向对象程序设计中的类有三大特性:继承,封装,多态,这个也是介绍类的时候,必须提到的话题,那么今天就来看一下OC中类的三大特性:

    一、封装

    封装就是对类中的一些字段,方法进行保护,不被外界所访问到,有一种权限的控制功能,Java中有四种访问权限修饰符:

    1
    public,default,protected,private

    访问权限依次递减,这样我们在定义类的时候,哪些字段和方法不想暴露出去,哪些字段和方法可以暴露,可以通过修饰符来完成,这就是封装,下面来看一个例子吧:

    Car.h

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    //  Car.h  
    //  05_ObjectDemo  
    //  
    //  Created by jiangwei on 14-10-11.  
    //  Copyright (c) 2014年 jiangwei. All rights reserved.  
    //  
       
    #import   
       
    @interface Car : NSObject{  
        //这个属性就是对外进行保密的相当于private,所以我们需要在外部访问的话,必须定义get/set方法  
        //默认的是private的,但是我们可以使用@public设置为public属性的,那么在外部可以直接访问:person->capcity = 2.8;  
        //当然我们一般不这么使用,因为这会破坏封装性,这种用法相当于C中的结构体中权限  
        //一共四种:@public,@protected,@private,@package,这个和Java中是相同的  
    @public  
        float _capcity; //油量属性  
    }  
       
    - (void)run:(float)t;  
       
    @end

    这里我们可以看到,OC中也是有四种访问权限修饰符:

    1
    @public、@protected、@private、@package

    其中默认的修饰符是@private

    但是这里要注意的是:OC中的方法是没有修饰符的概念的,这个和Java有很大的区别,一般都是公开访问的,即public的,但是我们怎么做到让OC中的一个方法不能被外界访问呢?

    OC中是这么做的,如果想让一个方法不被外界访问的话,只需要在.m文件中实现这个方法,不要在头文件中进行定义,说白了就是:该方法有实现,没定义,这样外界在导入头文件的时候,是没有这个方法的,但是这个方法我们可以在自己的.m文件中进行使用。

    为什么要介绍这点知识呢?因为在后面我们会说到单利模式,到时候就会用到这个知识点了。

    二、继承

    继承是类中的一个重要的特性,他的出现使得我们没必要别写重复的代码,可重用性很高。当然OC中的继承和Java中是一样的,没多大区别,这里在看一个例子吧:

    首先来看一下父类:Car

    Car.h

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    //  
    //  Car.h  
    //  06_ExtendDemo  
    //  
    //  Created by jiangwei on 14-10-11.  
    //  Copyright (c) 2014年 jiangwei. All rights reserved.  
    //  
       
    #import   
       
    @interface Car : NSObject{  
        NSString *_brand;  
        NSString *_color;  
    }  
       
    - (void)setBrand:(NSString *)brand;  
    - (void)setColor:(NSString *)color;  
    - (void)brake;  
    - (void)quicken;  
       
    @end

    在Car类中定义了两个属性,以及一些方法

    Car.m

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    //  
    //  Car.m  
    //  06_ExtendDemo  
    //  
    //  Created by jiangwei on 14-10-11.  
    //  Copyright (c) 2014年 jiangwei. All rights reserved.  
    //  
       
    #import "Car.h"  
       
    @implementation Car  
    - (void)setBrand:(NSString *)brand{  
        _brand = brand;  
    }  
    - (void)setColor:(NSString *)color{  
        _color = color;  
    }  
    - (void)brake{  
        NSLog(@"刹车");  
    }  
    - (void)quicken{  
        NSLog(@"加速");  
    }  
    @end

    方法的实现

    在来看一下子类:

    Taxi.h

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    //  
    //  Taxi.h  
    //  06_ExtendDemo  
    //  
    //  Created by jiangwei on 14-10-11.  
    //  Copyright (c) 2014年 jiangwei. All rights reserved.  
    //  
       
    #import "Car.h"  
       
    @interface Taxi : Car{  
        NSString *_company;//所属公司  
    }  
       
    //打印发票  
    - (void)printTick;  
       
    @end

    看到Taxi类继承了父类Car,这里需要导入父类的头文件,然后在Taxi类中多了一个属性和方法

    Taxi.m

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    //  
    //  Taxi.m  
    //  06_ExtendDemo  
    //  
    //  Created by jiangwei on 14-10-11.  
    //  Copyright (c) 2014年 jiangwei. All rights reserved.  
    //  
       
    #import "Taxi.h"  
       
    @implementation Taxi  
       
    - (void)printTick{  
        [super brake];  
        [self brake];  
        NSLog(@"%@出租车打印了发票,公司为:%@,颜色为:%@",_brand,_company,_color);  
    }  
       
    @end

     

    对方法的实现,这里我们看到实现文件中是不需要导入父类Car的头文件的,因为可以认为,Taxi.h头文件中已经包含了Car的头文件了。而且,这里可以使用super关键字来调用父类的方法,同时这里我们也是可以用self关键字来调用,这里看到其实这两种方式调用的效果是一样的,当我们在子类重新实现brake方法的时候(Java中的重写概念),那么这时候super关键字调用的还是父类的方法,而self调用的就是重写之后的brake方法了。同样,我们也是可以使用父类中的属性。

    再看一下另外一个子类:

    Truck.h

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    //  
    //  Truck.h  
    //  06_ExtendDemo  
    //  
    //  Created by jiangwei on 14-10-11.  
    //  Copyright (c) 2014年 jiangwei. All rights reserved.  
    //  
       
    #import "Car.h"  
    //卡车类继承Car  
    @interface Truck : Car{  
        float _maxWeight;//最大载货量  
    }  
       
    //覆盖父类的方法brake  
    //优先调用子类的方法  
    - (void)brake;  
       
    - (void)unload;  
       
    @end

    这里就自己定义了一个brake方法,这时候就会覆盖父类中的brake方法了。

    Truck.m

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    //  
    //  Truck.m  
    //  06_ExtendDemo  
    //  
    //  Created by jiangwei on 14-10-11.  
    //  Copyright (c) 2014年 jiangwei. All rights reserved.  
    //  
       
    #import "Truck.h"  
       
    @implementation Truck  
       
    - (void)brake{  
        [super brake];  
        NSLog(@"Truck类中的brake方法");  
    }  
       
    - (void)unload{  
        [super brake];//调用父类的方法  
        [self brake];//也是可以的  
        NSLog(@"%@的卡车卸货了,载货量:%.2f,汽车的颜色:%@",_brand,_maxWeight,_color);  
    }  
       
    @end

    这里就可以看到,我们会在brake方法中调用一下父类的brake方法,然后在实现我们自己的逻辑代码。

    好了,继承就说这么多了,其实封装和继承两个特性没什么难度的,很容易理解的,下面在来看一下最后一个特性:多态

    三、多态

    多态对于面向对象思想来说,个人感觉是真的很重要,他对以后的编写代码的优雅方式也是起到很重要的作用,其实现在很多设计模式中大部分都是用到了多态的特性,Java中的多态特性用起来很是方便的,但是C++中就很难用了,其实多态说白了就是:定义类型和实际类型,一般是基于接口的形式实现的,不多说了,直接看例子吧:

    打印机的例子

    抽象的打印机类Printer

    Printer.h

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    //  
    //  Printer.h  
    //  07_DynamicDemo  
    //  
    //  Created by jiangwei on 14-10-11.  
    //  Copyright (c) 2014年 jiangwei. All rights reserved.  
    //  
       
    #import   
       
    @interface Printer : NSObject  
       
    - (void) print;  
       
    @end

    就是一个简单的方法print

    Printer.m

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    //  
    //  Printer.m  
    //  07_DynamicDemo  
    //  
    //  Created by jiangwei on 14-10-11.  
    //  Copyright (c) 2014年 jiangwei. All rights reserved.  
    //  
       
    #import "Printer.h"  
       
    @implementation Printer  
       
    - (void)print{  
        NSLog(@"打印机打印纸张");  
    }  
       
    @end

    实现也是很简单的

    下面来看一下具体的子类

    ColorPrinter.h

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    //  
    //  ColorPrinter.h  
    //  07_DynamicDemo  
    //  
    //  Created by jiangwei on 14-10-11.  
    //  Copyright (c) 2014年 jiangwei. All rights reserved.  
    //  
       
    #import "Printer.h"  
       
    //修改父类的打印行为  
    @interface ColorPrinter : Printer  
    - (void)print;  
    @end

    ColorPrinter.m

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    //  
    //  ColorPrinter.m  
    //  07_DynamicDemo  
    //  
    //  Created by jiangwei on 14-10-11.  
    //  Copyright (c) 2014年 jiangwei. All rights reserved.  
    //  
       
    #import "ColorPrinter.h"  
       
    @implementation ColorPrinter  
       
    - (void)print{  
        NSLog(@"彩色打印机");  
    }  
       
    @end

    在看一下另外一个子类

    BlackPrinter.h

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    BlackPrinter.m
    //  
    //  BlackPrinter.m  
    //  07_DynamicDemo  
    //  
    //  Created by jiangwei on 14-10-11.  
    //  Copyright (c) 2014年 jiangwei. All rights reserved.  
    //  
       
    #import "BlackPrinter.h"  
       
    @implementation BlackPrinter  
       
    - (void)print{  
        NSLog(@"黑白打印机");  
    }  
       
    @end

    这里我们在定义一个Person类,用来操作具体的打印机

    Person.h 

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    Person.m
    //  
    //  Person.m  
    //  07_DynamicDemo  
    //  
    //  Created by jiangwei on 14-10-11.  
    //  Copyright (c) 2014年 jiangwei. All rights reserved.  
    //  
       
    #import "Person.h"  
       
    @implementation Person  
       
    /* 
    - (void) printWithColor:(ColorPrinter *)colorPrint{ 
        [colorPrint print]; 
      
    - (void) printWithBlack:(BlackPrinter *)blackPrint{ 
        [blackPrint print]; 
     */  
       
    - (void) doPrint:(Printer *)printer{  
        [printer print];  
    }  
       
    @end

    再来看一下测试代码:

    main.m

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    //  
    //  main.m  
    //  07_DynamicDemo  
    //  
    //  Created by jiangwei on 14-10-11.  
    //  Copyright (c) 2014年 jiangwei. All rights reserved.  
    //  
       
    #import   
       
    #import "Person.h"  
    #import "BlackPrinter.h"  
    #import "ColorPrinter.h"  
       
    int main(int argc, const charchar * argv[]) {  
        @autoreleasepool {  
               
            Person *person =[[Person alloc] init];  
               
            ColorPrinter *colorPrint = [[ColorPrinter alloc] init];  
            BlackPrinter *blackPrint = [[BlackPrinter alloc] init];  
               
            //多态的定义  
            /* 
            Printer *p1 = [[ColorPrinter alloc] init]; 
            Printer *p2 = [[BlackPrinter alloc] init]; 
              
            [person doPrint:p1]; 
            [person doPrint:p2]; 
             */  
               
            //通过控制台输入的命令来控制使用哪个打印机  
            int cmd;  
            do{  
                scanf("%d",&cmd);  
                if(cmd == 1){  
                    [person doPrint:colorPrint];  
                }else if(cmd == 2){  
                    [person doPrint:blackPrint];  
                }  
            }while (1);  
               
        }  
        return 0;  
    }

    下面就来详细讲解一下多态的好处

    上面的例子是一个彩色打印机和黑白打印机这两种打印机,然后Person类中有一个操作打印的方法,当然这个方法是需要打印机对象的,如果不用多态机制实现的话(Person.h中注释的代码部分),就是给两种打印机单独定义个操作的方法,然后在Person.m(代码中注释的部分)中用具体的打印机对象进行操作,在main.m文件中,我们看到,当Person需要使用哪个打印机的时候,就去调用指定的方法:

    1
    2
    [person printWithBlack:blackPrint];//调用黑白打印机  
    [person printWithColor:colorPrint];//调用彩色打印机

    这种设计就不好了,为什么呢?假如现在又有一种打印机,那么我们还需要在Person.h中定义一种操作这种打印机的方法,那么后续如果在添加新的打印机呢?还在添加方法吗?那么Person.h文件就会变得很臃肿。所以这时候多态就体现到好处了,使用父类类型,在Person.h中定义一个方法就可以了:

    1
    - (void) doPrint:(Printer *)printer;

    这里看到了,这个方法的参数类型就是父类的类型,这就是多态,定义类型为父类类型,实际类型为子类类型

    1
    2
    3
    - (void) doPrint:(Printer *)printer{  
        [printer print];  
    }

    这里调用print方法,就是传递进来的实际类型的print方法。

    1
    2
    3
    4
    5
    Printer *p1 = [[ColorPrinter alloc] init];  
    Printer *p2 = [[BlackPrinter alloc] init];  
               
    [person doPrint:p1];  
    [person doPrint:p2];

    这里的p1,p2表面上的类型是Printer,但是实际类型是子类类型,所以会调用他们自己对应的print方法。

    从上面的例子中我们就可以看到多态的特新很是重要,当然也是三大特性中比较难理解的,但是在coding的过程中,用多了就自然理解了,没必要去刻意的理解。

     
    搜索CocoaChina微信公众号:CocoaChina
    微信扫一扫
    订阅每日移动开发及APP推广热点资讯
    公众号:CocoaChina
    我要投稿  收藏文章
    29
    上一篇:imageIO完成渐进加载图片
    下一篇:怎样做一个iOS App的启动分层引导动画?
    我来说两句
    发表评论
    您还没有登录!请登录注册
    所有评论(1)
    11月的淡然2016-03-09 14:18:00
    不错~
      回复

    综合评论

    展开全文
  • 创建子类对象,a指向的应该是一个Dog类的对象,a.bark(d); 调用的应该是Dog重载的bark方法才对,可为什么最终调用的还是接口Animal的方法呢。 ```java public interface Animal { default void bark(Animal a) ...
  • 方法,一个子类要初始化需要先初始化父类。 1.1 类加载的时机 一个类型从被加载到虚拟机内存中开始,到卸载出内存为止,它的生命周期将会经历加载、验证、准备、解析、初始化、使用和卸载七个阶段。其中验证、准备、...
  • 假设有父类Fu ,其子类为Zi ,从对象的内存角度,假设Fu类里面的变量占内存2M, ...因为子类中有一个隐藏的引用super会指向父类实例,所以在实例化子类之前会先实例化一个父类,也就是说会先执行父类的构造函数.所以
  • 例如父类Person指向子类Student:Person p = new Student(); 父类类型实例p可以调用父类中定义的所有属性和方法,而对于子类中定义而父类中没有的方法,它是无可奈何的; 同时,父类中的一个方法只有在在父类中定义...
  • 前文:C++之继承探究(四):继承的使用 把子类对象当作父类对象使用 分三种情况:   (1) 将子类对象赋值给父类对象。    ——把等号右边的子类对象,赋值给等号左边的父类...父类引用指向子类对象时,当通过父
  • C++ 子类对象父类对象使用

    千次阅读 2017-11-15 15:35:20
    子类对象父类对象使用1⃣️把子类对象直接赋值给父类对象(等号左边是父类对象,等号右边是子类对象子类对象赋值给父类对象,仅仅把继承自父类部分成员函数赋值给父类对象 赋值完成后等号左边依然是一个父类...
  • 父类引用子类对象,父类对象和子类对象之间的引用关系和区别
  • Java继承,子类与父类初始化的执行顺序,父类引用和子类引用调用子类对象的区别子类与父类初始化的执行顺序父类引用和子类引用调用子类对象的区别 父类代码 public class Father { String name = "父类"; static...
  • 原文地址:关于C++的子类指针指向父类.作者:leon  1,直接用基类指针引用基类对象 2,直接用派生类指针引用派生类对象 3,用基类指针引用一个派生类对象,由于派生类对象也是基类的对象,所以这种引用是安全...
  • C++的子类指针指向父类

    千次阅读 2015-04-10 15:38:13
    1,直接用基类指针引用基类对象 2,直接用派生类指针引用派生类对象 3,用基类指针引用一个派生类对象,由于派生类对象也是基类的对象,所以这种引用是安全的, 但是只能引用基类成员。若试图通过基类指针引用...
  • 原文地址:关于C++的子类指针指向父类.作者:leon   1,直接用基类指针引用基类对象 2,直接用派生类指针引用派生类对象 3,用基类指针引用一个派生类对象,由于派生类对象也是基类的对象,所以这种引用是...

空空如也

空空如也

1 2 3 4 5 ... 20
收藏数 6,604
精华内容 2,641
关键字:

子类对象指向父类引用