打赏

objective-c 语法快速过(3)

oc 里的匿名对象 

 oc 这里,很少用到,因为并不适用于oc的内存管理,只是面试笔试也许出现,要求能看懂,不要在项目里这样写,因为写匿名对象,会造成内存泄露

#import <Foundation/Foundation.h>
@interface Car : NSObject
{
    @public
    int speed;
}
- (void)run;
@end

@implementation Car
- (void)run
{
    NSLog(@"%d", speed);
}
@end
int main()
{
    //所谓匿名对象,就是没有名字的对象,看不到,但是对象确实存在
    [Car new]->speed = 300;//没有指针变量指向对象,而是直接调用类的成员变量,因为每次使用[Car new]都会从新创建一个新对象,故不是300,而是默认初始化的值0
    [[Car new] run];//0
    
   // Car *c = [Car new];
   // c->speed = 100;
   // [c run];//100
    return 0;
}

能看懂什么意思就行

类的成员变量的命名规范

  • 成员变量都以下划线 _ 开头
  • 可以跟get方法的名称区分开
  • 可以跟其他局部变量区分开,一看到下划线开头的变量,肯定是类的成员变量

OC弱语法

1、OC是动态检测错误,OC里调用一个没有声明也没有实现的对象方法,则不会编译报错而是警告,链接也能通过,只有运行才检测出错

2、Oc里调用只有声明,但是没有实现的对象方法,这编译也是警告,链接通过,运行才出错

3、Oc调用只有实现(没声明)的方法,则没有问题!因为运行时才检测程序的问题,声明其实只是摆设,删掉是没事的。只不过开发中,必须规范!该写都要写上。即使不报错。

类方法

直接可以用类名来执行的方法(类本身会在内存中占据存储空间,里面有类\对象方法列表)

1.  类方法和对象方法对比

1)   对象方法

  • 减号-开头
  • 只能让对象调用,没有对象,这个方法根本不可能被执行
  • 对象方法能访问实例变量(成员变量)

2)   类方法

  • 加号+开头
  • 只能用类名调用,对象不能调用
  • 类方法中不能访问实例变量(成员变量)
  • 使用场合:当不需要访问成员变量的时候,尽量用类方法

3)       类方法和对象方法可以同名

4)       提高程序性能

类方法的好处和使用场合:不依赖于对象,执行效率高, 能用类方法,尽量用类方法

// 类方法都是以+开头
+ (void)printClassName;
- (void)test;
+ (void)test; //可以允许类方法和对象方法同名

self 关键字用来指明对象是当前方法的接收者。 

  • 当成员变量和局部变量同名时,采取就近原则,访问的是局部变量
  • 用self访问成员变量,区分同名的局部变量

细节:

1)    出现的地方:可以出现在所有的OC方法中(对象方法\类方法),但是不能出现在函数里

2)    作用:

使用 "self->成员变量名" 访问当前方法调用的成员变量

使用 "[self 方法名];" 来调用方法(对象方法\类方法)

#import <Foundation/Foundation.h>
@interface Person : NSObject
{
    int _age;
}
- (void)setAge:(int)age;
- (int)age;
- (void)test;
@end

@implementation Person
- (void)setAge:(int)age
{
    // _age = age;相当于
    self->_age = age;
}
- (int)age
{
    return self->_age;
}
- (void)test
{
    // self:指向了方向调用者,代表着当前对象
    int _age = 20;
    NSLog(@"Person的年龄是%d岁", self->_age);
}

@end

int main()
{
    Person *p = [Person new];
    
    [p setAge:10];
    
    [p test];
    
    return 0;
}
/* self的用途:
 1> 谁调用了当前方法,self就代表谁
 * self出现在对象方法中,self就代表对象
 * self出现在类方法中,self就代表类
 2> 在对象方法利用"self->成员变量名"访问当前对象内部的成员变
[self 方法名]可以调用其他对象方法\类方法 */
#import <Foundation/Foundation.h>
@interface Dog : NSObject
- (void)bark;
- (void)run;
@end

@implementation Dog
- (void)bark
{
    NSLog(@"汪汪汪");
}
- (void)run
{
    [self bark];//self代表指针d指向的对象,NSLog(@"汪汪汪");
    NSLog(@"跑跑跑");
}
@end

int main()
{
    Dog *d = [Dog new];
    
    [d run];
    
    return 0;
}

低级错误:

用self去调用函数

类方法中用self调用对象方法,对象方法中用self调用类方法

self死循环

#import <Foundation/Foundation.h>
@interface Person : NSObject
- (void)test;
+ (void)test;

- (void)test1;
+ (void)test2;

- (void)haha1;
+ (void)haha2;
@end

@implementation Person
- (void)test
{
    NSLog(@"调用了-test方法");
    // 如果有[self text];这句,就会引发死循环。因为[self test];self代表对象,一直调用对象方法test
}
+ (void)test
{
    NSLog(@"调用了+test方法");
    // 引发死循环 [self test];self代表类,一直调用类方法test
}
//自动识别
- (void)test1
{
    [self test]; // ok,-test
}

