iOS9關(guān)鍵字、泛型、__kindof和const、static、extern

nullable關(guān)鍵字(iOS9關(guān)鍵字)

  • 關(guān)鍵字:可以用于屬性,方法返回值和參數(shù)中
    關(guān)鍵字作用:提示作用,告訴開(kāi)發(fā)者屬性信息
    關(guān)鍵字目的:迎合swift,swift是個(gè)強(qiáng)語(yǔ)言,swift必須要指定一個(gè)對(duì)象是否為空
    關(guān)鍵字好處:提高代碼規(guī)劃,減少溝通成本
    關(guān)鍵字僅僅是提供警告,并不會(huì)報(bào)編譯錯(cuò)誤
    null_resettable

  • nullable作用:可能為空

nullable 語(yǔ)法1:@property (nonatomic, strong, nullable) NSString *name;
nullable 語(yǔ)法2 * 關(guān)鍵字 變量名: @property (nonatomic, strong) NSString * _Nullable name;
nullable 語(yǔ)法3 : @property (nonatomic, strong) NSString * __nullable name;

nonnull關(guān)鍵字(iOS9關(guān)鍵字)

  • nonnull作用:不能為空

nonnull 語(yǔ)法1 : @property (nonatomic, strong, nullable) NSString *name;
nonnull 語(yǔ)法2 * 關(guān)鍵字 變量名 : @property (nonatomic, strong) NSString * _Nonnull name;
nonnull 語(yǔ)法3 : @property (nonatomic, strong) NSString * __nonnull name;

null_resettable關(guān)鍵字(iOS9關(guān)鍵字)

null_resettable作用:get方法不能返回nil,set可以傳入為空,必須要處理為空情況,重寫get方法

null_resettable 語(yǔ)法(只有這一種語(yǔ)法) : @property (nonatomic, strong, null_resettable) NSString *name;

_Null_unspecified關(guān)鍵字(iOS9關(guān)鍵字)

  • _Null_unspecified作用 :不確定是否為空

_Null_unspecified語(yǔ)法:@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ù)組中取出來(lái),可以使用點(diǎn)語(yǔ)法
  • 2.給數(shù)組添加元素,有提示

*泛型在開(kāi)發(fā)中使用場(chǎng)景:1.用于限制集合類型

  • 主要:id是不能使用點(diǎn)語(yǔ)法

  • 自定義泛型

  • 什么時(shí)候使用泛型?在聲明類的時(shí)候,不確定某些屬性或者方法類型,在使用這個(gè)類的時(shí)候才確定,就可以采用泛型
  • 自定義Person,會(huì)一些編程語(yǔ)言(iOS,Java),在聲明Person,不確定這個(gè)人會(huì)什么,在使用Person才知道這個(gè)Person會(huì)什么語(yǔ)言
  • 如果沒(méi)有定義泛型.默認(rèn)就是id
@interface Language : NSObject

@end

@interface Java : Language

@end

@interface iOS : Language

@end

@interface Person<ObjectType> : NSObject
/**語(yǔ)言*/
@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
/**語(yǔ)言*/
@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
/**語(yǔ)言*/
@property (nonatomic , strong) ObjectType yuyan;

@end

泛型注意點(diǎn):在數(shù)組中,一般用可變數(shù)組添加方法,泛型才會(huì)生效,如果使用不可變數(shù)組,添加元素,泛型沒(méi)有效果。


__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ì)象,但是沒(méi)辦法提示返回值類型
  • (Person *):給返回值賦值為SubPerson *會(huì)報(bào)警告-->Incompatible pointer types initializing 'SubPerson *' with an expression of type 'Person *'
  • (__kindof Person *):表示返回值是Person *或者Person的子類

const,static,extern簡(jiǎn)介

一、const與宏的區(qū)別(面試題):

  • const簡(jiǎn)介:之前常用的字符串常量,一般是抽成宏,但是蘋果不推薦我們抽成宏,推薦我們使用const常量。
  • 執(zhí)行時(shí)刻:宏是預(yù)編譯(編譯之前處理),const是編譯階段。
  • 編譯檢查:宏不做檢查,不會(huì)報(bào)編譯錯(cuò)誤,只是替換,const會(huì)編譯檢查,會(huì)報(bào)編譯錯(cuò)誤。
  • 宏的好處:宏能定義一些函數(shù),方法。 const不能。
  • 宏的壞處:使用大量宏,容易造成編譯時(shí)間久,每次都需要重新替換。

