orphan: |
---|
Note
This guide documents current practice in the Swift standard library as of April 2015. API conventions are expected to evolve in the near future to better harmonize with Cocoa.
The current Swift Standard Library API conventions start with the Cocoa guidelines as discussed on these two wiki pages: [API Guidelines, Properties], and in this WWDC Presentation. Below, we list where and how the standard library's API conventions differ from those of Cocoa
Points in this section clash in one way or other with the Cocoa guidelines.
The first parameter to a function, method, or initializer typically does not have an argument label:
alligators.insert(fred) // yes if alligators.contains(george) { // yes return } alligators.insert(element: fred) // no if alligators.contains(element: george) { // no return }
Typically, no suffix is added to a function or method's base name in order to serve the same purpose as a label:
alligators.insertElement(fred) // no if alligators.containsElement(george) { // no return }
A preposition is added to the end of a function name if the role of the first parameter would otherwise be unclear:
// origin of measurement is aPosition aPosition.distanceTo(otherPosition) // we're not "indexing x" if let position = aSet.indexOf(x) { ... }
Argument labels are used on first parameters to denote special cases:
// Normal case: result has same value as argument (traps on overflow) Int(aUInt) // Special: interprets the sign bit as a high bit, changes value Int(bitPattern: aUInt) // Special: keeps only the bits that fit, losing information Int32(truncatingBitPattern: anInt64)
Argument labels are chosen to clarify the role of an argument, rather than its type:
x.replaceRange(r, with: someElements) p.initializeFrom(q, count: n)
Second and later parameters are always labeled except in cases where there's no useful distinction of roles:
swap(&a, &b) // OK let topOfPicture = min(topOfSquare, topOfTriangle, topOfCircle) // OK
We don't use namespace prefixes such as “
NS
”, relying instead on the language's own facilities.Names of types, protocols and enum cases are
UpperCamelCase
. Everything else islowerCamelCase
. When an initialism appears, it is uniformly upper- or lower-cased to fit the pattern:let v: String.UTF16View = s.utf16
Protocol names end in
Type
,able
, orible
. Other type names do not.
Points in this section place additional constraints on the standard library, but are compatible with the Cocoa guidelines.
We document the complexity of operations using big-O notation.
In API design, when deciding between a nullary function and a property for a specific operation, arguments based on performance characteristics and complexity of operations are not considered. Reading and writing properties can have any complexity.
We prefer methods and properties to free functions. Free functions are used when there's no obvious
self
min(x, y, z)
when the function is an unconstrained generic
print(x)
and when function syntax is part of the domain notation
-sin(x)
Type conversions use initialization syntax whenever possible, with the source of the conversion being the first argument:
let s0 = String(anInt) // yes let s1 = String(anInt, radix: 2) // yes let s1 = anInt.toString() // no
The exception is when the type conversion is part of a protocol:
protocol IntConvertible { func toInt() -> Int // OK }
Even unlabelled parameter names should be meaningful as they'll be referred to in comments and visible in “generated headers” (cmd-click in Xcode):
/// Reserve enough space to store `minimumCapacity` elements. /// /// PostCondition: `capacity >= minimumCapacity` and the array has /// mutable contiguous storage. /// /// Complexity: O(`count`) mutating func reserveCapacity(minimumCapacity: Int)
Type parameter names of generic types describe the role of the parameter, e.g.
struct Dictionary<Key, Value> { // not Dictionary<K, V>
Type parameter names of generic functions may be single characters:
func swap<T>(inout lhs: T, inout rhs: T)
lhs
andrhs
are acceptable names for binary operator or symmetric binary function parameters:func + (lhs: Int, rhs: Int) -> Int func swap<T>(inout lhs: T, inout rhs: T)
body
is an acceptable name for a trailing closure argument when the resulting construct is supposed to act like a language extension and is likely to have side-effects:func map<U>(transformation: T->U) -> [U] // not this one func forEach<S: SequenceType>(body: (S.Generator.Element)->())
Any
is used as a prefix to denote “type erasure,” e.g.AnySequence<T>
wraps any sequence with element typeT
, conforms toSequenceType
itself, and forwards all operations to the wrapped sequence. When handling the wrapper, the specific type of the wrapped sequence is fully hidden.Custom
is used as a prefix for special protocols that will always be dynamically checked for at runtime and don't make good generic constraints, e.g.CustomStringConvertible
.InPlace
is used as a suffix to denote the mutating member of a pair of related methods:extension Set { func union(other: Set) -> Set mutating func unionInPlace(other: Set) }
with
is used as a prefix to denote a function that executes a closure within a context, such as a guaranteed lifetime:s.withCString { let fd = fopen($0) ... } // don't use that pointer after the closing brace
Pointer
is used as a suffix to denote a non-class type that acts like a reference, c.f.ManagedBufferPointer
unsafe
orUnsafe
is always used as a prefix when a function or type allows the user to violate memory or type safety, except on methods of types whose names begin withUnsafe
, where the type name is assumed to convey that.C
is used as a prefix to denote types corresponding to C language types, e.g.CChar
.