Skip to content
This repository has been archived by the owner on Oct 1, 2022. It is now read-only.

Latest commit

 

History

History
775 lines (563 loc) · 42.2 KB

Generics.md

File metadata and controls

775 lines (563 loc) · 42.2 KB

Generics

Generic code permite que você possa escrever funções e tipos, flexíveis e reusáveis, que possam funcionar com qualquer tipo, sujeito aos requisitos que você definir. Você pode escrever código que evita a duplicação e expressa o seu intuito de uma maneira clara e abstrata.

Generics é um dos recursos mais poderosos do Swift, e grande parte da biblioteca padrão do Swift é criado com generic code. Na verdade, você tem usado generics através do Language Guide, mesmo que você não tenha percebido. Por exemplo, os tipos Array e Dictionary do Swift são coleções generics. Você pode criar um array que contém valores Int, ou um array que contém valores String, ou mesmo um array para qualquer outro tipo que possa ser criado em Swift. Da mesma forma, você pode criar um dictionary para armazenar valores de qualquer tipo específico, e não há nenhuma limitação de qual aquele tipo possa ser.

O Problema Que Generics Resolve

Aqui há uma função padrão não-genérica chamada swapTwoInts(_:_:), que troca dois valores Int:

func swapTwoInts(_ a: inout Int, _ b: inout Int) {
  let temporaryA = a
  a = b
  b = temporaryA
}

Essa função faz uso de parâmetros in-out para trocar os valores de a e b, como descritos em In-Out Parameters.

A funcão swapTwoInts(_:_:) troca o valor original de b pelo a, e o valor original de a em b. Você pode chamar essa função para trocar os valores de duas variáveis Int:

var someInt = 3
var anotherInt = 107
swapTwoInts(&someInt, &anotherInt)
print("someInt é agora \(someInt), e anotherInt é agora \(anotherInt)")
// Prints "someInt é agora 107, e anotherInt é agora 3"

A função swapTwoInts(_:_:) é útil, mas pode ser apenas usado com valores Int. Se você quer trocar dois valores String, ou dois valores Double, você tem que escrever mais funções, assim como as funções swapTwoStrings(_:_:) e swapTwoDoubles(_:_:) mostradas abaixo:

func swapTwoStrings(_ a: inout String, _ b: inout String) {
  let temporaryA = a
  a = b
  b = temporaryA
}

func swapTwoDoubles(_ a: inout Double, _ b: inout Double) {
  let temporaryA = a
  a = b
  b = temporaryA
}

Você pode ter notado que os corpos das funções swapTwoInts(_:_:), swapTwoStrings(_:_:), e swapTwoDoubles(_:_:) são idênticas. A única diferença é o tipo de valores que elas aceitam (Int, String, e Double).

É mais útil e consideravelmente mais flexível, escrever uma única função que troca dois valores de qualquer tipo. Código generic permite que você escreva tal função.(Uma versão generic de tais funções é definida abaixo.)

Nota

Em todas as três funções, os tipos de a e b tem que ser o mesmo. Se a e b não são do mesmo tipo, não é possível trocar os seus valores. Swift é uma linguagem fortemente tipada, e não permite(por exemplo) uma variável do tipo String e uma variável do tipo Double trocar valores entre si. Tentando fazer isso resulta erro em tempo de compilação.

Funções Generic

Funções Generic podem ser usadas com qualquer tipo. Aqui há uma versão generic da função swapTwoInts(_:_:) de cima, chamada swapTwoValues(_:_:):

func swapTwoValues<T>(_ a: inout T, _ b: inout T) {
  let temporaryA = a
  a = b
  b = temporaryA
}

O corpo da função swapTwoValues(_:_:) é idêntica ao corpo da função swapTwoInts(_:_:). No entanto, a primeira linha da swapTwoValues(_:_:) é significantemente diferente da swapTwoInts(_:_:). Veja como as primeiras linhas se comparam:

func swapTwoInts(_ a: inout Int, _ b: inout Int)
func swapTwoValues<T>(_ a: inout T, _ b: inout T)

A versão generic da função que usa um nome do tipo placeholder (chamado de T, nesse caso) ao invés de um tipo nome real (como Int, String, ou Double). O nome do tipo placeholder não diz nada sobre o que T deve ser, mas diz que a e b devem ser do mesmo tipo que T, o quer que seja T. O tipo real que vai ser usado no lugar de T é determinado cada vez que a função swapTwoValues(_:_:) é chamada.

