OC的特有语法-分类Category、 类的本质、description方法、SEL、NSLog输出增强、点语法、变量作用域、@property @synthesize关键字、Id、OC语言构造方法
一、 分类-Category
@interface 类名 (分类名称)
// 方法声明
@end
@implementation 类名 (分类名称)
// 方法实现
@end
二、 类的本质
typedef struct objc_class *Class;
Class c = [Person class]; // 类方法
或者
Person *p = [Person new];
Class c2 = [p class]; // 对象方法
Class c = [Person class];
Person *p2 = [c new];
使用NSLog和%@输出某个对象时,会调用对象的-description方法,并拿到返回值进行输出
使用NSLog和%@输出某个类对象时,会调用类的+description方法,并拿到返回值进行输出
区别
+description方法决定了类对象的输出结果,即类本身
-description方法决定了实例对象的输出结果,即Person创建的对象。
//
// main.m
// 07-description方法
//
// Created by apple on 13-8-8.
// Copyright (c) 2013年 itcast. All rights reserved.
// #import <Foundation/Foundation.h>
#import "Person.h" void test9()
{
// 输出当前函数名
NSLog(@"%s\n", __func__);
} int main()
{
// 输出行号
NSLog(@"%d", __LINE__); // NSLog输出C语言字符串的时候,不能有中文
// NSLog(@"%s", __FILE__); // 输出源文件的名称
printf("%s\n", __FILE__); test9(); Person *p = [[Person alloc] init]; // 指针变量的地址
NSLog(@"%p", &p);
// 对象的地址
NSLog(@"%p", p);
// <类名:对象地址>
NSLog(@"%@", p); return ;
} void test2()
{
Class c = [Person class]; // 1.会调用类的+description方法
// 2.拿到+description方法的返回值(NSString *)显示到屏幕上
NSLog(@"%@", c);
} void test1()
{
Person *p = [[Person alloc] init];
p.age = ;
p.name = @"Jack";
// 默认情况下,利用NSLog和%@输出对象时,结果是:<类名:内存地址> // 1.会调用对象p的-description方法
// 2.拿到-description方法的返回值(NSString *)显示到屏幕上
// 3.-description方法默认返回的是“类名+内存地址”
NSLog(@"%@", p); //Person *p2 = [[Person alloc] init];
//NSLog(@"%@", p2); //NSString *name = @"Rose"; //NSLog(@"我的名字是%@", name); Person *p2 = [[Person alloc] init];
p2.age = ;
p2.name = @"Jake"; NSLog(@"%@", p2);
}
typedef struct objc_selector *SEL;
SEL s = @selector(test);
SEL s2 = NSSelectorFromString(@"test");
// 将SEL对象转为NSString对象
NSString *str = NSStringFromSelector(@selector(test));
Person *p = [Person new];
// 调用对象p的test方法
[p performSelector:@selector(test)];
// 下面的代码会引发死循环
- (void)test {
[self performSelector:_cmd];
}
六、点语法
(一)认识点语法
声明一个Person类:
//
// Person.h
// 04-点语法
//
// Created by apple on 13-8-7.
// Copyright (c) 2013年 itcast. All rights reserved.
// #import <Foundation/Foundation.h> @interface Person : NSObject
{
int _age;
NSString *_name;
} - (void)setAge:(int)age;
- (int)age; - (void)setName:(NSString *)name;
- (NSString *)name; @end
Person类的实现:
//
// Person.m
// 04-点语法
//
// Created by apple on 13-8-7.
// Copyright (c) 2013年 itcast. All rights reserved.
// #import "Person.h" @implementation Person - (void)setAge:(int)age
{
//_age = age; NSLog(@"setAge:"); // 会引发死循环
//self.age = age; // [self setAge:age];
} - (int)age
{
NSLog(@"age");
return _age;
// 会引发死循环
//return self.age;// [self age];
} - (void)setName:(NSString *)name
{
_name = name;
} - (NSString *)name
{
return _name;
} @end
点语法的使用:
//
// main.m
// 04-点语法
//
// Created by apple on 13-8-7.
// Copyright (c) 2013年 itcast. All rights reserved.
// #import <Foundation/Foundation.h>
#import "Person.h" int main(int argc, const char * argv[])
{
Person *p = [Person new]; // 点语法的本质还是方法调用
p.age = ; // [p setAge:10]; int a = p.age; // [p age]; p.name = @"Jack"; NSString *s = p.name; NSLog(@"%@", s); return ;
}
二)点语法的作用
OC设计点语法的目的,是为了让其他语言的开发者可以很快的上手OC语言开发,使用点语法,让它和其他面向对象的语言如java很像。
(三)点语法的本质
点语法的本质是方法的调用,而不是访问成员变量,当使用点语法时,编译器会自动展开成相应的方法。切记点语法的本质是转换成相应的set和get方法,如果没有set和get方法,则不能使用点语法。
如:
Stu.age=10;展开为:[stu setAge:10];
int a=stu.age;展开为:[stu age];
编译器如何知道是set方法还是get方法?主要是看赋值(可以使用断点调试来查看)。
在OC中访问成员变量只有一种方式即使用-> 如stu->age,这种情况要求在@public的前提下。
(四)点语法的使用注意
下面的使用方式是一个死循环:
(1)在set方法中,self.age=age;相当于是[self setAge:age];
(2)在get方法中,return self.age;相当于是[self age];
点语法(找出不合理的地方)
#import <Foundation/Foundation.h>
@interface Person : NSObject
{
int _age;
}
- (void)setAge:(int)age;
- (int)age;
@end @implementation Person
{
int _age;
}
- (void)setAge:(int)age
{
_age = age;
// 会引发死循环
// self.age = age;
}
- (int)age
{
return _age;
// 会引发死循环
// return self.age;
}
@end // 2个不合理
七、变量作用域
(一)变量的作用域主要分为四种:
(1)@public (公开的)在有对象的前提下,任何地方都可以直接访问。
(2)@protected (受保护的)只能在当前类和子类的对象方法中访问
(3)@private (私有的)只能在当前类的对象方法中才能直接访问
(4)@package (框架级别的)作用域介于私有和公开之间,只要处于同一个框架中就可以直接通过变量名访问
#import <Foundation/Foundation.h> @interface Person : NSObject
{
int _no; @public // 在任何地方都能直接访问对象的成员变量
int _age; @private // 只能在当前类的对象方法中直接访问
int _height; @protected // 能在当前类和子类的对象方法中直接访问
int _weight;
int _money;
} - (void)setHeight:(int)height;
- (int)height; - (void)test;
@end
(二)使用注意和补充
(1)在类的实现即.m文件中也可以声明成员变量,但是因为在其他文件中通常都只是包含头文件而不会包含实现文件,所以在这里声明的成员变量是@private的。在.m中定义的成员变量不能喝它的头文件.h中的成员变量同名,在这期间使用@public等关键字也是徒劳的。
(2)在@interface @end之间声明的成员变量如果不做特别的说明,那么其默认是protected的。
(3)一个类继承了另一个类,那么就拥有了父类的所有成员变量和方法,注意所有的成员变量它都拥有,只是有的它不能直接访问。
八、@property @synthesize关键字
注意:这两个关键字是编译器特性,让xcode可以自动生成getter和setter的声明和实现。
(一)@property 关键字
@property 关键字可以自动生成某个成员变量的setter和getter方法的声明
@property int age;
编译时遇到这一行,则自动扩展成下面两句:
- (void)setAge:(int)age;
- (int)age;
(二)@synthesize关键字
@synthesize关键字帮助生成成员变量的setter和getter方法的实现。
语法:@synthesize age=_age;
相当于下面的代码:
- (void)setAge:(int)age
{
_age=age;
}
- (int)age
{
Return _age;
}
(三)关键字的使用和使用注意
类的声明部分:

