52個有效方法(6) - 理解“屬性”這一概念

来源:https://www.cnblogs.com/shikaiming/archive/2019/10/15/11679172.html
-Advertisement-
Play Games

1. 可以用@property語法來定義對象中所封裝的數據。 2. 通過“修飾詞”來指定存儲數據所需的正確語義。 3. 在設置屬性所對應的實例變數時,一定要遵從該屬性所聲明的語義。 4. 開發iOS程式時應該使用nonatomic屬性,因為atomic(同步鎖)屬性嚴重影響性能。 ...


屬性

“屬性”(property)是OC的一項特性,用於封裝對象中的數據。

@property

  • @Property是聲明屬性的語法(@property = ivar + getter + setter)

    OC對象通常會把其所需的數據保存為各種實例變數(ivar)。實例變數一般通過“存取方法”(accessmethod)來訪問。

    什麼是存取方法:gettersetter 方法(access method = getter + setter),其中getter 用於獲取變數value, 而setter 用於寫入value

  • @Property可以快速方便的為實例變數創建存取器。

//  Man.h
#import <Foundation/Foundation.h>

@interface Man : NSObject
@property (nonatomic,strong)NSString *name;
@property (nonatomic,strong)NSString *sex;
@end

與下麵的寫法等效

//  Man.h
#import <Foundation/Foundation.h>

@interface Man : NSObject
{
    // 實例變數
    NSString *name;
    NSString *sex;
}
// setter
- (void)setName:(NSString *)newName;

// getter
- (NSString *)name;

// setter
- (void)setSex:(NSString *)newSex;

// getter
- (NSString *)sex;
@end
  • 通常使用“點語法” 來讓編譯器自動調用相關的存取方法(access method = getter + setter)。

    self. name = @"sky";
    NSString *name = self. name;

    點語法有什麼優勢呢?

    1. 省時,省力 :如果使用了屬性,編譯器會自動編寫訪問屬性所需的方法。這個過程由編譯器在編譯期執行,看不到這些get set 源代碼。

    2. 編譯器會自動向類中添加適當類型的實例變數,並且在屬性名前添加下劃線。

  • 如果你不想讓編譯器自動合成存取方法,則可以自己實現。如果你只實現了其中一個存取方法,那麼另一個還是會由編譯器來合成。

  • 當我們同時重寫了setter and getter方式時,系統會報錯,原因是找不到實例變數。其解決方法: 在.m的文件中使用@synthesize

@synthesize

  • @synthesize是為屬性添加一個實例變數名,或者說別名。同時會為該屬性生成 setter/getter 方法。

  • protocol中使用property只會生成settergetter方法聲明,我們使用屬性的目的,是希望遵守我協議的對象能實現該屬性。需要使用@synthesize生成settergetter

  • 當你在子類中重載了父類中的屬性,你必須 使用@synthesize來手動合成ivar

  • 當我們同時重寫了setter and getter方式時,需要在.m的文件中使用@synthesize

    //  Man.m
    #import "Man.h"
    
    @implementation Man
    @synthesize name = _name;
    // setter
    - (void)setName:(NSString *)name
    {
      _name = name;
    }
    
    // getter
    - (NSString *)name
    {
      return _name;
    }
    @end
    • **@synthesize name = _name**

      • _name是成員變數

      • name是屬性

      • 作用是告訴編譯器name屬性為_name實例變數生成setter and getter方法的實現

      • name屬性的setter方法是setName,它操作的是_name這個變數

      • @synthesize中定義與變數名不同的settergetter的命名,以此來保護變數不會被不恰當的訪問setter=<name>這種不常用,也不推薦使用)

          //setter=<name>這種不常用,也不推薦使用
        @property (nonatomic, setter = mySetter,getter = myGetter ) NSString *name;
        
        @property (nonatomic,getter = isHidden ) BOOL hidden;
  • @property有兩個對應的詞,一個是 @synthesize,一個是 @dynamic。如果 @synthesize@dynamic都沒寫,那麼預設的就是@syntheszie var = _var

  • 如果某屬性已經在某處實現了自己的 setter/getter ,可以使用 @dynamic來阻止 @synthesize 自動生成新的 setter/getter 覆蓋。

@dynamic

  • @dynamic告訴編譯器:屬性的 settergetter 方法由用戶自己實現,不自動生成。(當然對於 readonly 的屬性只需提供 getter 即可)。

  • 假如一個屬性被聲明為 @dynamic var,然後你沒有提供@setter方法和 @getter 方法。編譯的時候沒問題,但是當程式運行到 instance.var = someVar,由於缺 setter 方法會導致程式崩潰。或者當運行到 someVar = var時,由於缺 getter 方法同樣會導致崩潰。

  • 編譯時沒問題,運行時才執行相應的方法,這就是所謂的動態綁定。