A outra diferença entre uma função generic e uma função não-generic é que o nome da função generic (swapTwoValues(_:_:)) é seguida de um nome do tipo placeholder(T) dentro dos angle brackets conhecido também como operador diamante,(<T>). O operador diamante diz para Swift que T é um nome do tipo placeholder dentro da definição da função swapTwoValues(_:_:). Como T é um placeholder, Swift não olha para um tipo real chamado T.

A função swapTwoValues(_:_:) agora pode ser chamada do mesmo jeito que swapTwoInts, exceto que pode ser passado dois valores de qualquer tipo, desde que esses dois valores sejam do mesmo tipo que o outro. Cada vez que swapTwoValues(_:_:) é chamada, o tipo que é usado para T é inferido dos tipos de valores passados para a função.

Nos dois exemplos abaixo, T é inferido ser Int e String, respectivamente:

var someInt = 3
var anotherInt = 107
swapTwoValues(&someInt, &anotherInt)
// someInt é agora 107, e anotherInt é agora 3

var someString = "hello"
var anotherString = "world"
swapTwoValues(&someString, &anotherString)
// someString é agora "world", e anotherString é agora "hello"

Nota

A função swapTwoValues(_:_:) definida acima é inspirada pela função generic chamada swap, que faz parte da biblioteca padrão do Swift, e é disponibilizada automaticamente para que você possa usar nos seus apps. Se você precisa do comportamento da função swapTwoValues(_:_:) no seu código, você pode usar a função swap(_:_:) já existente em Swift ao invés de fornecer a sua própria implementação

Type Parameters

No exemplo swapTwoValues(_:_:) acima, o tipo placeholder T é um exemplo de type parameter. Type Parameters especificam e nomeiam um tipo placeholder, e são escritos imediatamente após o nome da função genérica, entre angle brackets (como em ).

Quando você especificar um type parameter, você pode usar isso para definir o tipo dos parametros de uma função (como os parametros a e b no exemplo anterior), ou como o tipo de retorno da função, como a notação de tipo no corpo da função. Em cada caso, o type parameter é substituído pelo tipo de fato que é inferido quando a função quando é chamada. (No exemplo swapTwoValues (_: _ :) acima, T foi substituído por Int na primeira vez que a função foi chamada e foi substituído por String na segunda vez que foi chamada.)

Você pode provê mais de um type parameter escrevendo múltiplos nomes como type parameter dentro de angle brackets, separado por vírgulas. (<T, U, V, X>)

Nomeando Type Parameters

Na maioria dos casos, type parameters tem nomes descritivos, como Key e Value no Dictionary<Key, Value> e Element no Array<Element>, que nos comunica qual a relação entre os type parameters e a função genérica em que eles são usados. Entretanto, quando não é uma relação significativa entre eles, tradicionalmente eles são nomeados usando uma única letra como T, U, e V, como T na função swapTwoValues (_: _ :) acima.

Nota

Sempre nomeie type parameters com CamelCase (começando com letra maiúscula T e MyTypeParameter) para indicar que eles são um placeholder para um tipo, e não um valor.

Tipos Genéricos

Em adição a funções genéricas, em Swift você pode definir seus próprios generic types, ou tipos genéricos. Tipos genéricos são classes, structs e enums que podem trabalhar com qualquer tipo, de maneira similar a Array e Dictionary.

Essa sessão irá mostrar como escrever uma coleção genérica chamada Stack, ou Pilha. Uma Pilha é conjunto ordenado de valores, parecido com um array, mas com operações mais restritas. Um array permite que novos items sejam inseridos e removidos em qualquer lugar no array. Uma Pilha, entretanto, só permite que novos itens sejam adicionados após o último elemento, ou seja, no fim da coleção (essa operação é conhecida como push). Similarmente, uma Pilha só permite que itens sejam removidos no fim da coleção, ou seja, só é possível remover o último item da coleção (essa operação é conhecida como pop).

Nota

O conceito de uma pilha é usada pela classe UINavigationController para modelar as view controllers na sua hierarquia de navegação. Você pode chamar o método pushViewController(_:animated:) da classe UINavigationController para adicionar (push) uma view controller na pilha da navigationController, ou você pode chamar o método popViewControllerAnimated(_ :) para remover (pop) uma view controller da pilha de navegação da navigationController. Uma pilha é um modelo de coleção útil sempre que você precisa de um comportamento estritamente last in, first out (último a entrar, primeiro a sair) para gerenciá-la.

