Optimizing work in iOS runtime

Boris Bugor
ITNEXT
Published in
4 min readOct 15, 2023
Photo by Bill Jelen on Unsplash

There are a sufficient number of ways to increase the speed of a project at runtime. Let’s look at the most popular of them:

Limiting dynamic dispatch in classes

Under the hood, any new class will use dynamic (table) dispatch. This is a price for the ability to be inherited, but if the class is not inherited, you can save a significant amount of resources. The final prefix will turn your class into a statically dispatched object.

Before optimization:

// parent class with dynamic table dispatch
class A {}
// child class with dynamic table dispatch
class B: A {}

After optimization:

// child class with static dispatch
final class B: A {}

Examples of speed boost:

  • Parent class A:
  • Subclass B:
  • Final class B:

Limitation of dynamic dispatch at methods

A similar approach is used for class methods, but there are several options available:

  • If your method is not used outside the class, feel free to use private.
  • If you are going to protect your method from the possibility of override, use final.

Both prefixes will turn your method into a statically dispatched method, but with different scopes.

Before optimization:

// method with dynamic table dispatch
func method() {}

After optimization:

// private method with static dispatch
private func method() {}

// method with static dispatch cannot be override
final func method() {}

Examples of speed boost:

  • Nonfinal menthod:
  • Final or private method

Inline optimization

One of the most curious and rarely used types of optimization. Inline optimization allows you to copy code calculations directly to the place where a given function is called without calling a method marked with inline, which reduces the number of method calls and works faster than static dispatch.

Among the pitfalls, in order to force the compiler to call a method with inline optimization — it is necessary to mark the method not only @inlineable but also @inline(__always)

It is important not to mark methods that spend a large amount of resources on copying with these prefixes; there is a risk of filling the cache of all processor levels with calculations of these methods and, as a result, getting slower instead of speeding up.

Before optimization:

// test method with multiple calling
func testMethod() {}

After optimization:

// inlinable test method with multiple calling
@inlinable
@inline(__always)
func testMethod() {}

Examples of speed boost:

  • regular calling:
  • inlined calling:

Optimization in protocols

Under the hood, protocols in Swift can be implemented for reference types and value types. Due to the fact that the memory management mechanism for reference types and value types are different, you can significantly speed up the compiler by limiting protocols adoption to reference types only.

Before optimization:

// can be adopted to reference types and value types,
// but value types adoption is unnecessarily
protocol Implementable {}

After optimization:

// can be adopted to reference types only
protocol Implementable: AnyObject {}

Examples of speed boost:

  • creation array elements with protocol for reference and value type
  • creation array elements with protocol for reference type only

Arrays

Array was originally intended not only as a collection type for Swift, allowing elements of the same type to be stored sequentially, but also as a collection type providing compatibility with NSArray.

In case the array is intended to store non-ObjC data types, it is recommended to use ContagiousArray for more efficient operation of the array.

Before optimization:


// creation array with NSArray interop
let array: Array<Int> = [1, 2, 3]

After optimization:

// creation array without NSArray interop
let fastArray: ContiguousArray<Int> = [1, 2, 3]

Examples of speed boost:

  • creation Array:
  • creation ContagiousArray:

Don’t hesitate to contact me on Twitter if you have any questions. Also, you can always buy me a coffee.

Published in ITNEXT

ITNEXT is a platform for IT developers & software engineers to share knowledge, connect, collaborate, learn and experience next-gen technologies.

Written by Boris Bugor

Indie iOS Developer | Entrepreneur | Dreamer

Responses (5)

What are your thoughts?

Interesting topics. One should note that time/space tradeoffs involved in forcing the compiler to inline functions, as doing so can balloon your executable code size.
It's a technique best used judiciously, and only after metrics have shown you actually have a bottleneck.

--

Nice article!
Can you elaborate how memory management in ref type/value type and protocol conformance related to each other?

--

Did u use instruments to calculate the metrics ?

--