//  Man.h
#import <Foundation/Foundation.h>

@interface Man : NSObject
@property (nonatomic,strong)NSString *name;
@end
//  Man.m
#import "Man.h"

@implementation Man
@dynamic name;
// setter
// - (void)setName:(NSString *)name
// {
//   _name = name;
// }
// getter
- (NSString *)name
{
  return _name;
}
@end

調用時會出現崩潰

    Man *man = [[Man alloc] init];
    man.name = @"sky";//缺 setter 方法會導致程式崩潰
    NSString *name = man.name;//缺 getter 方法同樣會導致崩潰

屬性特質

原子性

  • atomic(預設):atomic意為操作是原子的,意味著只有一個線程訪問實例變數(生成的settergetter方法是一個原子操作)。atomic是線程安全的,至少在當前的存取器上是安全的。它是一個預設的特性,但是很少使用,因為比較影響效率。

  • nonatomicnonatomic意為操作是非原子的,可以被多個線程訪問。它的效率比atomic快。但不能保證在多線程環境下的安全性,開發中常用。

  • 開發iOS程式時應該使用nonatomic屬性,因為atomic(同步鎖)屬性嚴重影響性能。該屬性使用了同步鎖,會在創建時生成一些額外的代碼用於幫助編寫多線程程式,這會帶來性能問題,通過聲明nonatomic可以節省這些雖然很小但是不必要額外開銷。

存取器控制

  • readwrite(預設):readwrite是預設值,表示該屬性同時擁有settergetter

  • readonlyreadonly表示只有getter沒有setter

  • 有時候為了語意更明確可能需要自定義訪問器的名字。

//setter=<name>這種不常用,也不推薦使用
@property (nonatomic, setter = mySetter,getter = myGetter ) NSString *name;
  

@property (nonatomic,getter = isHidden ) BOOL hidden;

記憶體管理

  • assign(預設):assign用於非指針變數(值)類型,統一由系統棧進行記憶體管理。一般用於基礎類型和C數據類型,如intfloatdoubleNSIntegerCGFloat等表示單純的複製。還包括不存在所有權關係的對象,比如常見的delegate

  • retain:在setter方法中,需要對傳入的對象進行引用計數加1的操作。

  • strongstrong是在IOS引入ARC的時候引入的關鍵字,是retain的一個可選的替代。對傳入的對象的強引用,會增加對象的引用計數。strongretain的意思相同並產生相同的代碼,但是語意上更好更能體現對象的關係。

  • weak:對傳入的對象的弱引用,不增加對象的引用計數,也不持有對象,當對象消失後指針自動指向nil

  • copy:與strong類似,但區別在於copy是創建一個新對象,strong是創建一個指針,引用對象計數加1

舉例說明weakstrongcopy屬性特質的差異

  • 首先創建兩個自定義的Person類的實例變數,並分別用weakstrong修飾。
@property (nonatomic,strong) Person *strongPerson;
@property (nonatomic,weak) Person *weakPerson;
  • strongPerson屬性置nil
self.strongPerson = [[Person alloc] init];
self.weakPerson = self.strongPerson;
self.strongPerson = nil; 
NSLog(@"strongStr=%@,weakStr=%@",self.strongPerson,self.weakPerson);

輸出結果為:strongStr=(null),weakStr=(null)。說明weak修飾的屬性並不會使引用計數增加。

  • 如果使用NSString類進行上述類似操作,得到的結果是不同的。
@property (nonatomic,strong) NSString *strongStr;
@property (nonatomic,weak) NSString *weakStr;
···
self.strongStr = @"string";
self.weakStr = self.strongStr;
self.strongStr = nil;
NSLog(@"strongStr=%@,weakStr=%@",self.strongStr,self.weakStr);

輸出結果為:strongStr=(null),weakStr=string。這裡主要是因為NSString類型的賦值預設會加上copy,而copy會創建一個新的對象。這裡的賦值語句其實是

self.strongStr = [@"string" copy];
self.weakStr = [self.strongStr copy];
  • weakPerson屬性置nil
self.strongPerson = [[Person alloc] init];
self.weakPerson = self.strongPerson;
self.weakPerson = nil;
NSLog(@"strongStr=%@,weakStr=%@",self.strongPerson,self.weakPerson);

輸出結果如下:strongStr=<Person: 0x600000007d50>,weakStr=(null)。說明weak修飾的屬性只是對對象的弱引用,並不會真正的持有該對象。

  • 新建一個Person類實例變數p,賦值strongPerson後將pnil