A ilustração abaixo mostra o comportamento de push e pop de uma pilha:

../_images/stackPushPop_2x.png

  1. Existem inicialmente 3 valores na pilha.

  2. Um quarto value é inserido no topo da pilha. (push)

  3. A pilha agora tem 4 valores, com o mais recente no topo.

  4. O valor que está no topo é removido. (pop)

  5. Depois de remover esse valor, a pilha tem os 3 valores originais novamente.

Aqui está uma implementação não genérica de uma pilha, nesse caso uma pilha que tem valores do tipo Int:

struct IntStack {
  var items = [Int]()
  mutating func push(_ item: Int) {
    items.append(item)
  }
  mutating func pop() -> Int {
    return items.removeLast()
  }
}

Essa estrutura usa uma propriedade do tipo Array chamado items para armazenar valores na pilha, e implementa dois métodos, push e pop, para adicionar e remover valores da pilha. Esses métodos estão marcados como mutating porque eles precisam modificar (ou mutar) os valores originais da própria struct, que são, no caso, os elementos do array items.

O tipo IntStack mostrado acima só pode ser usado com valores do tipo Int. Seria muito mais útil definir uma classe genérica Stack, que pode gerenciar uma pilha contendo elementos de qualquer tipo.

Aqui está uma versão genérica do mesmo cóigo:

struct Stack<Element> {
  var items = [Element]()
  mutating func push(_ item: Element) {
    items.append(item)
  }
  mutating func pop() -> Element {
    return items.removeLast()
  }
}

Note como a versão genérica da Stack é essencialmente a mesma que a da versão não genérica, mas contendo um type parameter chamado Element, em vez de um tipo Int. Esse type parameter está escrito entre sinais de maior ou menor que, ou angle brackets em inglês (<Element>) logo após o nome da struct.

Element define um placeholder para um tipo ser provido depois. Esse tipo futuro pode ser referido como Element em qualquer lugar dentro da definição do strcut. Nesse caso, Element é usado como um placeholder em três lugares:

  • Para criar uma propriedade chamada items, que é inicializada com um array vazio de valores do tipo Element

  • Para especificar que o método push(_:) tem um único parametro chamado item, que deve ser do tipo Element

  • Para especificar que o valor retornado pelo método pop() será um valor do tipo Element

Porque é um tipo genérico, Stack pode ser usada para criar uma pilha de qualquer tipo válido em Swift, de maneira semelhante a um Array ou Dictionary.

Você pode criar uma nova instancia de Stack escrevendo o tipo a ser armazenado na pilha dentro de angle brackets. Por exemplo, para criar uma nova pilha de Strings, você escreve Stack<String>():

var stackOfStrings = Stack<String\>()
stackOfStrings.push("uno")
stackOfStrings.push("dos")
stackOfStrings.push("tres")
stackOfStrings.push("cuatro")
// the stack now contains 4 strings

É assim que stackOfStrings fica depois de inserir (push) esses 4 valores na pilha:

../_images/stackPushedFourStrings_2x.png

Remover o valor de uma pilha também o retorna. Por exemplo, se removermos um valor da pilha atual, o valor retornado será cuatro.

let fromTheTop = stackOfStrings.pop()
// fromTheTop is equal to "cuatro", and the stack now contains 3 strings

É assim que a pilha fica depois de uma operação pop remover o valor em seu topo.

../_images/stackPoppedOneString_2x.png

Extending a Generic Type

When you extend a generic type, you don’t provide a type parameter list as part of the extension’s definition. Instead, the type parameter list from the original type definition is available within the body of the extension, and the original type parameter names are used to refer to the type parameters from the original definition.

The following example extends the generic Stack type to add a read-only computed property called topItem, which returns the top item on the stack without popping it from the stack:

extension Stack {
  var topItem: Element? {
    return items.isEmpty ? nil : items\[items.count - 1\]
  }
}

The topItem property returns an optional value of type Element. If the stack is empty, topItem returns nil; if the stack isn’t empty, topItem returns the final item in the items array.

Note that this extension doesn’t define a type parameter list. Instead, the Stack type’s existing type parameter name, Element, is used within the extension to indicate the optional type of the topItem computed property.

The topItem computed property can now be used with any Stack instance to access and query its top item without removing it.

if let topItem = stackOfStrings.topItem {
print("The top item on the stack is \(topItem).")
}
// Prints "The top item on the stack is tres."

