DEV Community

Gamya
Gamya

Posted on

Customizing Parameter Labels in Swift ๐Ÿท๏ธ

You've already seen how Swift functions use named parameters to make calls self-explanatory. For example, a function that rolls a dice a certain number of times:

func rollDice(sides: Int, count: Int) -> [Int] {
    var rolls = [Int]()

    for _ in 1...count {
        let roll = Int.random(in: 1...sides)
        rolls.append(roll)
    }

    return rolls
}

let rolls = rollDice(sides: 6, count: 4)
Enter fullscreen mode Exit fullscreen mode

Even months later, rollDice(sides: 6, count: 4) reads clearly โ€” six sided dice, rolled four times.


๐Ÿงฉ Swift Uses Parameter Names to Tell Functions Apart

Parameter names are so important in Swift that they're actually used to figure out which function to call. This is valid Swift:

func recruitNinja(name: String) { }
func recruitNinja(village: String) { }
func recruitNinja(rank: String) { }
Enter fullscreen mode Exit fullscreen mode

Three functions, all called recruitNinja(), but Swift knows exactly which one you mean based on the parameter name. In documentation you'll often see these written as recruitNinja(name:), recruitNinja(village:), and so on.


๐Ÿ™ˆ Removing a Parameter Label Entirely

Think about hasPrefix():

let lyric = "I am the hidden leaf village's number one knucklehead ninja"
print(lyric.hasPrefix("I am"))
Enter fullscreen mode Exit fullscreen mode

We pass the prefix directly โ€” not hasPrefix(string:) or hasPrefix(prefix:). That's because Swift lets us give a parameter two names: one for the call site, and one for use inside the function. hasPrefix() uses _ as its external name, which means "no label here at all."

We can do the same thing ourselves. Take this function:

func isUppercase(string: String) -> Bool {
    string == string.uppercased()
}

let string = "BELIEVE IT!"
let result = isUppercase(string: string)
Enter fullscreen mode Exit fullscreen mode

string: string reads a bit repetitive โ€” what else would you pass in? Adding an underscore removes the external label:

func isUppercase(_ string: String) -> Bool {
    string == string.uppercased()
}

let string = "BELIEVE IT!"
let result = isUppercase(string)
Enter fullscreen mode Exit fullscreen mode

This is used a lot in Swift โ€” append() for adding to an array, or contains() for checking membership โ€” because the parameter is obvious without a label.


โœ๏ธ Giving a Parameter Two Different Names

Sometimes you want an external label, but the obvious one doesn't read naturally. Take this function:

func printTimesTables(number: Int) {
    for i in 1...12 {
        print("\(i) x \(number) is \(i * number)")
    }
}

printTimesTables(number: 5)
Enter fullscreen mode Exit fullscreen mode

printTimesTables(number: 5) is valid, but it doesn't read naturally. printTimesTables(for: 5) would read much better โ€” you could say "print times table for 5" out loud. The problem is for is a reserved word and can't be used as a parameter name inside the function.

The solution โ€” write two names, one external, one internal:

func printTimesTables(for number: Int) {
    for i in 1...12 {
        print("\(i) x \(number) is \(i * number)")
    }
}

printTimesTables(for: 5)
Enter fullscreen mode Exit fullscreen mode

Breaking that down:

  • for number: Int โ€” for is the external name, number is the internal name, and the type is Int
  • At the call site, we use the external name: printTimesTables(for: 5)
  • Inside the function, we use the internal name: print("\(i) x \(number) is \(i * number)")

So Swift gives us two tools: _ to remove an external name entirely, or a second name to have different external and internal labels.

๐Ÿ’ก Terminology tip: values you pass into a function are technically called arguments, and the names you use inside the function are parameters. When the distinction matters, we call them the "external parameter name" and "internal parameter name".


When Should You Omit a Parameter Label? ๐Ÿค”

Using _ for a parameter's external label is common, especially when the function name is a verb and the first parameter is the noun it acts on:

  • Summoning a creature would be summon(toad) rather than summon(creature: toad)
  • Equipping a weapon would be equip(kunai) rather than equip(item: kunai)
  • Finding a target would be find(target) rather than find(enemy: target)

This is especially useful when the label would just repeat the variable name being passed in:

  • Casting a jutsu would be cast(jutsu) rather than cast(jutsu: jutsu)
  • Activating a sharingan would be activate(sharingan) rather than activate(sharingan: sharingan)
  • Reading a scroll would be read(scroll) rather than read(scroll: scroll)

๐Ÿ’ก Before SwiftUI, apps were built with UIKit, AppKit, and WatchKit โ€” frameworks designed around an older language called Objective-C, where a function's first parameter was always unnamed. That's why you'll often see Swift functions from those frameworks with _ for their first parameter โ€” it keeps things compatible with Objective-C.


Why Does Swift Use Parameter Labels Anyway? ๐Ÿคท

Many languages don't use parameter labels at all, or make them optional. Swift is unusual โ€” it leans into them heavily, and even lets us split external and internal names!

Consider this kind of code, common in other languages:

setReactorStatus(true, true, false)
Enter fullscreen mode Exit fullscreen mode

Perfectly normal elsewhere, but rare in Swift โ€” because without labels, who can tell what each true or false actually means? Instead, Swift encourages this:

func setReactorStatus(primaryActive: Bool, backupActive: Bool, isEmergency: Bool) {
    // code here
}

setReactorStatus(primaryActive: true, backupActive: true, isEmergency: false)
Enter fullscreen mode Exit fullscreen mode

Now it's obvious what each value controls โ€” no need to memorize argument order.

Swift takes this further by allowing two labels per parameter โ€” one internal, one external:

func setAge(for person: String, to value: Int) {
    print("\(person) is now \(value)")
}

