nullable關(guān)鍵字(iOS9關(guān)鍵字)
關(guān)鍵字:可以用于屬性,方法返回值和參數(shù)中
關(guān)鍵字作用:提示作用,告訴開發(fā)者屬性信息
關(guān)鍵字目的:迎合swift,swift是個(gè)強(qiáng)語言,swift必須要指定一個(gè)對(duì)象是否為空
關(guān)鍵字好處:提高代碼規(guī)劃,減少溝通成本
關(guān)鍵字僅僅是提供警告,并不會(huì)報(bào)編譯錯(cuò)誤
null_resettablenullable作用:可能為空
nullable 語法1:@property (nonatomic, strong, nullable) NSString *name;
nullable 語法2 * 關(guān)鍵字 變量名: @property (nonatomic, strong) NSString * _Nullable name;
nullable 語法3 : @property (nonatomic, strong) NSString * __nullable name;
nonnull關(guān)鍵字(iOS9關(guān)鍵字)
- nonnull作用:不能為空
nonnull 語法1 : @property (nonatomic, strong, nullable) NSString *name;
nonnull 語法2 * 關(guān)鍵字 變量名 : @property (nonatomic, strong) NSString * _Nonnull name;
nonnull 語法3 : @property (nonatomic, strong) NSString * __nonnull name;
null_resettable關(guān)鍵字(iOS9關(guān)鍵字)
null_resettable作用:get方法不能返回nil,set可以傳入為空葵陵,必須要處理為空情況,重寫get方法
null_resettable 語法(只有這一種語法) : @property (nonatomic, strong, null_resettable) NSString *name;
_Null_unspecified關(guān)鍵字(iOS9關(guān)鍵字)
- _Null_unspecified作用 :不確定是否為空
_Null_unspecified語法:@property (nonatomic , copy ) NSString *_Null_unspecified email;
定義關(guān)鍵字的宏
NS_ASSUME_NONNULL_BEGIN和NS_ASSUME_NONNULL_END:在這兩個(gè)宏之間的屬性都是nonnull
注意 :關(guān)鍵字一般用于對(duì)象埃难,關(guān)鍵字不能用于基本數(shù)據(jù)類型
泛型
- 為什么要推出泛型:迎合swift
- 泛型作用:
- 1.限制類型
- 2.提高代碼規(guī)劃,減少溝通成本,一看就知道集合中是什么東西
- 泛型定義用法:類型<限制類型>
- 泛型聲明:在聲明類的時(shí)候,在類的后面<泛型名稱>
- 注意:泛型僅僅是報(bào)警告
- 泛型好處:
- 1.從數(shù)組中取出來,可以使用點(diǎn)語法
- 2.給數(shù)組添加元素,有提示
*泛型在開發(fā)中使用場景:1.用于限制集合類型
主要:id是不能使用點(diǎn)語法
自定義泛型
- 什么時(shí)候使用泛型?在聲明類的時(shí)候,不確定某些屬性或者方法類型,在使用這個(gè)類的時(shí)候才確定,就可以采用泛型
- 自定義Person,會(huì)一些編程語言(iOS,Java),在聲明Person,不確定這個(gè)人會(huì)什么,在使用Person才知道這個(gè)Person會(huì)什么語言
- 如果沒有定義泛型.默認(rèn)就是id
@interface Language : NSObject
@end
@interface Java : Language
@end
@interface iOS : Language
@end
@interface Person<ObjectType> : NSObject
/**語言*/
@property (nonatomic , strong) ObjectType yuyan;
@end
/*自定義泛型的使用*/
Java *java= [Java new];
iOS *ios = [iOS new];
Person<Java *> *p = [Person new];
p.yuyan = java;
Person<iOS *> *p1 = [Person new];
p1.yuyan = ios;
- 泛型協(xié)變:__covariant
- 子類轉(zhuǎn)父類
Java *java= [Java new];
Person<Java *> *p = [Person new];
p.yuyan = java;
Person<Language *> *p1;
p1 = p;
會(huì)報(bào)如下警告:( 'Person<Java *> *不能轉(zhuǎn)為Person<Language *> *):
Incompatible pointer types assigning to 'Person<Language *> *' from 'Person<Java *> *
將Person改為如下就不會(huì)有警告:
@interface Person<__covariant ObjectType> : NSObject
/**語言*/
@property (nonatomic , strong) ObjectType yuyan;
@end
- 泛型逆變:__contravariant
- 父類轉(zhuǎn)子類
Language *language= [Language new];
Person<Language *> *p = [Person new];
p.yuyan = language;
Person<Java *> *p1 ;
p1 = p;
會(huì)報(bào)如下警告:( 'Person<Language *> *不能轉(zhuǎn)為Person<Java *> *):
Incompatible pointer types assigning to 'Person<Java *> *' from 'Person<Language *> *
將Person改為如下就不會(huì)有警告:
@interface Person< __contravariant ObjectType> : NSObject
/**語言*/
@property (nonatomic , strong) ObjectType yuyan;
@end
泛型注意點(diǎn):在數(shù)組中响迂,一般用可變數(shù)組添加方法细疚,泛型才會(huì)生效疯兼,如果使用不可變數(shù)組,添加元素待侵,泛型沒有效果姨裸。
__kindof關(guān)鍵字
- kindof:相當(dāng)于
- __kindof:表示當(dāng)前類或者它的子類
@interface Person : NSObject
+(__kindof Person *)person;
@end
@implementation Person
+(__kindof Person *)person{
return [[self alloc] init];
}
@end
@interface SubPerson : Person
@end
@implementation SubPerson
@end
- Person類中person類方法返回值分析
- (id):可以調(diào)用任何對(duì)象傀缩,不能進(jìn)行編譯檢查(id可以調(diào)用任何對(duì)象的方法,不合理)
- (instancetype):自動(dòng)識(shí)別當(dāng)前類的對(duì)象售淡,但是沒辦法提示返回值類型
- (Person *):給返回值賦值為SubPerson *會(huì)報(bào)警告-->Incompatible pointer types initializing 'SubPerson *' with an expression of type 'Person *'
- (__kindof Person *):表示返回值是Person *或者Person的子類
const,static,extern簡介
一揖闸、const與宏的區(qū)別(面試題):
-
const簡介
:之前常用的字符串常量,一般是抽成宏鹤啡,但是蘋果不推薦我們抽成宏递瑰,推薦我們使用const常量隙畜。 -
執(zhí)行時(shí)刻
:宏是預(yù)編譯(編譯之前處理),const是編譯階段慎颗。 -
編譯檢查
:宏不做檢查言询,不會(huì)報(bào)編譯錯(cuò)誤运杭,只是替換,const會(huì)編譯檢查撇眯,會(huì)報(bào)編譯錯(cuò)誤虱咧。 -
宏的好處
:宏能定義一些函數(shù),方法玄坦。 const不能营搅。 -
宏的壞處
:使用大量宏梆砸,容易造成編譯時(shí)間久,每次都需要重新替換休蟹。
注意:很多Blog都說使用宏,會(huì)消耗很多內(nèi)存绑榴,我這驗(yàn)證并不會(huì)生成很多內(nèi)存盈魁,宏定義的是常量,常量都放在常量區(qū)赤套,只會(huì)生成一份內(nèi)存容握。
[圖片上傳中车柠。。谈跛。(1)]
// 常見的常量:抽成宏
#define XMGAccount @"account"
#define XMGUserDefault [NSUserDefaults standardUserDefaults]
// 字符串常量
static NSString * const account = @"account";
- (void)viewDidLoad {
[super viewDidLoad];
// 偏好設(shè)置存儲(chǔ)
// 使用宏
[XMGUserDefault setValue:@"123" forKey:XMGAccount];
// 使用const常量
[[NSUserDefaults standardUserDefaults] setValue:@"123" forKey:account];
}
二币旧、const作用:限制類型
1.const僅僅用來修飾右邊的變量(基本數(shù)據(jù)變量p猿妈,指針變量*p)
2.被const修飾的變量是只讀的彭则。
const基本使用
- (void)viewDidLoad {
[super viewDidLoad];
// 定義變量
int a = 1;
// 允許修改值
a = 20;
// const兩種用法
// const:修飾基本變量p
// 這兩種寫法是一樣的俯抖,const只修飾右邊的基本變量b
const int b = 20; // b:只讀變量
int const b = 20; // b:只讀變量
// 不允許修改值
b = 1;
// const:修飾指針變量*p瓦胎,帶*的變量搔啊,就是指針變量.
// 定義一個(gè)指向int類型的指針變量,指向a的地址
int *p = &a;
int c = 10;
p = &c;
// 允許修改p指向的地址漫蛔,
// 允許修改p訪問內(nèi)存空間的值
*p = 20;
// const修飾指針變量訪問的內(nèi)存空間莽龟,修飾的是右邊*p1,
// 兩種方式一樣
const int *p1; // *p1:常量 p1:變量
int const *p1; // *p1:常量 p1:變量
// const修飾指針變量p1
int * const p1; // *p1:變量 p1:常量
// 第一個(gè)const修飾*p1 第二個(gè)const修飾 p1
// 兩種方式一樣
const int * const p1; // *p1:常量 p1:常量
int const * const p1; // *p1:常量 p1:常量
}
三剃毒、const開發(fā)中使用場景:
- 1.需求1:提供一個(gè)方法赘阀,這個(gè)方法的參數(shù)是地址厂镇,里面只能通過地址讀取值,不能通過地址修改值
- 2.需求2:提供一個(gè)方法,這個(gè)方法的參數(shù)是地址酌媒,里面不能修改參數(shù)的地址迄靠。
@implementation ViewController
// const放*前面約束參數(shù)掌挚,表示*a只讀
// 只能修改地址a,不能通過a修改訪問的內(nèi)存空間
- (void)test:(const int * )a
{
// *a = 20;
}
// const放*后面約束參數(shù),表示a只讀
// 不能修改a的地址陡厘,只能修改a訪問的值
- (void)test1:(int * const)a
{
int b;
// 會(huì)報(bào)錯(cuò)
a = &b;
*a = 2;
}
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
int a = 10;
// 需求1:提供一個(gè)方法糙置,這個(gè)方法的參數(shù)是地址是目,里面只能通過地址讀取值,不能通過地址修改值。
// 這時(shí)候就需要使用const揉抵,約束方法的參數(shù)只讀.
[self test:&a];
// 需求2:提供一個(gè)方法嗤疯,這個(gè)方法的參數(shù)是地址身弊,里面不能修改參數(shù)的地址列敲。
[self test1:&a];
}
@end
四戴而、static和extern簡單使用(要使用一個(gè)東西翩蘸,先了解其作用)
-
static作用
:- 修飾局部變量:
1.延長局部變量的生命周期,程序結(jié)束才會(huì)銷毀催首。
2.局部變量只會(huì)生成一份內(nèi)存,只會(huì)初始化一次。
3.改變局部變量的作用域秧耗。
-
修飾全局變量
1.只能在本文件中訪問,修改全局變量的作用域,生命周期不會(huì)改
2.避免重復(fù)定義全局變量
-
extern作用
: - 只是用來獲取全局變量(包括全局靜態(tài)變量)的值舶治,不能用于定義變量
-
extern工作原理
: - 先在當(dāng)前文件查找有沒有全局變量霉猛,沒有找到,才會(huì)去其他文件查找瘫辩。
// 全局變量:只有一份內(nèi)存坛悉,所有文件共享,與extern聯(lián)合使用弧械。
int a = 20;
// static修飾全局變量
static int age = 20;
- (void)test
{
// static修飾局部變量
static int age = 0;
age++;
NSLog(@"%d",age);
}
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
[self test];
[self test];
extern int age;
NSLog(@"%d",age);
}
I
五、static與const聯(lián)合使用
- static與const作用:聲明一個(gè)只讀的靜態(tài)變量
- 開發(fā)使用場景:在
一個(gè)文件中
經(jīng)常使用的字符串常量羞迷,可以使用static與const組合
// 開發(fā)中常用static修飾全局變量,只改變作用域
// 為什么要改變?nèi)肿兞孔饔糜蛳挝停乐怪貜?fù)聲明全局變量。
// 開發(fā)中聲明的全局變量葫慎,有些不希望外界改動(dòng),只允許讀取偷办。
// 比如一個(gè)基本數(shù)據(jù)類型不希望別人改動(dòng)
// 聲明一個(gè)靜態(tài)的全局只讀常量
static const int a = 20;
// staic和const聯(lián)合的作用:聲明一個(gè)靜態(tài)的全局只讀常量
// iOS中staic和const常用使用場景椒涯,是用來代替宏,把一個(gè)經(jīng)常使用的字符串常量祖搓,定義成靜態(tài)全局只讀變量.
// 開發(fā)中經(jīng)常拿到key修改值湖苞,因此用const修飾key,表示key只讀财骨,不允許修改。
static NSString * const key = @"name";
// 如果 const修飾 *key1,表示*key1只讀滑肉,key1還是能改變摘仅。
static NSString const *key1 = @"name";
六、extern與const聯(lián)合使用
開發(fā)中使用場景:在
多個(gè)文件中
經(jīng)常使用的同一個(gè)字符串常量六荒,可以使用extern與const組合掏击。原因:
static與const組合:在每個(gè)文件都需要定義一份靜態(tài)全局變量秩铆。
extern與const組合:只需要定義一份全局變量,多個(gè)文件共享捅膘。
全局常量正規(guī)寫法:開發(fā)中便于管理所有的全局變量寻仗,通常搞一個(gè)GlobeConst文件凡壤,里面專門定義全局變量耙替,統(tǒng)一管理俗扇,要不然項(xiàng)目文件多不好找混坞。
GlobeConst.h
/*******************************首頁****************************/
extern NSString * const nameKey = @"name";
/*******************************首頁****************************/
- GlobeConst.m
#import <Foundation/Foundation.h>
/*******************************首頁****************************/
NSString * const nameKey = @"name";
/*******************************首頁****************************/