Extensions of a generic type can also include requirements that instances of the extended type must satisfy in order to gain the new functionality, as discussed in Extensions with a Generic Where Clause below.

Type Constraints

The swapTwoValues(_:_:) function and the Stack type can work with any type. However, it’s sometimes useful to enforce certain type constraints on the types that can be used with generic functions and generic types. Type constraints specify that a type parameter must inherit from a specific class, or conform to a particular protocol or protocol composition.

For example, Swift’s Dictionary type places a limitation on the types that can be used as keys for a dictionary. As described in Dictionaries, the type of a dictionary’s keys must be hashable. That is, it must provide a way to make itself uniquely representable. Dictionary needs its keys to be hashable so that it can check whether it already contains a value for a particular key. Without this requirement, Dictionary could not tell whether it should insert or replace a value for a particular key, nor would it be able to find a value for a given key that is already in the dictionary.

This requirement is enforced by a type constraint on the key type for Dictionary, which specifies that the key type must conform to the Hashable protocol, a special protocol defined in the Swift standard library. All of Swift’s basic types (such as String, Int, Double, and Bool) are hashable by default.

You can define your own type constraints when creating custom generic types, and these constraints provide much of the power of generic programming. Abstract concepts like Hashable characterize types in terms of their conceptual characteristics, rather than their concrete type.

Type Constraint Syntax

You write type constraints by placing a single class or protocol constraint after a type parameter’s name, separated by a colon, as part of the type parameter list. The basic syntax for type constraints on a generic function is shown below (although the syntax is the same for generic types):

func someFunction<T: SomeClass, U: SomeProtocol>(someT: T, someU: U) {
  // function body goes here
}

The hypothetical function above has two type parameters. The first type parameter, T, has a type constraint that requires T to be a subclass of SomeClass. The second type parameter, U, has a type constraint that requires U to conform to the protocol SomeProtocol.

Type Constraints in Action

Here’s a nongeneric function called findIndex(ofString:in:), which is given a String value to find and an array of String values within which to find it. The findIndex(ofString:in:) function returns an optional Int value, which will be the index of the first matching string in the array if it’s found, or nil if the string can’t be found:

func findIndex(ofString valueToFind: String, in array: [String]) -> Int? {
  for (index, value) in array.enumerated() {
    if value == valueToFind {
      return index
    }
  }
  return nil
}

The findIndex(ofString:in:) function can be used to find a string value in an array of strings:

let strings = ["cat", "dog", "llama", "parakeet", "terrapin"]
if let foundIndex = findIndex(ofString: "llama", in: strings) {
  print("The index of llama is \(foundIndex)")
}
// Prints "The index of llama is 2"

The principle of finding the index of a value in an array isn’t useful only for strings, however. You can write the same functionality as a generic function by replacing any mention of strings with values of some type T instead.

Here’s how you might expect a generic version of findIndex(ofString:in:), called findIndex(of:in:), to be written. Note that the return type of this function is still Int?, because the function returns an optional index number, not an optional value from the array. Be warned, though—this function doesn’t compile, for reasons explained after the example:

func findIndex<T>(of valueToFind: T, in array:[T]) -> Int? {
  for (index, value) in array.enumerated() {
    if value == valueToFind {
      return index
    }
  }
  return nil
}

This function doesn’t compile as written above. The problem lies with the equality check, “if value == valueToFind”. Not every type in Swift can be compared with the equal to operator (==). If you create your own class or structure to represent a complex data model, for example, then the meaning of “equal to” for that class or structure isn’t something that Swift can guess for you. Because of this, it isn’t possible to guarantee that this code will work for every possible type T, and an appropriate error is reported when you try to compile the code.

All is not lost, however. The Swift standard library defines a protocol called Equatable, which requires any conforming type to implement the equal to operator (==) and the not equal to operator (!=) to compare any two values of that type. All of Swift’s standard types automatically support the Equatable protocol.

Any type that is Equatable can be used safely with the findIndex(of:in:) function, because it’s guaranteed to support the equal to operator. To express this fact, you write a type constraint of Equatable as part of the type parameter’s definition when you define the function:

func findIndex<T: Equatable>(of valueToFind: T, in array:[T]) -> Int? {
  for (index, value) in array.enumerated() {
    if value == valueToFind {
      return index
    }
  }
  return nil
}

The single type parameter for findIndex(of:in:) is written as T: Equatable, which means “any type T that conforms to the Equatable protocol.”

