Most active commenters
  • est(3)
  • stfwn(3)
  • sametmax(3)

←back to thread

317 points est | 28 comments | | HN request time: 0.642s | source | bottom
Show context
oooooof ◴[] No.17448560[source]
What is it? The link points to a discussion more deep than I’m willing to read.
replies(10): >>17448567 #>>17448570 #>>17448571 #>>17448572 #>>17448575 #>>17448579 #>>17448584 #>>17448591 #>>17448617 #>>17448638 #
1. est ◴[] No.17448572[source]
It's a controversial PEP https://www.python.org/dev/peps/pep-0572/ which allows you to write Python like this:

    def foo():
        if n := randint(0, 3):
            return n ** 2
        return 1337


    [(x, y, x/y) for x in input_data if (y := f(x)) > 0]
replies(7): >>17448580 #>>17448633 #>>17448694 #>>17448731 #>>17448946 #>>17449000 #>>17449023 #
2. ForHackernews ◴[] No.17448580[source]
Ick.
replies(1): >>17448601 #
3. kibibu ◴[] No.17448601[source]
I've come around to it purely based on the application in list comprehensions.
4. aviraldg ◴[] No.17448633[source]
It also seems include a special case for if/while that lets you do:

    def foo():
        if randint(0, 3) as n:
            return n ** 2
        return 1337
which looks a bit better to me.
replies(2): >>17448751 #>>17449041 #
5. chombier ◴[] No.17448694[source]
Can somebody comment on why is this PEP controversial?
replies(3): >>17448727 #>>17448733 #>>17449044 #
6. detaro ◴[] No.17448727[source]
It makes list expressions and some other things more powerful, but some feel the potential to create difficult-to-understand constructs with it is too high and the current ways of writing such code are clear enough.
7. sluukkonen ◴[] No.17448731[source]
Would've making regular assignment an expression broken too much existing code?
replies(2): >>17448916 #>>17449059 #
8. est ◴[] No.17448733[source]
For many people (including me) who learned Python the way that, in languages like C, the `if x=2` assignment combined with condition is an anti-pattern and prone to errors.

This PEP solves very little problem, saves a few characters of code, but adds complexity to readability.

9. icebraining ◴[] No.17448751[source]
I think that's a rejected alternative proposal, not part of this PEP.
10. icebraining ◴[] No.17448916[source]
Probably not, since expressions can already be statements. But that would allow dangerous code like "if a = 3", which I don't think the Python devs would want to allow.
11. stfwn ◴[] No.17448946[source]
This immediately looks useful for things like:

    if foo := bar[baz]:
        bar[baz] += 1
        return foo
    else:
        bar[baz] = 1
        return 0
Where foo is a dict keeping track of multiple things, and a non-existing key (baz) is never an error but rather the start of a new count. Faster and more readable than

    if baz in list(bar.keys()):
    ....
Similar to Swift’s ‘if let’, it seems.
replies(4): >>17448968 #>>17449008 #>>17449051 #>>17449074 #
12. kelnos ◴[] No.17448968[source]
For stuff like that I'd just use `defaultdict`. That if/else tree then reduces to 2 lines total.
replies(1): >>17448988 #
13. stfwn ◴[] No.17448988{3}[source]
That’s a good tip, thanks!
14. i_do_not_agree ◴[] No.17449000[source]
This is horrible. It looks like ":=" is a comparison operator. The last line is dangerously close to Erlang list comprehensions:

[ {X, Y, X/Y} || X <- Some_Function (), Y <- Some_Other_Function () ]

And people bitch about Erlang syntax.

Edit: "/" is the division operator