类的实现部分:

测试程序:

新版本中:
类的声明部分:

类的实现部分:

测试程序:

(1)在老式的代码中,@property只能写在@interface @end中,@synthesize只能写在@implementation @end中,自从xcode 4.4后,@property就独揽了@property和@synthesize的功能。
(2)@property int age;这句话完成了3个功能:1)生成_age成员变量的get和set方法的声明;2)生成_age成员变量set和get方法的实现;3)生成一个_age的成员变量。
注意:这种方式生成的成员变量是private的。
(3)可以通过在{}中加上int _age;显示的声明_age为protected的。
(4)原则:get和set方法同变量一样,如果你自己定义了,那么就使用你已经定义的,如果没有定义,那么就自动生成一个。
(5)手动实现:
1)如果手动实现了set方法,那么编译器就只生成get方法和成员变量;
2)如果手动实现了get方法,那么编译器就只生成set方法和成员变量;
3)如果set和get方法都是手动实现的,那么编译器将不会生成成员变量。

九、Id
id 是一种类型,万能指针,能够指向\操作任何的对象。
注意:在id的定义中,已经包好了*号。Id指针只能指向os的对象。
id 类型的定义
Typedef struct objc object{
Class isa;
} *id;
局限性:调用一个不存在的方法,编译器会马上报错。
//
// main.m
// 01-id
//
// Created by apple on 13-8-8.
// Copyright (c) 2013年 itcast. All rights reserved.
// #import <Foundation/Foundation.h>
#import "Person.h" void test(id d)
{ } int main(int argc, const char * argv[])
{ @autoreleasepool {
Person *p = [Person new];
//[p fsdfdsfd]; NSObject *o = [Person new]; // id == NSObject *
// 万能指针,能指向\操作任何OC对象
id d = [Person new]; [d setAge:]; [d setObj:@""]; NSLog(@"%d", [d age]);
}
return ;
}
十、OC语言构造方法
一、构造方法
构造方法的调用
完整的创建一个可用的对象:Person *p=[Person new];
New方法的内部会分别调用两个方法来完成2件事情,1)使用alloc方法来分配存储空间(返回分配的对象);2)使用init方法来对对象进行初始化。
可以把new方法拆开如下:
1.调用类方法+alloc分配存储空间,返回未经初始化的对象
Person *p1=[person alloc];
2.调用对象方法-init进行初始化,返回对象本身
Person *p2=[p1 init];
3.以上两个过程整合为一句:
Person *p=[[Person alloc] init];
说明:init方法就是构造方法,是用来初始化对象的方法,注意这是一个对象方法,一减号开头。默认初始化完毕后,所有成员变量的值都为0。
构造方法使用注意
(1)子类拥有的成员变量包括自己的成员变量以及从父类继承而来的成员变量,在重写构造方法的时候应该首先对从父类继承而来的成员变量先进行初始化。
(2)原则:先初始化父类的,再初始化子类的。
(3)重写构造方法的目的:为了让对象方法一创建出来,成员变量就会有一些固定的值。
(4)注意点:#1先调用父类的构造方法[super init]; #2再进行子类内部成员变量的初始化。
二、自定义构造方法
(一)自定义构造方法的规范
(1)一定是对象方法,以减号开头
(2)返回值一般是id类型
(3)方法名一般以initWith开头
(二)自定义构造方法的代码实现
Person类的声明,其中声明了两个接收参数的自定义构造方法
//
// Person.h
// 03-自定义构造方法
//
// Created by apple on 13-8-8.
// Copyright (c) 2013年 itcast. All rights reserved.
// #import <Foundation/Foundation.h> @interface Person : NSObject
@property NSString *name;
@property int age; /*
自定义构造方法的规范
1.一定是对象方法,一定以 - 开头
2.返回值一般是id类型
3.方法名一般以initWith开头
*/ - (id)initWithName:(NSString *)name; - (id)initWithAge:(int)age; // initWithName:andAge:
- (id)initWithName:(NSString *)name andAge:(int)age; @end
Person类的实现
//
// Person.m
// 03-自定义构造方法
//
// Created by apple on 13-8-8.
// Copyright (c) 2013年 itcast. All rights reserved.
// #import "Person.h" @implementation Person - (id)init
{
if ( self = [super init] )
{
_name = @"Jack";
}
return self;
} - (id)initWithName:(NSString *)name
{ if ( self = [super init] )
{
_name = name;
} return self;
} - (id)initWithAge:(int)age
{
if ( self = [super init] )
{
_age = age;
}
return self;
} - (id)initWithName:(NSString *)name andAge:(int)age
{
if ( self = [super init] )
{
_name = name;
_age = age;
}
return self;
} @end
Student继承自Person类,声明了一个接收三个参数的构造方法
//
// Student.h
// 03-自定义构造方法
//
// Created by apple on 13-8-8.
// Copyright (c) 2013年 itcast. All rights reserved.
// #import "Person.h" @interface Student : Person
@property int no; - (id)initWithNo:(int)no; - (id)initWithName:(NSString *)name andAge:(int)age andNo:(int)no; @end
Student类的实现
//
// Student.m
// 03-自定义构造方法
//
// Created by apple on 13-8-8.
// Copyright (c) 2013年 itcast. All rights reserved.
// #import "Student.h" @implementation Student
- (id)initWithNo:(int)no
{
if ( self = [super init] )
{
_no = no;
}
return self;
} // 父类的属性交给父类方法去处理,子类方法处理子类自己的属性
- (id)initWithName:(NSString *)name andAge:(int)age andNo:(int)no
{
// 将name、age传递到父类方法中进行初始化
if ( self = [super initWithName:name andAge:age])
{
_no = no;
} return self;
} //- (id)initWithName:(NSString *)name andAge:(int)age andNo:(int)no
//{
// if ( self = [super init] )
// {
// _no = no;
// //_name = name;
// self.name = name;
// self.age = age;
//
// //[self setName:name];
// //[self setAge:age];
// }
//
// return self;
//}
@end
测试主程序
//
// main.m
// 03-自定义构造方法
//
// Created by apple on 13-8-8.
// Copyright (c) 2013年 itcast. All rights reserved.
// #import <Foundation/Foundation.h>
#import "Person.h"
#import "Student.h" int main(int argc, const char * argv[])
{ @autoreleasepool {
Student *p = [[Student alloc] initWithName:@"Jim" andAge: andNo:];
NSLog(@"");
}
return ;
}
(三)自定义构造方法的使用注意
(1)自己做自己的事情
(2)父类的方法交给父类的方法来处理,子类的方法处理子类自己独有的属性
OC的特有语法-分类Category、 类的本质、description方法、SEL、NSLog输出增强、点语法、变量作用域、@property @synthesize关键字、Id、OC语言构造方法的更多相关文章
- OC @property @synthesize和id
文顶顶 OC语言@property @synthesize和id OC语言@property @synthesize和id 一.@property @synthesize关键字 注意:这两个关键字 ...
- OC语言@property @synthesize和id
OC语言@property @synthesize和id 一.@property @synthesize关键字 注意:这两个关键字是编译器特性,让xcode可以自动生成getter和setter的声明 ...
- 李洪强iOS开发之OC语言@property @synthesize和id
OC语言@property @synthesize和id 一.@property @synthesize关键字 注意:这两个关键字是编译器特性,让xcode可以自动生成getter和setter的声明 ...
- 类的本质、description方法、SEL、NSLog输出增强
一.类的本质 1.类也是个对象 其实类也是一个对象,是Class类型的对象,简称“类对象” Class类型的定义 typedef struct objc_class *Class; 类名就代表着类对象 ...
- 「OC」@property @synthesize和id
一.@property @synthesize关键字 这两个关键字是编译器特性,让Xcode可以自动生成getter和setter. (一)@property 关键字 @property 关键字可以自 ...
- OC特有语法:分类category,给NSString增加方法计算字符串中数字的个数
1:分类的使用场景:想对一个类,扩充一些功能,而又不改变原来类的模型,也不用继承,这时OC中的特有语法:分类可以做到: 当然分类也是一个类,也需要声明和实现,声明在.h文件中,实现在.m文件中,格式如 ...
- iOS分类(category),类扩展(extension)—史上最全攻略
背景: 在大型项目,企业级开发中多人同时维护同一个类,此时程序员A因为某项需求只想给当前类currentClass添加一个方法newMethod,那该怎么办呢? 最简单粗暴的方式是把newMethod ...
- Objective-C 点语法 成员变量的作用域 @property和@synthesize关键字 id类型
点语法 1.利用点语法替换set方法和get方法 方法调用 Student *stu = [Student new]; [stu setAge : 18]; int age = [stu age]; ...
- C# 中一些类关系的判定方法 C#中关于增强类功能的几种方式 Asp.Net Core 轻松学-多线程之取消令牌
1. IsAssignableFrom实例方法 判断一个类或者接口是否继承自另一个指定的类或者接口. public interface IAnimal { } public interface ID ...
随机推荐
- 数据抓取分析(python + mongodb)
分享点干货!!! Python数据抓取分析 编程模块:requests,lxml,pymongo,time,BeautifulSoup 首先获取所有产品的分类网址: def step(): try: ...
- 在centos7上实现LAMP的全过程及实现wordpress
原理 http使用方法一编译安装,php独立服务fpm实现. 软件版本 在本次实验中,我们需要用到的软件版本如下: apr-1.6.2 apr-util-1.6.0 httpd-2.4.28 mari ...
- 程序员节应该写博客之.NET下使用HTTP请求的正确姿势
程序员节应该写博客之.NET下使用HTTP请求的正确姿势 一.前言 去年9月份的时候我看到过外国朋友关于.NET Framework下HttpClient缺陷的分析后对HttpClient有了一定的了 ...
- 域名系统DNS
一.域名系统是什么 域名系统其实就是一个把主机名解析为IP地址的名字系统. 因特网使用层次树状结构的命名方法,并使用分布式的域名系统DNS.因特网的域名系统DNS被设计成一个联机分布式数据库系统,并采 ...
- [LeetCode] Reverse Pairs 翻转对
Reverse Pairs 翻转对 题意 计算数组里面下标i小于j,但是i的值要大于j的值的两倍的搭配的个数(也就是可能会有多种搭配):网址 做法 这道题显然是不允许使用最简单的方法:两次循环,逐次进 ...
- Leetcode题解(29)
93. Restore IP Addresses 题目 分析:多重循环,判断小数点合适的位置 代码如下(copy网上) class Solution { public: vector<strin ...
- Leetcode题解(25)
77. Combinations 题目 分析:求给定数字n,k的组合数,方法是采用深度搜索算法,代码如下(copy网上代码) class Solution { public: void dfs77(v ...
- python 进程池(multiprocessing.Pool)和线程池(threadpool.ThreadPool)的区别与实例
一般我们是通过动态创建子进程(或子线程)来实现并发服务器的,但是会存在这样一些缺点: 1.动态创建进程(或线程)比较耗费时间,这将导致较慢的服务器响应. 2.动态创建的子进程通常只用来为一个客户服务 ...
- c++学习笔记---02---从一个小程序说起
从一个小程序说起 这一讲的主要目的是帮助大家在C语言的背景知识上与C++建立联系. 问题探索 问题:对一个整型数组求和. 要求:定义一个存储着 n 个元素的数组,要求用C语言完成这个任务. 赶紧的:大 ...
- js实现前端下载文件
在前端下载文本格式的文件时,可采用下面的方式: (1)创建基于文件内容的Blob对象: (2)通过URL上的createObjectURL方法,将blob对象转换成一个能被浏览器解析的文件地址. (3 ...