Let's fix `if let` syntax

Overall, I'm sympathetic to the impulse to revisit this topic (if skeptical that this iteration will move the needle). But this formulation of @chockenberry's motivating problem, taken together with @jrose's examples of what would need to be excluded at least as a first pass, seems to slice the problem exceedingly fine:

  1. The variable x needs to be meaningfully named, such that it's important to write the name out n times inside the 'if' statement for clarity to the reader.
  2. It needs to be cumbersome enough, however, that writing it out n + 1 times (due to the unwrapping assignment) is once too many.
  3. The name, though cumbersome, needs to be so aptly named that even shortening it for ease of writing n + 1 times would lead to suboptimal clarity.
  4. But not so perfectly named that a refactoring wouldn't possibly rename it, such that the unwrapped name would then be inaccurate.
  5. It can't be a nested name x.y, or some other scenario mentioned by @jrose where the name is likely to be under the control of another party.
3 Likes

FWIW, I don't think that the items that @jrose called out need to be excluded in the first iteration of this feature. I think it's also a perfectly reasonable position (in addition to "these corner cases are excluded") to say "the if let x { ... } form is sufficiently indicative of rebinding/shadowing behavior, and so behaves exactly like if let x = x { ... } in all cases." It doesn't solve the issue with nested names (aside from self), but for me it would address the vast majority of situations where I find myself reaching for this feature.

IMO, the "cumbersome" aspect of this pattern isn't the n to n + 1 step for the entire if statement, it's the 1 to 2 step in the binding itself. For a verbose variable name this results in a long line that is needlessly repetitive and more difficult to quickly scan (on top of the autocomplete and refactoring issues that @chockenberry has called out).

2 Likes

Right, so we're thinking about the situation where the variable must be too verbose for the 'if' expression to read well, but not so much that it affects the readability of the other n uses inside the braces such that it isn't better off renamed upon unwrapping.

The other n uses inside the braces may or may not suffer from the same immediate repetition issue that plagues the binding. It's perfectly plausible that a variable name is terse enough to be readable in isolation in n different expressions, but verbose enough to hurt readability issues as you're forced to repeat it multiple times in the same expression (and I see such names relatively frequently for view controllers in particular, which @chockenberry notes in the OP).

1 Like

Yes, it’s plausible, but I’m pointing out that this seems to narrow the extent of the problem at hand significantly.

1 Like

I just read all the comments in this thread, and I still don't understand how the suggested approaches improve anything else but the brevity of the code. The Fundamentals section of the API Design Guidelines of Swift is very clear on that:

Clarity is more important than brevity. Although Swift code can be compact, it is a non-goal to enable the smallest possible code with the fewest characters. Brevity in Swift code, where it occurs, is a side-effect of the strong type system and features that naturally reduce boilerplate.

None of the suggested approaches to me add any value that can't be covered by the existing if let unwrapped = optional or guard let unwrapped = optional syntax, they just aim to make the code shorter. But for this brevity, they introduce yet another thing everyone of us has to learn and the compiler needs to support. So, from my point of view, they only make the language more complicated without making it more safe or allowing any new concepts.

Thus, a clear -1 on this from me.

8 Likes

When you reduce if let x = x to (for example) if let x, it's not just more brief, it's less repetitive. Removing repetition from code is nearly always a good thing.

11 Likes