15. antoinealb ◴[] No.17449008[source]
As pointed, you can use either a default dict or just simply, and [more pythonic](https://blogs.msdn.microsoft.com/pythonengineering/2016/06/2...):

    try:
      bar[baz] += 1
    except KeyError:
      bar[baz] = 1
Also you can check if a key is in a dict simply by doing "if baz in bar" no need for "list(bar.keys())", which will be slow (temp object + linear scan) vs O(1) hashmap lookup.
replies(3): >>17449075 #>>17449181 #>>17449409 #
16. mFixman ◴[] No.17449023[source]
Reminds me of the kind of hacks you would find in an old-school K&R book.
17. s3m4j ◴[] No.17449041[source]
https://www.python.org/dev/peps/pep-0572/#alternative-spelli...
18. ATsch ◴[] No.17449044[source]
I don't think the controversy here is with the feature itself, more with the implementation. Many, me included, would have preferred to seen a different implementation of solutions to the same problems.

Code starts becoming a lot harder to reason about when more than one state is mutated on the same line. The good design of Python makes this harder than in say C and I think this is a step in the wrong direction in that regard.

The two real things this solves are checking for truthyness in an if and reusing values in a filterting comprehension. Instead of the syntax we have now that can be used anywhere, adds a whole new concept and feels kind of out-of-place, I would have much preferred a solution that can only be used in vetted places, doesn't add a new thing people need to learn and follows the style of the language

For example, my preferred solution for `if` would have been:

    if thing() as t:
        print(t)
Usage of `as` is already established by the `with` block

    [value for x in y
     if value
     where value = x * 2]
The order is unfortunately a bit weird here, but there is no need to add the whole concept of a different type of assignment and this syntax will feel instantly recognizable to people familiar mathematical notation, which is where the existing list comprehension syntax comes from and so has been established as well.
replies(1): >>17449064 #
19. sametmax ◴[] No.17449051[source]
Don't wait for 3.8, and don't bother with defaultdict.

collections.Counter is what you want for the counting case.

dict.get() + dict.setdefault() for the general case.

defaultdict is only useful if the factory is expensive to call.

20. sametmax ◴[] No.17449059[source]
It's a voluntary design choice since the beginning of Python to avoid the very common mistake of doing:

    while continue = "yes":
instead of:

    while continue == "yes":
Those mistakes introduce bugs that are hard to spot because they don't cause an immediate error, linters can hardly help with them and even a senior can make them while being tired.
replies(1): >>17453480 #
21. sametmax ◴[] No.17449064{3}[source]
I wanted "as" too. But the accepted operator has the benefit of integrating perfectly with type hints.
22. eesmith ◴[] No.17449074[source]
The place I see using it is in (quoting Python's "python.exe-gdb.py"):

        m = re.match(r'\s*(\d+)\s*', args)
        if m:
            start = int(m.group(0))
            end = start + 10

        m = re.match(r'\s*(\d+)\s*,\s*(\d+)\s*', args)
        if m:
            start, end = map(int, m.groups())
With the new syntax this becomes:

        if m := re.match(r'\s*(\d+)\s*', args):
            start = int(m.group(0))
            end = start + 10

        if m := re.match(r'\s*(\d+)\s*,\s*(\d+)\s*', args)
            start, end = map(int, m.groups())
This pattern occurs just often enough to be a nuisance. For another example drawn from the standard library, here's modified code from "platform.py"

    # Parse the first line
    if (m := _lsb_release_version.match(firstline)) is not None:
        # LSB format: "distro release x.x (codename)"
        return tuple(m.groups())

    # Pre-LSB format: "distro x.x (codename)"
    if (m := _release_version.match(firstline)) is not None:
        return tuple(m.groups())

    # Unknown format... take the first two words
    if l := firstline.strip().split():
        version = l[0]
        if len(l) > 1:
            id = l[1]
replies(1): >>17454726 #
23. stfwn ◴[] No.17449075{3}[source]
The error-catching method seemed too drastic to me before, but the article explains the LBYL vs. EAFP arugument quite well. Thanks!

I should find a way to get more code reviews, I really enjoy learning these small nuggets of info.

24. bb88 ◴[] No.17449181{3}[source]
It's also time saving since the hash lookup needs to be done at most 1, as well. GP has two lookups in the hash list.
25. bocklund ◴[] No.17449409{3}[source]
Alternatively

`bar[baz] = bar.get(baz, 0) + 1`

One line and no error checking.

But the OP was probably just illustrating a basic example where you might have some more intense logic

26. bluecalm ◴[] No.17453480{3}[source]
I don't know about linters but GCC warns me about that every time I make that typo. They could just require parenthesis when assignment value is used as boolean.
27. est ◴[] No.17454726{3}[source]
It' a problem with re module really.

re.match should return a match object no matter what, and .group() should return strings, empty string if non were matched.

replies(1): >>17455316 #
28. eesmith ◴[] No.17455316{4}[source]
I don't see how that would improve things. Could you sketch a solution based around your ideas?