The findIndex(of:in:) function now compiles successfully and can be used with any type that is Equatable, such as Double or String:

let doubleIndex = findIndex(of: 9.3, in: [3.14159, 0.1, 0.25])
// doubleIndex is an optional Int with no value, because 9.3 isn't in the array
let stringIndex = findIndex(of: "Andrea", in: ["Mike", "Malcolm", "Andrea"])
// stringIndex is an optional Int containing a value of 2

Associated Types

When defining a protocol, it’s sometimes useful to declare one or more associated types as part of the protocol’s definition. An associated type gives a placeholder name to a type that is used as part of the protocol. The actual type to use for that associated type isn’t specified until the protocol is adopted. Associated types are specified with the associatedtype keyword.

Associated Types in Action

Here’s an example of a protocol called Container, which declares an associated type called Item:

protocol Container {
  associatedtype Item
  mutating func append(\_ item: Item)
  var count: Int { get }
  subscript(i: Int) -> Item { get }
}

The Container protocol defines three required capabilities that any container must provide:

  • It must be possible to add a new item to the container with an append(_:) method.

  • It must be possible to access a count of the items in the container through a count property that returns an Int value.

  • It must be possible to retrieve each item in the container with a subscript that takes an Int index value.

This protocol doesn’t specify how the items in the container should be stored or what type they’re allowed to be. The protocol only specifies the three bits of functionality that any type must provide in order to be considered a Container. A conforming type can provide additional functionality, as long as it satisfies these three requirements.

Any type that conforms to the Container protocol must be able to specify the type of values it stores. Specifically, it must ensure that only items of the right type are added to the container, and it must be clear about the type of the items returned by its subscript.

To define these requirements, the Container protocol needs a way to refer to the type of the elements that a container will hold, without knowing what that type is for a specific container. The Container protocol needs to specify that any value passed to the append(_:) method must have the same type as the container’s element type, and that the value returned by the container’s subscript will be of the same type as the container’s element type.

To achieve this, the Container protocol declares an associated type called Item, written as associatedtype Item. The protocol doesn’t define what Item is—that information is left for any conforming type to provide. Nonetheless, the Item alias provides a way to refer to the type of the items in a Container, and to define a type for use with the append(_:) method and subscript, to ensure that the expected behavior of any Container is enforced.

Here’s a version of the nongeneric IntStack type from Generic Types above, adapted to conform to the Container protocol:

struct IntStack: Container {
  // original IntStack implementation
  var items = [Int]()
  mutating func push(_ item: Int) {
    items.append(item)
  }
  mutating func pop() -> Int {
    return items.removeLast()
  }
  // conformance to the Container protocol
  typealias Item = Int
  mutating func append(_ item: Int) {
    self.push(item)
  }
  var count: Int {
    return items.count
  }
  subscript(i: Int) -> Int {
    return items[i]
  }
}

The IntStack type implements all three of the Container protocol’s requirements, and in each case wraps part of the IntStack type’s existing functionality to satisfy these requirements.

Moreover, IntStack specifies that for this implementation of Container, the appropriate Item to use is a type of Int. The definition of typealias Item = Int turns the abstract type of Item into a concrete type of Int for this implementation of the Container protocol.

Thanks to Swift’s type inference, you don’t actually need to declare a concrete Item of Int as part of the definition of IntStack. Because IntStack conforms to all of the requirements of the Container protocol, Swift can infer the appropriate Item to use, simply by looking at the type of the append(_:) method’s item parameter and the return type of the subscript. Indeed, if you delete the typealias Item = Int line from the code above, everything still works, because it’s clear what type should be used for Item.

You can also make the generic Stack type conform to the Container protocol:

struct Stack<Element\>: Container {
  // original Stack<Element> implementation
  var items = [Element]()
  mutating func push(_ item: Element) {
    items.append(item)
  }
  mutating func pop() -> Element {
    return items.removeLast()
  }
  // conformance to the Container protocol
  mutating func append(_ item: Element) {
    self.push(item)
  }
  var count: Int {
    return items.count
  }
  subscript(i: Int) -> Element {
    return items[i]
  }
}

This time, the type parameter Element is used as the type of the append(_:) method’s item parameter and the return type of the subscript. Swift can therefore infer that Element is the appropriate type to use as the Item for this particular container.

Extending an Existing Type to Specify an Associated Type