setAge(for: "Itachi", to: 21)
Enter fullscreen mode Exit fullscreen mode

This solves two problems at once:

  • At the call site, setAge(for: "Itachi", to: 21) reads like a sentence โ€” "set age for Itachi to 21"
  • Inside the function, person and value are meaningful names to work with

Compare the alternatives:

  • Using only person and value as labels would force setAge(person: "Itachi", value: 21) โ€” "set age person Itachi value 21" isn't natural English
  • Using only for and to as labels would make the function body read print("\(for) is now \(to)") โ€” and Swift wouldn't even allow this, because it would think for was starting a loop!

Having both internal and external names lets functions read naturally in both places. They're optional โ€” plenty of functions only need one label โ€” but they're a powerful tool when a function needs to read well on both sides. ๐Ÿƒ


Default Parameters ๐ŸŽฏ

Default parameters let us provide sensible fallback values, so callers can ignore parameters entirely when the defaults are fine โ€” but still customize them when needed.

Imagine a function for planning a route between two locations:

func findDirections(from: String, to: String, route: String = "fastest", avoidHighways: Bool = false) {
    // code here
}
Enter fullscreen mode Exit fullscreen mode

Most people want the fastest route without avoiding highways โ€” so those become the defaults. This means the same function can be called in multiple ways:

findDirections(from: "Konoha", to: "Suna")
findDirections(from: "Konoha", to: "Suna", route: "scenic")
findDirections(from: "Konoha", to: "Suna", route: "scenic", avoidHighways: true)
Enter fullscreen mode Exit fullscreen mode

Shorter code most of the time, with full flexibility when something custom is needed. ๐Ÿ—บ๏ธ


Variadic Functions ๐Ÿ“ฆ

Variadic parameters let a function accept any number of values of the same type, separated by commas. Inside the function, they arrive as an array that you can loop over, index into, and so on.

The real power is that a variadic parameter can be used exactly like a normal one most of the time. Imagine an open() function for opening files:

open("photo.jpg")
Enter fullscreen mode Exit fullscreen mode

If open()'s parameter is variadic, the exact same function could also open multiple files at once:

open("photo.jpg", "recipes.txt", "myCode.swift")
Enter fullscreen mode Exit fullscreen mode

Nothing about how the function is called needs to change for the single-file case โ€” variadics just unlock extra functionality on top.

You probably won't reach for variadic functions much while learning, since early projects tend to be small and specific. But as your skills grow, you'll find you can turn existing functions variadic without breaking anything that already calls them โ€” adding new functionality without disturbing what's already there. ๐ŸŒฑ


I know these Swift concepts well from hands-on practice โ€” I use AI to help draft and organize my explanations, and every example and structure choice is something I've reviewed and stand behind.


Wrap Up ๐ŸŽฌ

  • Parameter names aren't just documentation โ€” Swift uses them to tell overloaded functions apart
  • Use _ before a parameter name to remove its external label entirely โ€” common when a verb function acts directly on a noun, like summon(toad)
  • Give a parameter two names (for number: Int) when you want a label that reads naturally at the call site but can't be used as an internal variable name
  • Default parameter values (route: String = "fastest") let callers skip parameters they don't care about, while still allowing full customization
  • Variadic parameters (numbers: Int...) let a function accept any number of values of the same type, arriving inside as an array โ€” and can often be added later without breaking existing calls

Top comments (6)

Collapse
 
junhao profile image
Ahmed bahar

MY DEAR GAMYA

Great article! I learned several useful details about Swift parameter labels that I hadn't fully understood before.
Your explanation of external and internal parameter names was clear and easy to remember.
I really liked the practical examples because they show exactly how these features are used in real projects.
The section on using _ to remove parameter labels was especially helpful and concise. This post does an excellent job of explaining why Swift prioritizes readability at the call site.
The comparison between labeled and unlabeled parameters clearly demonstrates the benefits of Swift's approach. I appreciate how you included default parameters and variadic functions to provide a more complete picture of the topic.
The examples were well chosen and made complex concepts feel approachable. This tutorial is a valuable resource for beginners who want to write cleaner and more idiomatic Swift code. Thanks for sharing such a detailed, informative, and well-structured explanation.

Collapse
 
gamya_m profile image
Gamya

Thank you so much! ๐Ÿฅน๐Ÿ’™ Really glad the external/internal name distinction stuck โ€” that one took a while to phrase clearly, so happy it landed. Appreciate the thoughtful read!

Collapse
 
sloan profile image
Sloan the DEV Moderator

Hey, this article appears to have been generated with the assistance of ChatGPT or possibly some other AI tool.

We allow our community members to use AI assistance when writing articles as long as they abide by our guidelines. Please review the guidelines and edit your post to add a disclaimer.

Failure to follow these guidelines could result in DEV admin lowering the score of your post, making it less visible to the rest of the community. Or, if upon review we find this post to be particularly harmful, we may decide to unpublish it completely.

We hope you understand and take care to follow our guidelines going forward!

Collapse
 
gamya_m profile image
Gamya

Hi Sloan, thank you for flagging this and for the clear guidelines! ๐Ÿ˜Š I've added a disclosure note to the article acknowledging the AI assistance used in drafting and organizing the explanations. I'll make sure to include this going forward on future posts as well. Appreciate you taking the time to point me to the policy!

Collapse
 
technogamerz profile image
๐•‹๐•™๐•– ๐•ƒ๐•’๐•ซ๐•ช ๐”พ๐•š๐•ฃ๐•

Thank you!! Thank you!! So much ๐Ÿ’—

Collapse
 
gamya_m profile image
Gamya

You're so welcome! ๐Ÿ˜Š๐ŸŒธ