+ (void)test2
{
    [self test]; // ok,+test
}

- (void)haha1
{
    NSLog(@"haha1-----");
}
//函数
void haha3()
{  
}

+ (void)haha2
{
     haha3();//ok,一定注意函数和方法是不一样的!
    //[self haha3];//error,不能用self调用函数
    //[self haha1];//error,类方法里,不能用self调用对象方法,相反,在对象方法里,也不能用self调用类方法
}
@end

int main()
{
    [Person haha2];//直接调用类方法
    //Person *p = [Person new];
    //[p test1];
    return 0;
}

 

原则上(如果不使用 ARC,也就是自动引用计数),那么

创建一个新对象,都要请求分配内存,在完成对该对象的操作时,必须释放其所用的内存空间。类似 c++的内存管理。

 

与 C 语言兼容的地方:

预处理:

#define 语句和 c 一样

#运算符: #define str(x) #x

表示在调用该宏时,预处理程序根据宏参数创建C 风格的常量字符串。

例如:str("hello")将产生"\"hello"\"

##运算符: 表示用于把两个标记连在一起

#import 语句;相当于#include 语句,但是 #import 可自动防止同一个文件被导入多次。

#条件编译语句(#ifdef 、#endif 、 #else 、 #ifndef)和 C 一样

#undef 语句 消除特定名称的定义


其他基本的C 语言特性:

数组、函数、指针、结构、联合的用法和C 一样。

Compound Literal 是包含在括号之内的类型名称,之后是一个初始化列表。

例如

如果 intPtr 为 int * 类型:
intPtr = (int[100]){[0] = 1, [50] = 50, [99] = 99};

如果数组大小没有说明,则有初始化列表确定。

其他如循环语句 (do while、while、for) 、条件语句(if 语句(if-else、复合判断条件等) 、switch 语句)、 Boolean(YES NO)、条件运算符、goto 语句、空语句、逗号表达式、sizeof 运算符、命令行参数、位操作都 和 C 一样 。

oc 的继承

oc的继承只支持单一继承,和java类似,也就是儿子只能有一个爸爸,但是

可以通过 Objective-C 的分类和协议特性获取多继承的优点 

而c++支持单一和多重继承。

好处:

不改变原来模型的基础上,拓充方法

建立了类与类之间的联系

抽取了公共代码

坏处:

耦合性强

基本上所有类的根类是NSObject类

如图:

animal类拥有NSObject类的new方法,子类dog和cat同时拥有前两层类的所有方法和属性。类似 c++和 java

#import <Foundation/Foundation.h>
/* 1.继承的好处:
 1> 抽取重复代码
 2> 建立了类之间的关系
 3> 子类可以拥有父类中的所有成员变量和方法
 2.注意点
 1> 基本上所有类的根类是NSObject
 */
/********Animal的声明*******/
@interface Animal : NSObject
{
    int _age;
    double _weight;
}

- (void)setAge:(int)age;
- (int)age;

- (void)setWeight:(double)weight;
- (double)weight;
@end
/********Animal的实现*******/
@implementation Animal
- (void)setAge:(int)age
{
    _age = age;
}
- (int)age
{
    return _age;
}

- (void)setWeight:(double)weight
{
    _weight = weight;
}
- (double)weight
{
    return _weight;
}
@end

/********Dog*******/
// 继承了Animal,相当于拥有了Animal里面的所有成员变量和方法
// Animal称为Dog的父类,Dog称为Animal的子类
@interface Dog : Animal
@end

@implementation Dog
@end

/********Cat*******/
@interface Cat : Animal
@end

@implementation Cat
@end

int main()
{
    Dog *d = [Dog new];
    
    [d setAge:10];
    
    NSLog(@"age=%d", [d age]);
    return 0;
}

oc 继承里的细节(类似其他面向对象语言)

父类必须声明在子类的前面

子类和父类不能有相同的成员变量,但是方法可以重写

方法的重写问题:子类重新实现父类中的某个方法,也就是覆盖了父法

调用某个方法时,优先去当前类中找,如果找不到,去父类中找

这是内部原理。

每个对象都有一个isa指针,指向对象属于的类,且记住:每个类里(oc的)都有一个superclass指针,指向自己的父类。

这样通过对象就能找到对象属于的类,也能找到类的父类。而这些指针就在NSSobject类里。

内存结构:

继承的使用场合

 1> 当两个类拥有相同属性和方法的时候,就可以将相同的东西抽取到一个父类中

 2> 当A类完全拥有B类中的部分属性和方法时,可以考虑让B类继承A类

 // 继承:xx 是 xxx

 // 组合(也叫聚合关系):xxx 拥有 xxx

super关键字

实现重写之后,还可以调用父类的对象方法和类方法,super的作用;直接调用父类中的某个方法

 1、super处在对象方法中,那么就会调用父类的对象方法

 2、super处在类方法中,那么就会调用父类的类方法

使用场合:

子类重写父类的方法时想保留父类的一些行为。