注意:很多Blog都說(shuō)使用宏,會(huì)消耗很多內(nèi)存,我這驗(yàn)證并不會(huì)生成很多內(nèi)存,宏定義的是常量,常量都放在常量區(qū),只會(huì)生成一份內(nèi)存。

[圖片上傳中。。。(1)]

// 常見(jiàn)的常量:抽成宏
#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僅僅用來(lái)修飾右邊的變量(基本數(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訪問(wèn)內(nèi)存空間的值
    *p = 20;
    
    // const修飾指針變量訪問(wèn)的內(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開(kāi)發(fā)中使用場(chǎng)景:

  • 1.需求1:提供一個(gè)方法,這個(gè)方法的參數(shù)是地址,里面只能通過(guò)地址讀取值,不能通過(guò)地址修改值
  • 2.需求2:提供一個(gè)方法,這個(gè)方法的參數(shù)是地址,里面不能修改參數(shù)的地址。
@implementation ViewController

// const放*前面約束參數(shù),表示*a只讀
// 只能修改地址a,不能通過(guò)a修改訪問(wèn)的內(nèi)存空間
- (void)test:(const int * )a
{
//    *a = 20;
}

// const放*后面約束參數(shù),表示a只讀
// 不能修改a的地址,只能修改a訪問(wèn)的值
- (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ù)是地址,里面只能通過(guò)地址讀取值,不能通過(guò)地址修改值。
    
    // 這時(shí)候就需要使用const,約束方法的參數(shù)只讀.
    [self test:&a];
    
    // 需求2:提供一個(gè)方法,這個(gè)方法的參數(shù)是地址,里面不能修改參數(shù)的地址。
    [self test1:&a];
}

@end

四、static和extern簡(jiǎn)單使用(要使用一個(gè)東西,先了解其作用)

  • static作用:

    • 修飾局部變量:

    1.延長(zhǎng)局部變量的生命周期,程序結(jié)束才會(huì)銷毀。

2.局部變量只會(huì)生成一份內(nèi)存,只會(huì)初始化一次。

3.改變局部變量的作用域。

  • 修飾全局變量

    1.只能在本文件中訪問(wèn),修改全局變量的作用域,生命周期不會(huì)改

2.避免重復(fù)定義全局變量

  • extern作用:
  • 只是用來(lái)獲取全局變量(包括全局靜態(tài)變量)的值,不能用于定義變量
  • extern工作原理:
  • 先在當(dāng)前文件查找有沒(méi)有全局變量,沒(méi)有找到,才會(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)變量
  • 開(kāi)發(fā)使用場(chǎng)景:在一個(gè)文件中經(jīng)常使用的字符串常量,可以使用static與const組合
// 開(kāi)發(fā)中常用static修飾全局變量,只改變作用域

// 為什么要改變?nèi)肿兞孔饔糜?,防止重?fù)聲明全局變量。

// 開(kāi)發(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常用使用場(chǎng)景,是用來(lái)代替宏,把一個(gè)經(jīng)常使用的字符串常量,定義成靜態(tài)全局只讀變量.

// 開(kā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)合使用

  • 開(kāi)發(fā)中使用場(chǎng)景:在多個(gè)文件中經(jīng)常使用的同一個(gè)字符串常量,可以使用extern與const組合。

  • 原因:

  • static與const組合:在每個(gè)文件都需要定義一份靜態(tài)全局變量。

  • extern與const組合:只需要定義一份全局變量,多個(gè)文件共享。

  • 全局常量正規(guī)寫法:開(kāi)發(fā)中便于管理所有的全局變量,通常搞一個(gè)GlobeConst文件,里面專門定義全局變量,統(tǒng)一管理,要不然項(xiàng)目文件多不好找。

  • GlobeConst.h

/*******************************首頁(yè)****************************/

extern NSString * const nameKey = @"name";

/*******************************首頁(yè)****************************/
  • GlobeConst.m
#import <Foundation/Foundation.h>

/*******************************首頁(yè)****************************/

NSString * const nameKey = @"name";

/*******************************首頁(yè)****************************/

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容