You can extend an existing type to add conformance to a protocol, as described in Adding Protocol Conformance with an Extension. This includes a protocol with an associated type.

Swift’s Array type already provides an append(_:) method, a count property, and a subscript with an Int index to retrieve its elements. These three capabilities match the requirements of the Container protocol. This means that you can extend Array to conform to the Container protocol simply by declaring that Array adopts the protocol. You do this with an empty extension, as described in Declaring Protocol Adoption with an Extension:

extension Array: Container {}

Array’s existing append(_:) method and subscript enable Swift to infer the appropriate type to use for Item, just as for the generic Stack type above. After defining this extension, you can use any Array as a Container.

Adding Constraints to an Associated Type

You can add type constraints to an associated type in a protocol to require that conforming types satisfy those constraints. For example, the following code defines a version of Container that requires the items in the container to be equatable.

protocol Container {
  associatedtype Item: Equatable
  mutating func append(_ item: Item)
  var count: Int { get }
  subscript(i: Int) -> Item { get }
}

To conform to this version of Container, the container’s Item type has to conform to the Equatable protocol.

Using a Protocol in Its Associated Type’s Constraints

A protocol can appear as part of its own requirements. For example, here’s a protocol that refines the Container protocol, adding the requirement of a suffix(_:) method. The suffix(_:) method returns a given number of elements from the end of the container, storing them in an instance of the Suffix type.

protocol SuffixableContainer: Container {
  associatedtype Suffix: SuffixableContainer where Suffix.Item == Item
  func suffix(_ size: Int) -> Suffix
}

In this protocol, Suffix is an associated type, like the Item type in the Container example above. Suffix has two constraints: It must conform to the SuffixableContainer protocol (the protocol currently being defined), and its Item type must be the same as the container’s Item type. The constraint on Item is a generic where clause, which is discussed in Associated Types with a Generic Where Clause below.

Here’s an extension of the Stack type from Generic Types above that adds conformance to the SuffixableContainer protocol:

extension Stack: SuffixableContainer {
  func suffix(_ size: Int) -> Stack {
    var result = Stack()
    for index in (count-size)..<count {
      result.append(self[index])
    }
    return result
  }
// Inferred that Suffix is Stack.
}
var stackOfInts = Stack<Int>()
stackOfInts.append(10)
stackOfInts.append(20)
stackOfInts.append(30)
let suffix = stackOfInts.suffix(2)
// suffix contains 20 and 30

In the example above, the Suffix associated type for Stack is also Stack, so the suffix operation on Stack returns another Stack. Alternatively, a type that conforms to SuffixableContainer can have a Suffix type that’s different from itself—meaning the suffix operation can return a different type. For example, here’s an extension to the nongeneric IntStack type that adds SuffixableContainer conformance, using Stack<Int> as its suffix type instead of IntStack:

extension IntStack: SuffixableContainer {
  func suffix(_ size: Int) -> Stack<Int> {
    var result = Stack<Int>()
    for index in (count-size)..<count {
      result.append(self[index])
    }
    return result
  }
// Inferred that Suffix is Stack<Int>.
}

Generic Where Clauses

Type constraints, as described in Type Constraints, enable you to define requirements on the type parameters associated with a generic function, subscript, or type.

It can also be useful to define requirements for associated types. You do this by defining a generic where clause. A generic where clause enables you to require that an associated type must conform to a certain protocol, or that certain type parameters and associated types must be the same. A generic where clause starts with the where keyword, followed by constraints for associated types or equality relationships between types and associated types. You write a generic where clause right before the opening curly brace of a type or function’s body.

The example below defines a generic function called allItemsMatch, which checks to see if two Container instances contain the same items in the same order. The function returns a Boolean value of true if all items match and a value of false if they don’t.

The two containers to be checked don’t have to be the same type of container (although they can be), but they do have to hold the same type of items. This requirement is expressed through a combination of type constraints and a generic where clause:

func allItemsMatch<C1: Container, C2: Container>
(_ someContainer: C1, _ anotherContainer: C2) -> Bool
where C1.Item == C2.Item, C1.Item: Equatable {

  // Check that both containers contain the same number of items.
  if someContainer.count != anotherContainer.count {
    return false
  }

  // Check each pair of items to see if they're equivalent.
  for i in 0..<someContainer.count {
    if someContainer[i] != anotherContainer[i] {
      return false
    }
  }

  // All items match, so return true.
  return true
}