I actually really like this. Well, maybe not this syntax exactly (it's not very Swifty, but we can bikeshed later), but the logic behind it is very solid. We already have a syntax for "perform this control flow and extract the value." Now we need syntax for "give this new binding the same name as that binding without making me retype it."

This would work for many of the previously mentioned tricky cases:

if let '' = value() { /* use value */ }
guard let '' = x.y else { ... } /* use y */

In addition to being able to be used in other (non-optional-binding) contexts:

func f(x: String = "Hello") {  
    var '' = x 
    x.append("!")
    print(x) // → "Hello!"
}

struct Foo { let bar = "Hello" }
let foo = Foo() 
let '' = foo.bar
print(bar) // => "Hello"
2 Likes

It seems to me part of the problem is the strong bias towards the existence of some value or object. The names of the optionals reveal it. For example, fooAutomationViewController presumably has a type like Optional<FooAutomationViewController>. That is not the same as FooAutomationViewController. So I think we need to consider the provider API of Optional<FooAutomationViewController> values and ask if this discussion is motivated by dealing with symptoms of poor API design resulting in excessively optionals or is there something more to the picture that I am not seeing?

Perhaps these discussions should be grounded more in a consideration of common, recurring situations where optionals are produced and what, if any, language features could be provided to deal with the existence/non-existence of values and objects in a readable, non-repetitive way.

2 Likes

I like if let x { and if var x { more than if x? {

1 Like

I like Jens suggestion of:

if case let fooViewController? {
  fooViewController.view
}

as it build on existing syntax. It also will be a happy sibling with other pattern matching.

You might say it’s slightly too long, but I think that’s an argument to make the case let form shorter and easier to write — make all pattern matching nicer to write.

3 Likes

To me, when I first saw if let x, I thought this would mean that the optional binding was being deferred, much like let x: Int defers the assignment of x. In fact, if we ever wanted deferred optional binding for some reason, this is probably the syntax I'd prefer.

I’ve always known Swift as a language that tries to be friendly to people either coming from other languages, or getting into programming for the first time.

For the latter, like others have said here before, I agree that if let foo = foo isn’t very read-friendly. In fact, it requires cmd+clicking on both the first as well as the 2nd foo to even begin understating the meaning of it.

For the former, I assume there are other, less niche examples out there, but in Hack the “magic” of if foo != nil is used and it was very obvious to me as I first started using Hack.
It also checks the “read aloud” test. It might fail the “no magic” test, but few people fully understand (and keep track) of block scopes (unless they come from cpp maybe) - so the “magic” here is not really that magical to most people in my opinion. Instead, what they do get is code that is very straightforward and clear - “if blah isn’t null, do something with blah”. I think that if let foo works too, but if foo != nil is even less ambiguous.

Count me in the "it's fine the way it is" camp, but in case we land on using a new keyword here, we already have a word for this concept. Consider

if let shadow foo { }

if var shadow foo {}

You could even consider shortening the let version to

if shadow foo {}
3 Likes

if let x = isn't just shadowing. It's creating a non-Optional from an Optional. The syntax isn't even allowed if the RHS isn't an Optional type.

1 Like

That's true.
However I also believe this thread is basically discussing about shorthand for variable shadowing but not about shorthand for unwrapping optional value directly.


If a new keyword doesn't have to be introduced, I feel if let optionalValue { ... } natural.
Because I see something common in Swift already (lazy deferred initialization):

let value: Int

if true {
    value = 1
} else {
    value = 0
}

So:

let value: Int? = 1
let condition: Bool = true

if let value {
    ...
}

if let value,
   condition {
    ...
}

Not sure how that would work, and if it would work, would the conditional of an if-statement be the right place to put it? What would it be checking other than "is x an optional?" which should already be statically known in most cases.

I think the if let x = x form is here to stay because of:

  • Not breaking existing code
  • Scenarios like if let z = x.y.z which might not be supported by the new syntax
  • Scenarios where you want an entirely new variable: while let x = arr.popLast()

So I'm in favor of a spelling which stays close to the existing one, i.e. if let x or if let x?. The autocomplete experience will also be superior by keeping the if let keyword combo.

With that assumption, I'd like to add that clarity probably wouldn't be affected too much, because optional unwrapping is one of the first things you learn, long before you're studying some existing codebase. Those teaching can introduce the if let x = x form first and show the shorthand form later. It will be less confusing to hear "you can just omit the repetitive part of the expression" than "there's this whole other syntax..."

Full disclosure: I did not learn Swift as someone new to programming and didn't struggle with optionals at all, and I don't teach Swift either, so I might be off the mark.

2 Likes

let '' = expression seems nice but would contain too many limitation. Considering these codes, we would agree only variables (and possibly properties) should be allowed. However, such limitations wouldn't match with developers' feeling that the right side of let abc = ... expression is not limited.

let '' = 3                 // ???
let '' = x.y               // y?
let '' = x.y + x.z         // ???
let '' = String()          // String as a constant?
let '' = String.init()     // init as a constant? 
let '' = array[0]          // ???
let '' = tuple.0           // ???
let '' = $0                // (inside closures) $0?
let '' = makeInt()         // makeInt as a constant?
let '' = makeString(1)     // makeString as a constant?

What we need actually is feature like let '' = expression that auto-generates variable name that makes sense to solve the problem of refactoring, autocomplete, and repetition. But it is impossible, because such problems are omnipresent in programs. if let variable {} solves only the problem for variable inside statements. let '' = expression could support other cases, but that is not enough.


We should rather give up than introduce half-baked solution. I can imagine myself using this feature, who try to change if let x to x?.foo, write if let x?.foo(), find it is disallowed, and unwillingly rewrite as if let x = x?.foo(). It's much worse than ever. Using if let x = x {} always, is much more easy to write and understandable.

1 Like

[Let's consider guard as it has more strict requirements, and a potential new syntax working for guard would work for all the other cases]

In terms of teaching Swift and progressive disclosure, a potential learner should initially consider that what we're doing here is matching an Optional instance to a pattern, and binding a value to a new instance:

let optionalValue: Int? = 42

guard case .some(let value) = optionalValue else {
    return
}

print(value)

As a next step, the learner would consider the shorthand syntax for matching Optional as sugar for the previous form:

let optionalValue: Int? = 42

guard case let value? = optionalValue else {
    return
}

print(value)

Finally, the even shorter one for some specific cases of pattern matching (like guard):

let optionalValue: Int? = 42

guard let value = optionalValue else {
    return
}

print(value)

This makes a lot of sense, it takes into account every possible combination of Optional-returning expressions, var vs let, et cetera, and would be pretty conclusive to me.

The unfortunate reality is that it's often the case that the Optional instance will already be called value, or something the doesn't include optional in the name, which makes sense because the optionality is in the type (we call instances let name: String, not let nameString: String). Also optional in the instance name is pretty heavy, and usually undesirable, except for very specific cases. Thus, it seems to me that the "real" problem here is that we tend to assign the same exact name to the original Optional instance and the newly bound non-Optional one.

But if the learner is at the point of clearly and intuitively understanding the final shorthand syntax, the fact that they're going to write and read something like guard let value = value is only a concern insofar that it "feels a little strange" (not really, actually, after a few years of writing stuff like that). Thus I've learned to try and add some hint to the Optional instance name, in order to distinguish it from a potential non-Optional one (for example, using optValue).

I agree that this "feels" strange and I don't have a solution for this, but in my opinion possible solutions should not undermine some of the core foundations of Swift, and should not obfuscate the fact that we are actually doing an assignment to a new, completely different constant.

This suggests some new kind of keyword, operator or declaration, that could maintain the assignment part intact, and produce a shorthand syntax that better conveys the intent. Circling back to guard case .some(let value) = optionalValue, I'd suggest that the guard let value part should be maintained. I can think of a few options:

guard let value =? else { return }
guard let value = unwrapped else { return }
guard let value unwrapped else { return }
guard let value? else { return }
guard case let value? else { return }
guard unwrap let value else { return }

I'm not sure if any of those is good. I probably prefer the last one guard unwrap let value else { return } where unwrap is a new keyword that replaces case in cases where the pattern that we're matching to is assignment from an unwrapped Optional.

Also guard case let value? else { return } make sense to me because what we're logically doing is using the intermediate form guard case let value? = optionalValue else { return } but omitting the = optionalValue part because the name of the instance is the same: this feels consistent and rational, because it uses already existing syntax and omits a redundant element.

2 Likes

A lot of these suggestions are gravitating towards using a keyword to signify the unwrapping, but really it’s the ‘create a new variable with the same name as the old’ part that is novel. What about using a keyword specifically for the name referral and using the existing force unwrap sigil to signify unwrapping:

if let value = itself! {}

1 Like