Person *p = [[Person alloc] init];
self.strongPerson = p;
self.weakPerson = self.strongPerson;
p = nil; 
NSLog(@"strongStr=%@,weakStr=%@",self.strongPerson,self.weakPerson);

輸出結果為:strongStr=<Person: 0x600000200b50>,weakStr=<Person: 0x600000200b50>。因為strong屬性會強引用該對象並使該對象的引用計數+1,所以即使把p設置為nil,該對象也並沒有釋放,要想釋放該對象,還得把strongStr設置為nil:self.strongPerson = nil;。這樣輸出結果才為 strongStr=(null),weakStr=(null)

  • 在給Person類加了一個name屬性。並用copy修飾 :(@property (nonatomic,copy) NSString *name)。
NSString *a = @"xiaoming";
Person *p = [[Person alloc] init];
p.name = a;
NSLog(@"before p.name=%@",p.name);
a = @"xiaohua";
NSLog(@"after p.name=%@",p.name);

輸出結果:before p.name=xiaomingafter p.name=xiaoming。因為copy關鍵字修飾的屬性是將對象拷貝一份賦值,所以你改變原對象並不會對拷貝後的對象有任何改變。

註:用@property聲明 NSStringNSArrayNSDictionary 經常使用copy關鍵字,是因為他們有對應的可變類型:NSMutableStringNSMutableArrayNSMutableDictionary,他們之間可能進行賦值操作,為確保對象中的字元串值不會無意間變動,應該在設置新屬性值時拷貝一份.

要點

  1. 可以用@property語法來定義對象中所封裝的數據。

  2. 通過“修飾詞”來指定存儲數據所需的正確語義。

  3. 在設置屬性所對應的實例變數時,一定要遵從該屬性所聲明的語義。

  4. 開發iOS程式時應該使用nonatomic屬性,因為atomic(同步鎖)屬性嚴重影響性能。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 今天在檢查oracle rac集群時,突然才發現伺服器的根目錄下麵占用了很多空間,照道理不應該出現這種情況,初步猜想可能是哪個日誌或跟蹤文件太大導致。切換到跟目錄,使用du -sh *來一層一層查看到底是哪個文件占用了這麼多空間,最後定位到目錄/u01/app/11.2.0/grid/crf/db/ ...
  • 創建資料庫 在MySQL中,使用 CREATE DATABASE 或 CREATE SCHEMA 語句創建資料庫 語法結構: : 表示為可選 : 用於分隔花括弧中的選項,表示任選一項語法 : 標識具體的資料庫命名,必須符合操作系統文件夾命名規則,在MySQL中不區分大小寫 : 預設值 : 指定資料庫 ...
  • 現需要限定特定的用戶只能查看並訪問特定的資料庫,防止多個用戶對資料庫操作時一些誤操作。 參考i6first的如何讓用戶只能訪問特定的資料庫(MSSQL)博文 1.新建登錄用戶 以管理員身份登陸資料庫(許可權最高的身份如sa),點擊安全性->登錄名,右鍵新建登錄名,輸入登錄名和密碼,取消強制實施密碼策略 ...
  • 1、去官網查找最新(你需要的)安裝包版本 # https://dev.mysql.com/downloads/repo/yum/ 2、下載MySQL安裝包 # wget http://dev.mysql.com/get/mysql80-community-release-el7-3.noarch.r ...
  • 概述 數據完整性指資料庫中數據的 正確性、相容性和一致性 。包括現實世界中的應用需求的完整性。數據的完整性由完整性規則來定義。 關係模型的完整性規則是對關係的某種約束,提供一種手段來保證用戶對資料庫的修改時不會破壞資料庫中數據的完整性。保證數據是有意義的。 關係模型分三類約束:實體完整性約束、參照完 ...
  • 在SQL Server中重建索引(Rebuild Index)與重組索引(Reorganize Index)會觸發統計信息更新嗎? 那麼我們先來測試、驗證一下: 我們以AdventureWorks2014為測試環境,如下所示: Person.Person表的統計信息最後一次更新為2014-07-17... ...
  • 本文基於 Android 9.0 , 代碼倉庫地址 : "android_9.0.0_r45" 文中源碼鏈接: "SystemServer.java" "ActivityManagerService.java" "Process.java" "ZygoteProcess.java" 對 和 啟動流程 ...
  • 1. 在對象內部讀取數據時,應該直接通過實例變數來讀,而寫入數據時,則應通過屬性來寫。 2. 在初始化方法及dealloc方法中,總是應該直接通過實例變數來讀寫數據。 3. 使用Lazy Initialization配置的數據,應該通過屬性來讀取數據。 4. 不要在setter/g... ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...