在oc里,简单的多,调用某方法,先是在所在类就近找,找不到去父类找。太简单了。如果重写了父类方法,那么只能调用子类重写的这个方法了,如果想保留父类原方法定义的功能,可以用super。

oc 的多态

多态的体现

Person *p = [Student new];

p->age = 100;

[p walk];

子类对象赋值给父类指针,父类指针访问对应的子类的继承来的属性和方法

多态的局限性

不能访问子类的特有的属性或方法(可以考虑强制转换)

多态的细节

动态绑定:在运行时根据对象的类型确定动态调用的方法

注意点:

 1.没有继承就没有多态

 2.代码的体现:父类类型的指针指向子类对象

 3.好处:如果函数\方法参数中使用的是父类类型,可以传入父类、或者子类对象

 4.局限性:父类类型的变量 不能 直接调用子类特有的方法。必须强转为子类类型变量后,才能直接调用子类特有的方法(类似c++的赋值兼容)

#import <Foundation/Foundation.h>
// 动物
@interface Animal : NSObject
- (void)eat;
@end

@implementation Animal
- (void)eat
{
    NSLog(@"Animal-吃东西----");
}
@end
//
@interface Dog : Animal
- (void)run;//子类新增的对象方法run
@end

@implementation  Dog
- (void)run
{
    NSLog(@"Dog---跑起来");
}
- (void)eat//重写父类的-eat方法
{
    NSLog(@"Dog-吃东西----");
}
@end

//
@interface Cat : Animal
@end

@implementation Cat
- (void)eat//重写父类的对象方法eat
{
    NSLog(@"Cat-吃东西----");
}
@end
// 如果参数中使用的是父类类型指针,可以传入父类or子类对象
void feed(Animal *a)
{
    [a eat];
}

int main()
{
    Animal *aa = [Dog new];//父类指针指向子类的对象
    //[aa run];弱语法,只警告!但是在java或者c++里,早就报错了!父类指针不能访问子类特有的方法,虽然弱语法,但不推荐,自己要认为是错的
    // 将父类对象aa转为子类 Dog * 类型的变量就ok了,和c++类似
    Dog *dd = (Dog *)aa;
    [dd run];//ok,访问的是子类的对象方法 run 
    
    //Dog *d = [Dog new];
    //[d run];//ok
    
    /* Animal *aa = [Animal new];父类对象的指针aa
    feed(aa);可以传入父类对象做参数,调用的父类的eat方法
    Dog *dd = [Dog new];子类对象指针dd
    feed(dd); 也可以传入子类的对象做参数,调用的子类的eat方法
    Cat *cc = [Cat new];
    feed(cc);调用子类eat,传入子类对象参数   */

    // 多态:父类指针指向子类对象
    Animal *a = [Dog new];
    // 调用方法时会检测对象的真实形象,动态
    [a eat];调用的时子类的eat方法
    */
    return 0;
}

多态的局限性:

父类类型的变量 不能 直接 调用子类特有的方法。

联系c++

c++是使用了虚函数,(包括纯虚函数和抽象类)对公有继承的子类的方法,重写虚函数,父类指针或者引用指向子类,那么就能调用子类的重写虚函数,指向父类,就是调用父类的虚函数,实现动态联编。

oc里,没有那么复杂,就是直接子类继承父类,那么重写父类某个对象方法,这只需要父类指针指向子类对象,那么就调用子类的重写方法,同样也不能调用子类特有的方法。

由此断定,oc的方法都是虚方法!不用和c++一样用virtual声明!且oc的重写也是多态的一种,oc里所有的方法访问属性都是公有的!而类成员变量默认是保护的。和c++有些区别,c++是默认都是私有的。

再看oc的弱语法!比如把子类对象指针指向父类

Cat *cat = [Animal new];

c++里肯定错误,但oc里没问题,只警告,但这样不好,不推荐。完全没道理。

再看:

NSString *s = [Cat new];

动物怎么了成了字符串对象了?oc里xcode不报错!但是绝对是不对,不规范。还有,同一个层次的类,猫类指针指向狗类对象,调用eat方法,在oc的弱语法下,还是没问题的,调狗的eat。

oc太弱了!但是在c++里直接就报错了。不可以这样写,即使编译器不报错。

多态的好处

用父类对象接收参数,方法或者函数,即可以接受子类对象,也能接受父类对象,节省代码。否则还要分开写多个方法或者函数。

充分体现面向对象的抽象和具体,通过子类重写继承来的父类的虚方法(oc默认都是),通过父类指针指向子类对象,实现动态多态性!不看指针的类型,而是看指针指向的哪个对象类型,就调用哪个子类对象的虚方法。注意子类特有的方法不可以,必须强转。

 

欢迎关注

dashuai的博客是终身学习践行者,大厂程序员,且专注于工作经验、学习笔记的分享和日常吐槽,包括但不限于互联网行业,附带分享一些PDF电子书,资料,帮忙内推,欢迎拍砖!

 

posted @ 2015-03-04 16:57  dashuai的博客  阅读(898)  评论(0编辑  收藏  举报
Flag Counter欢迎关注微信公众号