Apple Mac Developer Library
In addition to the messaging behavior covered in the previous chapter, an object also encapsulates data through its properties.
This chapter describes the Objective-C syntax used to declare properties for an object and explains how those properties are implemented by default through synthesis of accessor methods and instance variables. If a property is backed by an instance variable, that variable must be set correctly in any initialization methods.
If an object needs to maintain a link to another object through a property, it’s important to consider the nature of the relationship between the two objects. Although memory management for Objective-C objects is mostly handled for you through Automatic Reference Counting (ARC), it’s important to know how to avoid problems like strong reference cycles, which lead to memory leaks. This chapter explains the lifecycle of an object, and describes how to think in terms of managing your graph of objects through relationships.
Properties Encapsulate an Object’s Values
The ultimate source for iPhone, iPad, Mac, and Apple Watch content. Never miss out on news, deals, reviews, or how-tos! Learn about the products your love, and never worry about buyer's remorse with our buying guides. All this and more on iMore. Time to upgrade! Get an iPhone 12 mini + Unlimited Data for $60/mo. The Mac version of the Apple Developer App will be similar to the iOS version. It will offer the discover section which will have new and relevant news for developers, it will also have a video section which includes the sessions, and the Apple Developer App will also have a separate WWDC section. The developers can access Apple’s Keynote.
Apple Books is the single destination for all the books you love, and the ones you’re about to. Browse the Book Store, join Oprah’s new Book Club, keep track of what you’ve read and want to read, and listen to enthralling audiobooks. 9to5Mac - Nearly two months after macOS Big Sur’s public release, the Apple human interface team has finally released an updated Sketch library optimized for its top to bottom redesign. User interface designers and developers have been anxiously awaiting this new design kit for quite awhile.
Most objects need to keep track of information in order to perform their tasks. Some objects are designed to model one or more values, such as a Cocoa NSNumber
class to hold a numeric value or a custom XYZPerson
class to model a person with a first and last name. Some objects are more general in scope, perhaps handling the interaction between a user interface and the information it displays, but even these objects need to keep track of user interface elements or the related model objects.
Declare Public Properties for Exposed Data
Mac For Developer
Objective-C properties offer a way to define the information that a class is intended to encapsulate. As you saw in Properties Control Access to an Object’s Values, property declarations are included in the interface for a class, like this:
In this example, the XYZPerson
class declares string properties to hold a person’s first and last name.
Given that one of the primary principles in object-oriented programming is that an object should hide its internal workings behind its public interface, it’s important to access an object’s properties using behavior exposed by the object rather than trying to gain access to the internal values directly.
Use Accessor Methods to Get or Set Property Values
You access or set an object’s properties via accessor methods:
By default, these accessor methods are synthesized automatically for you by the compiler, so you don’t need to do anything other than declare the property using @property
in the class interface.
The synthesized methods follow specific naming conventions:
The method used to access the value (the getter method) has the same name as the property.
The getter method for a property called
firstName
will also be calledfirstName
.The method used to set the value (the setter method) starts with the word “set” and then uses the capitalized property name.
The setter method for a property called
firstName
will be calledsetFirstName:
.
If you don’t want to allow a property to be changed via a setter method, you can add an attribute to a property declaration to specify that it should be readonly
:
As well as showing other objects how they are supposed to interact with the property, attributes also tell the compiler how to synthesize the relevant accessor methods.
In this case, the compiler will synthesize a fullName
getter method, but not a setFullName:
method.
Note: The opposite of readonly
is readwrite
. There’s no need to specify the readwrite
attribute explicitly, because it is the default.
If you want to use a different name for an accessor method, it’s possible to specify a custom name by adding attributes to the property. In the case of Boolean properties (properties that have a YES
or NO
value), it’s customary for the getter method to start with the word “is.” The getter method for a property called finished
, for example, should be called isFinished
.
Again, it’s possible to add an attribute on the property:
If you need to specify multiple attributes, simply include them as a comma-separated list, like this:
In this case, the compiler will synthesize only an isFinished
method, but not a setFinished:
method.
Note: In general, property accessor methods should be Key-Value Coding (KVC) compliant, which means that they follow explicit naming conventions.
See Key-Value Coding Programming Guide for more information.
Dot Syntax Is a Concise Alternative to Accessor Method Calls
As well as making explicit accessor method calls, Objective-C offers an alternative dot syntax to access an object’s properties.
Dot syntax allows you to access properties like this:
Dot syntax is purely a convenient wrapper around accessor method calls. When you use dot syntax, the property is still accessed or changed using the getter and setter methods mentioned above:
Getting a value using
somePerson.firstName
is the same as using[somePerson firstName]
Setting a value using
somePerson.firstName = @'Johnny'
is the same as using[somePerson setFirstName:@'Johnny']
This means that property access via dot syntax is also controlled by the property attributes. If a property is marked readonly
, you’ll get a compiler error if you try to set it using dot syntax.
Most Properties Are Backed by Instance Variables
By default, a readwrite
property will be backed by an instance variable, which will again be synthesized automatically by the compiler.
An instance variable is a variable that exists and holds its value for the life of the object. The memory used for instance variables is allocated when the object is first created (through alloc
), and freed when the object is deallocated.
Unless you specify otherwise, the synthesized instance variable has the same name as the property, but with an underscore prefix. For a property called firstName
, for example, the synthesized instance variable will be called _firstName
.
Although it’s best practice for an object to access its own properties using accessor methods or dot syntax, it’s possible to access the instance variable directly from any of the instance methods in a class implementation. The underscore prefix makes it clear that you’re accessing an instance variable rather than, for example, a local variable:
In this example, it’s clear that myString
is a local variable and _someString
is an instance variable.
In general, you should use accessor methods or dot syntax for property access even if you’re accessing an object’s properties from within its own implementation, in which case you should use self
:
The exception to this rule is when writing initialization, deallocation or custom accessor methods, as described later in this section.
You Can Customize Synthesized Instance Variable Names
As mentioned earlier, the default behavior for a writeable property is to use an instance variable called _propertyName
.
If you wish to use a different name for the instance variable, you need to direct the compiler to synthesize the variable using the following syntax in your implementation:
For example:
In this case, the property will still be called firstName
, and be accessible through firstName
and setFirstName:
accessor methods or dot syntax, but it will be backed by an instance variable called ivar_firstName
.
Important: If you use @synthesize
without specifying an instance variable name, like this:
In this example, the instance variable will also be called firstName
, without an underscore.
You Can Define Instance Variables without Properties
It’s best practice to use a property on an object any time you need to keep track of a value or another object.
If you do need to define your own instance variables without declaring a property, you can add them inside braces at the top of the class interface or implementation, like this:
Note: You can also add instance variables at the top of a class extension, as described in Class Extensions Extend the Internal Implementation.
Access Instance Variables Directly from Initializer Methods
Setter methods can have additional side-effects. They may trigger KVC notifications, or perform further tasks if you write your own custom methods.
You should always access the instance variables directly from within an initialization method because at the time a property is set, the rest of the object may not yet be completely initialized. Even if you don’t provide custom accessor methods or know of any side effects from within your own class, a future subclass may very well override the behavior.
A typical init
method looks like this:
An init
method should assign self
to the result of calling the superclass’s initialization method before doing its own initialization. A superclass may fail to initialize the object correctly and return nil
so you should always check to make sure self
is not nil
before performing your own initialization.
By calling [super init]
as the first line in the method, an object is initialized from its root class down through each subclass init
implementation in order. Figure 3-1 shows the process for initializing an XYZShoutingPerson
object.
As you saw in the previous chapter, an object is initialized either by calling init
, or by calling a method that initializes the object with specific values.
In the case of the XYZPerson
class, it would make sense to offer an initialization method that set the person’s initial first and last names:
You’d implement the method like this:
The Designated Initializer is the Primary Initialization Method
If an object declares one or more initialization methods, you should decide which method is the designated initializer. This is often the method that offers the most options for initialization (such as the method with the most arguments), and is called by other methods you write for convenience. You should also typically override init
to call your designated initializer with suitable default values.
If an XYZPerson
also had a property for a date of birth, the designated initializer might be:
This method would set the relevant instance variables, as shown above. If you still wished to provide a convenience initializer for just first and last names, you would implement the method to call the designated initializer, like this:
You might also implement a standard init
method to provide suitable defaults:
If you need to write an initialization method when subclassing a class that uses multiple init
methods, you should either override the superclass’s designated initializer to perform your own initialization, or add your own additional initializer. Either way, you should call the superclass’s designated initializer (in place of [super init];
) before doing any of your own initialization.
You Can Implement Custom Accessor Methods
Properties don’t always have to be backed by their own instance variables.
As an example, the XYZPerson
class might define a read-only property for a person’s full name:
Rather than having to update the fullName
property every time the first or last name changed, it would be easier just to write a custom accessor method to build the full name string on request:
This simple example uses a format string and specifiers (as described in the previous chapter) to build a string containing a person’s first and last names separated by a space.
Note: Although this is a convenient example, it’s important to realize that it’s locale-specific, and is only suitable for use in countries that put a person’s given name before the family name.
If you need to write a custom accessor method for a property that does use an instance variable, you must access that instance variable directly from within the method. For example, it’s common to delay the initialization of a property until it’s first requested, using a “lazy accessor,” like this:
Before returning the value, this method first checks whether the _someImportantObject
instance variable is nil
; if it is, it allocates an object.
Note: The compiler will automatically synthesize an instance variable in all situations where it’s also synthesizing at least one accessor method. If you implement both a getter and a setter for a readwrite
property, or a getter for a readonly
property, the compiler will assume that you are taking control over the property implementation and won’t synthesize an instance variable automatically.
If you still need an instance variable, you’ll need to request that one be synthesized:
Properties Are Atomic by Default
By default, an Objective-C property is atomic:
This means that the synthesized accessors ensure that a value is always fully retrieved by the getter method or fully set via the setter method, even if the accessors are called simultaneously from different threads.
Because the internal implementation and synchronization of atomic accessor methods is private, it’s not possible to combine a synthesized accessor with an accessor method that you implement yourself. You’ll get a compiler warning if you try, for example, to provide a custom setter for an atomic
, readwrite
property but leave the compiler to synthesize the getter.
You can use the nonatomic
property attribute to specify that synthesized accessors simply set or return a value directly, with no guarantees about what happens if that same value is accessed simultaneously from different threads. For this reason, it’s faster to access a nonatomic
property than an atomic
one, and it’s fine to combine a synthesized setter, for example, with your own getter implementation:
Note: Property atomicity is not synonymous with an object’s thread safety.
Consider an XYZPerson
object in which both a person’s first and last names are changed using atomic accessors from one thread. If another thread accesses both names at the same time, the atomic getter methods will return complete strings (without crashing), but there’s no guarantee that those values will be the right names relative to each other. If the first name is accessed before the change, but the last name is accessed after the change, you’ll end up with an inconsistent, mismatched pair of names.
This example is quite simple, but the problem of thread safety becomes much more complex when considered across a network of related objects. Thread safety is covered in more detail in Concurrency Programming Guide.
Manage the Object Graph through Ownership and Responsibility
As you’ve already seen, memory for Objective-C objects is allocated dynamically (on the heap), which means you need to use pointers to keep track of an object’s address. Unlike scalar values, it’s not always possible to determine an object’s lifetime by the scope of one pointer variable. Instead, an object must be kept active in memory for as long as it is needed by other objects.
Rather than trying to worry about managing the lifecycle of each object manually, you should instead think about the relationships between objects.
In the case of an XYZPerson
object, for example, the two string properties for firstName
and lastName
are effectively “owned” by the XYZPerson
instance. This means they should stay in memory as long as the XYZPerson
object stays in memory.
When one object relies on other objects in this way, effectively taking ownership of those other objects, the first object is said to have strong references to the other objects. In Objective-C, an object is kept alive as long as it has at least one strong reference to it from another object. The relationships between the XYZPerson
instance and the two NSString
objects is shown in Figure 3-2.
When an XYZPerson
object is deallocated from memory, the two string objects will also be deallocated, assuming there aren’t any other strong references left to them.
To add a little more complexity to this example, consider the object graph for an application like that shown in Figure 3-3.
When the user clicks the Update button, the badge preview is updated with the relevant name information.
The first time a person’s details are entered and the update button clicked, the simplified object graph might look like Figure 3-4.
When the user modifies the person’s first name, the object graph changes to look like Figure 3-5.
The badge display view maintains a strong relationship to the original @'John'
string object, even though the XYZPerson
object now has a different firstName
. This means the @'John'
object stays in memory, used by the badge view to print the name.
Once the user clicks the Update button a second time, the badge view is told to update its internal properties to match the person object, so the object graph looks like Figure 3-6.
At this point, the original @'John'
object no longer has any strong references to it, so it is removed from memory.
By default, both Objective-C properties and variables maintain strong references to their objects. This is fine for many situations, but it does cause a potential problem with strong reference cycles.
Avoid Strong Reference Cycles
Although strong references work well for one-way relationships between objects, you need to be careful when working with groups of interconnected objects. If a group of objects is connected by a circle of strong relationships, they keep each other alive even if there are no strong references from outside the group.
One obvious example of a potential reference cycle exists between a table view object (UITableView
for iOS and NSTableView
for OS X) and its delegate. In order for a generic table view class to be useful in multiple situations, it delegates some decisions to external objects. This means it relies on another object to decide what content it displays, or what to do if the user interacts with a specific entry in the table view.
A common scenario is that the table view has a reference to its delegate and the delegate has a reference back to the table view, as shown in Figure 3-7.
A problem occurs if the other objects give up their strong relationships to the table view and delegate, as shown in Figure 3-8.
Even though there is no need for the objects to be kept in memory—there are no strong relationships to the table view or delegate other than the relationships between the two objects—the two remaining strong relationships keep the two objects alive. This is known as a strong reference cycle.
The way to solve this problem is to substitute one of the strong references for a weak reference. A weak reference does not imply ownership or responsibility between two objects, and does not keep an object alive.
If the table view is modified to use a weak relationship to its delegate (which is how UITableView
and NSTableView
solve this problem), the initial object graph now looks like Figure 3-9.
When the other objects in the graph give up their strong relationships to the table view and delegate this time, there are no strong references left to the delegate object, as shown in Figure 3-10.
This means that the delegate object will be deallocated, thereby releasing the strong reference on the table view, as shown in Figure 3-11.
Once the delegate is deallocated, there are no longer any strong references to the table view, so it too is deallocated.
Use Strong and Weak Declarations to Manage Ownership
By default, object properties declared like this:
use strong references for their synthesized instance variables. To declare a weak reference, add an attribute to the property, like this:
Note: The opposite to weak
is strong
. There’s no need to specify the strong
attribute explicitly, because it is the default.
Local variables (and non-property instance variables) also maintain strong references to objects by default. This means that the following code will work exactly as you expect:
In this example, the local variable originalDate
maintains a strong reference to the initial lastModificationDate
object. When the lastModificationDate
property is changed, the property no longer keeps a strong reference to the original date, but that date is still kept alive by the originalDate
strong variable.
Note: A variable maintains a strong reference to an object only as long as that variable is in scope, or until it is reassigned to another object or nil
.
If you don’t want a variable to maintain a strong reference, you can declare it as __weak
, like this:
Because a weak reference doesn’t keep an object alive, it’s possible for the referenced object to be deallocated while the reference is still in use. To avoid a dangerous dangling pointer to the memory originally occupied by the now deallocated object, a weak reference is automatically set to nil
when its object is deallocated.
This means that if you use a weak variable in the previous date example:
the originalDate
variable may potentially be set to nil
. When self.lastModificationDate
is reassigned, the property no longer maintains a strong reference to the original date. If there are no other strong references to it, the original date will be deallocated and originalDate
set to nil
.
Weak variables can be a source of confusion, particularly in code like this:
In this example, the newly allocated object has no strong references to it, so it is immediately deallocated and someObject
is set to nil
.
Note: The opposite to __weak
is __strong
. Again, you don’t need to specify __strong
explicitly, because it is the default.
It’s also important to consider the implications of a method that needs to access a weak property several times, like this:
In situations like this, you might want to cache the weak property in a strong variable to ensure that it is kept in memory as long as you need to use it:
In this example, the cachedObject
variable maintains a strong reference to the original weak property value so that it can’t be deallocated as long as cachedObject
is still in scope (and hasn’t been reassigned another value).
Apple App Developer
It’s particularly important to keep this in mind if you need to make sure a weak property is not nil
before using it. It’s not enough just to test it, like this:
because in a multi-threaded application, the property may be deallocated between the test and the method call, rendering the test useless. Instead, you need to declare a strong local variable to cache the value, like this:
In this example, the strong reference is created in line 1, meaning that the object is guaranteed to be alive for the test and method call. In line 5, cachedObject
is set to nil
, thereby giving up the strong reference. If the original object has no other strong references to it at this point, it will be deallocated and someWeakProperty
will be set to nil
.
Use Unsafe Unretained References for Some Classes
There are a few classes in Cocoa and Cocoa Touch that don’t yet support weak references, which means you can’t declare a weak property or weak local variable to keep track of them. These classes include NSTextView
, NSFont
and NSColorSpace
; for the full list, see Transitioning to ARC Release Notes.
If you need to use a weak reference to one of these classes, you must use an unsafe reference. For a property, this means using the unsafe_unretained
attribute:
For variables, you need to use __unsafe_unretained
:
An unsafe reference is similar to a weak reference in that it doesn’t keep its related object alive, but it won’t be set to nil
if the destination object is deallocated. This means that you’ll be left with a dangling pointer to the memory originally occupied by the now deallocated object, hence the term “unsafe.” Sending a message to a dangling pointer will result in a crash.
Copy Properties Maintain Their Own Copies
In some circumstances, an object may wish to keep its own copy of any objects that are set for its properties.
As an example, the class interface for the XYZBadgeView
class shown earlier in Figure 3-4 might look like this:
Two NSString
properties are declared, which both maintain implicit strong references to their objects.
Consider what happens if another object creates a string to set as one of the badge view’s properties, like this:
This is perfectly valid, because NSMutableString
is a subclass of NSString
. Although the badge view thinks it’s dealing with an NSString
instance, it’s actually dealing with an NSMutableString
.
This means that the string can change:
In this case, although the name was “John” at the time it was originally set for the badge view’s firstName
property, it’s now “Johnny” because the mutable string was changed.
You might choose that the badge view should maintain its own copies of any strings set for its firstName
and lastName
properties, so that it effectively captures the strings at the time that the properties are set. By adding a copy
attribute to the two property declarations:
the view now maintains its own copies of the two strings. Even if a mutable string is set and subsequently changed, the badge view captures whatever value it has at the time it is set. For example:
This time, the firstName
held by the badge view will be an unaffected copy of the original “John” string.
The copy
attribute means that the property will use a strong reference, because it must hold on to the new object it creates.
Apple Developer Mac Mini
Note: Any object that you wish to set for a copy
property must support NSCopying
, which means that it should conform to the NSCopying
protocol.
Protocols are described in Protocols Define Messaging Contracts. For more information on NSCopying
, see NSCopying
or the Advanced Memory Management Programming Guide.
If you need to set a copy
property’s instance variable directly, for example in an initializer method, don’t forget to set a copy of the original object:
Exercises
Modify the
sayHello
method from theXYZPerson
class to log a greeting using the person’s first name and last name.Declare and implement a new designated initializer used to create an
XYZPerson
using a specified first name, last name and date of birth, along with a suitable class factory method.Don’t forget to override
init
to call the designated initializer.Test what happens if you set a mutable string as the person’s first name, then mutate that string before calling your modified
sayHello
method. Change theNSString
property declarations by adding thecopy
attribute and test again.Try creating
XYZPerson
objects using a variety of strong and weak variables in themain()
function. Verify that the strong variables keep theXYZPerson
objects alive at least as long as you expect.In order to help verify when an
XYZPerson
object is deallocated, you might want to tie into the object lifecycle by providing adealloc
method in theXYZPerson
implementation. This method is called automatically when an Objective-C object is deallocated from memory, and is normally used to release any memory you allocated manually, such as through the Cmalloc()
function, as described in Advanced Memory Management Programming Guide.For the purposes of this exercise, override the
dealloc
method inXYZPerson
to log a message, like this:Try setting each
XYZPerson
pointer variable tonil
to verify that the objects are deallocated when you expect them to be.Note: The Xcode project template for a Command Line Tool use an
@autoreleasepool { }
block inside themain()
function. In order to use the Automatic Retain Count feature of the compiler to handle memory management for you, it’s important that any code you write inmain()
goes inside this autorelease pool block.Autorelease pools are outside the scope of this document, but are covered in detail in Advanced Memory Management Programming Guide.
When you’re writing a Cocoa or Cocoa Touch application rather than a command line tool, you won’t usually need to worry about creating your own autorelease pools, because you’re tying into a framework of objects that will ensure one is already in place.
Modify the
XYZPerson
class description so that you can keep track of a spouse or partner.You’ll need to decide how best to model the relationship, thinking carefully about object graph management.
Copyright © 2014 Apple Inc. All Rights Reserved. Terms of Use | Privacy Policy | Updated: 2014-09-17
Welcome to the future of Mac.
Submit your next generation
app to the App Store.
App Store
Explore the features and comprehensive benefits of distributing your apps on the App Store for iPhone, iPad, Mac, Apple TV, and Apple Watch.
Introducing Xcode 12
App Clips
Provide a new way to quickly access and experience what your app has to offer. An app clip is a small part of your app that lets users start and finish an experience in seconds, even before downloading your app.
Widgets
Now it’s even easier to build and make widgets available across iOS, iPadOS, and macOS with the new WidgetKit framework, widget API for SwiftUI, widget gallery, and Smart Stacks.
macOS Big Sur
A whole new level of power and beauty.
watchOS 7
Develop even more powerful and personal apps for Apple Watch.
iOS 14
Take advantage of the latest features in the world’s most advanced mobile operating system.
iPadOS 14
New features and enhancements in iPadOS 14 empower you to create amazing experiences.