This function takes two arguments called someContainer and anotherContainer. The someContainer argument is of type C1, and the anotherContainer argument is of type C2. Both C1 and C2 are type parameters for two container types to be determined when the function is called.

The following requirements are placed on the function’s two type parameters:

  • C1 must conform to the Container protocol (written as C1: Container).

  • C2 must also conform to the Container protocol (written as C2: Container).

  • The Item for C1 must be the same as the Item for C2 (written as C1.Item == C2.Item).

  • The Item for C1 must conform to the Equatable protocol (written as C1.Item: Equatable).

The first and second requirements are defined in the function’s type parameter list, and the third and fourth requirements are defined in the function’s generic where clause.

These requirements mean:

  • someContainer is a container of type C1.

  • anotherContainer is a container of type C2.

  • someContainer and anotherContainer contain the same type of items.

  • The items in someContainer can be checked with the not equal operator (!=) to see if they’re different from each other.

The third and fourth requirements combine to mean that the items in anotherContainer can also be checked with the != operator, because they’re exactly the same type as the items in someContainer.

These requirements enable the allItemsMatch(_:_:) function to compare the two containers, even if they’re of a different container type.

The allItemsMatch(_:_:) function starts by checking that both containers contain the same number of items. If they contain a different number of items, there’s no way that they can match, and the function returns false.

After making this check, the function iterates over all of the items in someContainer with a for-in loop and the half-open range operator (..<). For each item, the function checks whether the item from someContainer isn’t equal to the corresponding item in anotherContainer. If the two items aren’t equal, then the two containers don’t match, and the function returns false.

If the loop finishes without finding a mismatch, the two containers match, and the function returns true.

Here’s how the allItemsMatch(_:_:) function looks in action:

var stackOfStrings = Stack<String>()
stackOfStrings.push("uno")
stackOfStrings.push("dos")
stackOfStrings.push("tres")

var arrayOfStrings = ["uno", "dos", "tres"]

if allItemsMatch(stackOfStrings, arrayOfStrings) {
  print("All items match.")
} else {
  print("Not all items match.")
}
// Prints "All items match."

The example above creates a Stack instance to store String values, and pushes three strings onto the stack. The example also creates an Array instance initialized with an array literal containing the same three strings as the stack. Even though the stack and the array are of a different type, they both conform to the Container protocol, and both contain the same type of values. You can therefore call the allItemsMatch(_:_:) function with these two containers as its arguments. In the example above, the allItemsMatch(_:_:) function correctly reports that all of the items in the two containers match.

Extensions with a Generic Where Clause

You can also use a generic where clause as part of an extension. The example below extends the generic Stack structure from the previous examples to add an isTop(_:) method.

extension Stack where Element: Equatable {
  func isTop(_ item: Element) -> Bool {
    guard let topItem = items.last else {
      return false
    }
    return topItem == item
  }
}

This new isTop(_:) method first checks that the stack isn’t empty, and then compares the given item against the stack’s topmost item. If you tried to do this without a generic where clause, you would have a problem: The implementation of isTop(_:) uses the == operator, but the definition of Stack doesn’t require its items to be equatable, so using the == operator results in a compile-time error. Using a generic where clause lets you add a new requirement to the extension, so that the extension adds the isTop(_:) method only when the items in the stack are equatable.

Here’s how the isTop(_:) method looks in action:

if stackOfStrings.isTop("tres") {
  print("Top element is tres.")
} else {
  print("Top element is something else.")
}
// Prints "Top element is tres."

If you try to call the isTop(_:) method on a stack whose elements aren’t equatable, you’ll get a compile-time error.

struct NotEquatable { }
var notEquatableStack = Stack<NotEquatable>()
let notEquatableValue = NotEquatable()
notEquatableStack.push(notEquatableValue)
notEquatableStack.isTop(notEquatableValue) // Error

You can use a generic where clause with extensions to a protocol. The example below extends the Container protocol from the previous examples to add a startsWith(_:) method.

extension Container where Item: Equatable {
  func startsWith(_ item: Item) -> Bool {
    return count >= 1 && self[0] == item
  }
}

The startsWith(_:) method first makes sure that the container has at least one item, and then it checks whether the first item in the container matches the given item. This new startsWith(_:) method can be used with any type that conforms to the Container protocol, including the stacks and arrays used above, as long as the container’s items are equatable.

if [9, 9, 9].startsWith(42) {
  print("Starts with 42.")
} else {
  print("Starts with something else.")
}
// Prints "Starts with something else."

The generic where clause in the example above requires Item to conform to a protocol, but you can also write a generic where clauses that require Item to be a specific type. For example:

extension Container where Item == Double {
func average() -> Double {
  var sum = 0.0
  for index in 0..<count {
    sum += self[index]
  }
  return sum / Double(count)
  }
}
print([1260.0, 1200.0, 98.6, 37.0].average())
// Prints "648.9"

This example adds an average() method to containers whose Item type is Double. It iterates over the items in the container to add them up, and divides by the container’s count to compute the average. It explicitly converts the count from Int to Double to be able to do floating-point division.

You can include multiple requirements in a generic where clause that is part of an extension, just like you can for a generic where clause that you write elsewhere. Separate each requirement in the list with a comma.

Contextual Where Clauses

You can write a generic where clause as part of a declaration that doesn’t have its own generic type constraints, when you’re already working in the context of generic types. For example, you can write a generic where clause on a subscript of a generic type or on a method in an extension to a generic type. The Container structure is generic, and the `where clauses in the example below specify what type constraints have to be satisfied to make these new methods available on a container.

extension Container {
    func average() -> Double where Item == Int {
        var sum = 0.0
        for index in 0..<count {
            sum += Double(self[index])
        }
        return sum / Double(count)
    }
    func endsWith(_ item: Item) -> Bool where Item: Equatable {
        return count >= 1 && self[count-1] == item
    }
}
let numbers = [1260, 1200, 98, 37]
print(numbers.average())
// Prints "648.75"
print(numbers.endsWith(37))
// Prints "true"

This example adds an average() method to Container when the items are integers, and it adds an endsWith(_:) method when the items are equatable. Both functions include a generic where clause that adds type constraints to the generic Item type parameter from the original declaration of Container.

If you want to write this code without using contextual where clauses, you write two extensions, one for each generic where clause. The example above and the example below have the same behavior.

extension Container where Item == Int {
    func average() -> Double {
        var sum = 0.0
        for index in 0..<count {
            sum += Double(self[index])
        }
        return sum / Double(count)
    }
}
extension Container where Item: Equatable {
    func endsWith(_ item: Item) -> Bool {
        return count >= 1 && self[count-1] == item
    }
}

In the version of this example that uses contextual where clauses, the implementation of average() and endsWith(_:) are both in the same extension because each method’s generic where clause states the requirements that need to be satisfied to make that method available. Moving those requirements to the extensions’ generic where clauses makes the methods available in the same situations, but requires one extension per requirement.

Associated Types with a Generic Where Clause

You can include a generic where clause on an associated type. For example, suppose you want to make a version of Container that includes an iterator, like what the Sequence protocol uses in the standard library. Here’s how you write that:

protocol Container {
  associatedtype Item
  mutating func append(_ item: Item)
  var count: Int { get }
  subscript(i: Int) -> Item { get }

  associatedtype Iterator: IteratorProtocol where Iterator.Element == Item
  func makeIterator() -> Iterator
}

The generic where clause on Iterator requires that the iterator must traverse over elements of the same item type as the container’s items, regardless of the iterator’s type. The makeIterator() function provides access to a container’s iterator.

For a protocol that inherits from another protocol, you add a constraint to an inherited associated type by including the generic where clause in the protocol declaration. For example, the following code declares a ComparableContainer protocol that requires Item to conform to Comparable:

protocol ComparableContainer: Container where Item: Comparable { }

Generic Subscripts

Subscripts can be generic, and they can include generic where clauses. You write the placeholder type name inside angle brackets after subscript, and you write a generic where clause right before the opening curly brace of the subscript’s body. For example:

extension Container {
subscript<Indices: Sequence>(indices: Indices) -> [Item]
where Indices.Iterator.Element == Int {
  var result = [Item]()
  for index in indices {
    result.append(self[index])
  }
  return result
  }
}

This extension to the Container protocol adds a subscript that takes a sequence of indices and returns an array containing the items at each given index. This generic subscript is constrained as follows:

  • The generic parameter Indices in angle brackets has to be a type that conforms to the Sequence protocol from the standard library.

  • The subscript takes a single parameter, indices, which is an instance of that Indices type.

  • The generic where clause requires that the iterator for the sequence must traverse over elements of type Int. This ensures that the indices in the sequence are the same type as the indices used for a container.

Taken together, these constraints mean that the value passed for the indices parameter is a sequence of integers.