2019-06-07 20:18:07 +02:00
< p align = " center " > < img src = " images/logo.png " alt = " " > < / p >
2018-02-28 21:37:48 +01:00
< h1 align = " center " > What the f * ck Python ! 🐍 < / h1 >
< p align = " center " > An interesting collection of surprising snippets and lesser - known Python features . < / p >
2017-08-26 20:31:11 +02:00
[ ! [ WTFPL 2.0 ] [ license - image ] ] [ license - url ]
2018-12-16 11:53:39 +01:00
Translations : [ Chinese 中文 ] ( https : / / github . com / leisurelicht / wtfpython - cn )
2017-08-26 20:31:11 +02:00
2018-01-19 14:33:37 +01:00
Python , being a beautifully designed high - level and interpreter - based programming language , provides us with many features for the programmer ' s comfort. But sometimes, the outcomes of a Python snippet may not seem obvious to a regular user at first sight.
2017-08-28 21:22:34 +02:00
2018-01-29 18:07:52 +01:00
Here is a fun project to collect such tricky & counter - intuitive examples and lesser - known features in Python , attempting to discuss what exactly is happening under the hood !
2017-08-29 21:10:21 +02:00
2019-06-10 19:52:08 +02:00
While some of the examples you see below may not be WTFs in the truest sense , but they ' ll reveal some of the interesting parts of Python that you might be unaware of. I find it a nice way to learn the internals of a programming language, and I believe that you ' ll find it interesting too !
2017-08-29 21:10:21 +02:00
2018-01-23 11:04:06 +01:00
If you ' re an experienced Python programmer, you can take it as a challenge to get most of them right in first attempt. You may be already familiar with some of these examples, and I might be able to revive sweet old memories of yours being bitten by these gotchas :sweat_smile:
2018-12-04 23:46:32 +01:00
PS : If you ' re a returning reader, you can learn about the new modifications [here](https://github.com/satwikkansal/wtfpython/releases/).
2017-08-26 20:31:11 +02:00
2018-01-22 13:04:02 +01:00
So , here we go . . .
2017-08-30 18:24:35 +02:00
2017-08-26 20:31:11 +02:00
# Table of Contents
2019-07-08 18:52:04 +02:00
< ! - - toc - - >
2018-01-22 14:07:35 +01:00
2017-08-30 13:21:15 +02:00
- [ Structure of the Examples ] ( #structure-of-the-examples)
2019-07-08 18:52:04 +02:00
+ [ ▶ Some fancy Title ] ( #-some-fancy-title)
2017-08-30 19:27:10 +02:00
- [ Usage ] ( #usage)
- [ 👀 Examples ] ( #-examples)
2018-01-22 15:05:37 +01:00
- [ Section : Strain your brain ! ] ( #section-strain-your-brain)
2019-07-08 18:31:41 +02:00
- [ ▶ Strings can be tricky sometimes ] ( #%E2%96%B6-strings-can-be-tricky-sometimes)
- [ ▶ Splitsies ] ( #%E2%96%B6-splitsies)
2019-07-07 18:48:35 +02:00
- [ ▶ Time for some hash brownies ! ] ( #%E2%96%B6-time-for-some-hash-brownies)
2019-07-08 18:31:41 +02:00
- [ ▶ The disorder within order ] ( #%E2%96%B6-the-disorder-within-order)
- [ ▶ Keep trying ? ] ( #%E2%96%B6-keep-trying)
- [ ▶ Deep down , we ' re all the same](# %E 2 %96% B6-deep-down-were-all-the-same)
2019-07-07 18:48:35 +02:00
- [ ▶ For what ? ] ( #%E2%96%B6-for-what)
2019-07-08 18:31:41 +02:00
- [ ▶ Evaluation time discrepancy ] ( #%E2%96%B6-evaluation-time-discrepancy)
2019-07-07 18:48:35 +02:00
- [ ▶ Messing around with ` is ` operator ^ ] ( #%E2%96%B6-messing-around-with-is-operator%5E)
- [ ▶ A tic - tac - toe where X wins in the first attempt ! ] ( #%E2%96%B6-a-tic-tac-toe-where-x-wins-in-the-first-attempt)
- [ ▶ The sticky output function ] ( #%E2%96%B6-the-sticky-output-function)
2019-07-08 18:31:41 +02:00
- [ ▶ The chicken - egg problem ] ( #%E2%96%B6-the-chicken-egg-problem)
2019-07-07 18:48:35 +02:00
- [ ▶ ` is not . . . ` is not ` is ( not . . . ) ` ] ( #%E2%96%B6-is-not--is-not-is-not-)
- [ ▶ The surprising comma ] ( #%E2%96%B6-the-surprising-comma)
2019-07-08 18:31:41 +02:00
- [ ▶ Strings and the backslashes ] ( #%E2%96%B6-strings-and-the-backslashes)
2019-07-07 18:48:35 +02:00
- [ ▶ not knot ! ] ( #%E2%96%B6-not-knot)
- [ ▶ Half triple - quoted strings ] ( #%E2%96%B6-half-triple-quoted-strings)
- [ ▶ Midnight time doesn ' t exist?](# %E 2 %96% B6-midnight-time-doesnt-exist)
- [ ▶ What ' s wrong with booleans?](# %E 2 %96% B6-whats-wrong-with-booleans)
- [ ▶ Class attributes and instance attributes ] ( #%E2%96%B6-class-attributes-and-instance-attributes)
- [ ▶ yielding None ] ( #%E2%96%B6-yielding-none)
- [ ▶ Mutating the immutable ! ] ( #%E2%96%B6-mutating-the-immutable)
- [ ▶ The disappearing variable from outer scope ] ( #%E2%96%B6-the-disappearing-variable-from-outer-scope)
- [ ▶ When True is actually False ] ( #%E2%96%B6-when-true-is-actually-false)
- [ ▶ Lossy zip of iterators ] ( #%E2%96%B6-lossy-zip-of-iterators)
- [ ▶ From filled to None in one instruction . . . ] ( #%E2%96%B6-from-filled-to-none-in-one-instruction)
2019-07-08 18:31:41 +02:00
- [ ▶ Subclass relationships ] ( #%E2%96%B6-subclass-relationships)
- [ ▶ The mysterious key type conversion ] ( #%E2%96%B6-the-mysterious-key-type-conversion)
2019-07-07 18:48:35 +02:00
- [ ▶ Let ' s see if you can guess this?](# %E 2 %96% B6-lets-see-if-you-can-guess-this)
2018-01-25 18:46:33 +01:00
- [ Section : Appearances are deceptive ! ] ( #section-appearances-are-deceptive)
2019-07-07 18:48:35 +02:00
- [ ▶ Skipping lines ? ] ( #%E2%96%B6-skipping-lines)
2019-07-08 18:31:41 +02:00
- [ ▶ Teleportation ] ( #%E2%96%B6-teleportation)
2019-07-07 18:48:35 +02:00
- [ ▶ Well , something is fishy . . . ] ( #%E2%96%B6-well-something-is-fishy)
2018-01-22 15:05:37 +01:00
- [ Section : Watch out for the landmines ! ] ( #section-watch-out-for-the-landmines)
2019-07-07 18:48:35 +02:00
- [ ▶ Modifying a dictionary while iterating over it ] ( #%E2%96%B6-modifying-a-dictionary-while-iterating-over-it)
- [ ▶ Deleting a list item while iterating ] ( #%E2%96%B6-deleting-a-list-item-while-iterating)
- [ ▶ Loop variables leaking out ! ] ( #%E2%96%B6-loop-variables-leaking-out)
- [ ▶ Beware of default mutable arguments ! ] ( #%E2%96%B6-beware-of-default-mutable-arguments)
- [ ▶ Catching the Exceptions ] ( #%E2%96%B6-catching-the-exceptions)
- [ ▶ Same operands , different story ! ] ( #%E2%96%B6-same-operands-different-story)
- [ ▶ The out of scope variable ] ( #%E2%96%B6-the-out-of-scope-variable)
- [ ▶ Be careful with chained operations ] ( #%E2%96%B6-be-careful-with-chained-operations)
- [ ▶ Name resolution ignoring class scope ] ( #%E2%96%B6-name-resolution-ignoring-class-scope)
- [ ▶ Needles in a Haystack ^ ] ( #%E2%96%B6-needles-in-a-haystack-%5E)
2019-09-13 04:03:27 +02:00
- [ ▶ Yielding from . . . return ! ] ( #-yielding-from-return)
2019-07-07 18:48:35 +02:00
- [ ▶ Wild imports ] ( #%E2%96%B6-wild-imports)
2018-01-25 18:46:33 +01:00
- [ Section : The Hidden treasures ! ] ( #section-the-hidden-treasures)
2019-07-08 18:31:41 +02:00
- [ ▶ Okay Python , Can you make me fly ? ] ( #%E2%96%B6-okay-python-can-you-make-me-fly)
- [ ▶ ` goto ` , but why ? ] ( #%E2%96%B6-goto-but-why)
- [ ▶ Brace yourself ! ] ( #%E2%96%B6-brace-yourself)
- [ ▶ Let ' s meet Friendly Language Uncle For Life](# %E 2 %96% B6-lets-meet-friendly-language-uncle-for-life)
- [ ▶ Even Python understands that love is complicated ] ( #%E2%96%B6-even-python-understands-that-love-is-complicated)
2019-07-07 18:48:35 +02:00
- [ ▶ Yes , it exists ! ] ( #%E2%96%B6-yes-it-exists)
2019-07-08 18:31:41 +02:00
- [ ▶ Ellipsis ] ( #%E2%96%B6-ellipsis)
- [ ▶ Inpinity ] ( #%E2%96%B6-inpinity)
- [ ▶ Let ' s mangle](# %E 2 %96% B6-lets-mangle)
2018-02-28 21:37:48 +01:00
- [ Section : Miscellaneous ] ( #section-miscellaneous)
2019-07-07 18:48:35 +02:00
- [ ▶ ` + = ` is faster ] ( #%E2%96%B6--is-faster)
- [ ▶ Let ' s make a giant string!](# %E 2 %96% B6-lets-make-a-giant-string)
- [ ▶ Explicit typecast of strings ] ( #%E2%96%B6-explicit-typecast-of-strings)
- [ ▶ Minor Ones ] ( #%E2%96%B6-minor-ones)
2019-07-08 18:52:04 +02:00
* [ Section : Strain your brain ! ] ( #section-strain-your-brain)
+ [ ▶ Strings can be tricky sometimes ] ( #-strings-can-be-tricky-sometimes)
+ [ ▶ Splitsies ] ( #-splitsies)
+ [ ▶ Time for some hash brownies ! ] ( #-time-for-some-hash-brownies)
+ [ ▶ The disorder within order ] ( #-the-disorder-within-order)
+ [ ▶ Keep trying ? ] ( #-keep-trying)
+ [ ▶ Deep down , we ' re all the same](#-deep-down-were-all-the-same)
+ [ ▶ For what ? ] ( #-for-what)
+ [ ▶ Evaluation time discrepancy ] ( #-evaluation-time-discrepancy)
+ [ ▶ Messing around with ` is ` operator ] ( #-messing-around-with-is-operator)
+ [ ▶ A tic - tac - toe where X wins in the first attempt ! ] ( #-a-tic-tac-toe-where-x-wins-in-the-first-attempt)
+ [ ▶ The sticky output function ] ( #-the-sticky-output-function)
+ [ ▶ The chicken - egg problem ] ( #-the-chicken-egg-problem)
+ [ ▶ ` is not . . . ` is not ` is ( not . . . ) ` ] ( #-is-not--is-not-is-not-)
+ [ ▶ The surprising comma ] ( #-the-surprising-comma)
+ [ ▶ Strings and the backslashes ] ( #-strings-and-the-backslashes)
+ [ ▶ not knot ! ] ( #-not-knot)
+ [ ▶ Half triple - quoted strings ] ( #-half-triple-quoted-strings)
+ [ ▶ Midnight time doesn ' t exist?](#-midnight-time-doesnt-exist)
+ [ ▶ What ' s wrong with booleans?](#-whats-wrong-with-booleans)
+ [ ▶ Class attributes and instance attributes ] ( #-class-attributes-and-instance-attributes)
+ [ ▶ yielding None ] ( #-yielding-none)
+ [ ▶ Mutating the immutable ! ] ( #-mutating-the-immutable)
+ [ ▶ The disappearing variable from outer scope ] ( #-the-disappearing-variable-from-outer-scope)
+ [ ▶ When True is actually False ] ( #-when-true-is-actually-false)
+ [ ▶ Lossy zip of iterators ] ( #-lossy-zip-of-iterators)
+ [ ▶ From filled to None in one instruction . . . ] ( #-from-filled-to-none-in-one-instruction)
+ [ ▶ Subclass relationships ] ( #-subclass-relationships)
+ [ ▶ The mysterious key type conversion ] ( #-the-mysterious-key-type-conversion)
+ [ ▶ Let ' s see if you can guess this?](#-lets-see-if-you-can-guess-this)
* [ Section : Appearances are deceptive ! ] ( #section-appearances-are-deceptive)
+ [ ▶ Skipping lines ? ] ( #-skipping-lines)
+ [ ▶ Teleportation ] ( #-teleportation)
+ [ ▶ Well , something is fishy . . . ] ( #-well-something-is-fishy)
* [ Section : Watch out for the landmines ! ] ( #section-watch-out-for-the-landmines)
+ [ ▶ Modifying a dictionary while iterating over it ] ( #-modifying-a-dictionary-while-iterating-over-it)
+ [ ▶ Stubborn ` del ` operation ] ( #-stubborn-del-operation)
+ [ ▶ Deleting a list item while iterating ] ( #-deleting-a-list-item-while-iterating)
+ [ ▶ Loop variables leaking out ! ] ( #-loop-variables-leaking-out)
+ [ ▶ Beware of default mutable arguments ! ] ( #-beware-of-default-mutable-arguments)
+ [ ▶ Catching the Exceptions ] ( #-catching-the-exceptions)
+ [ ▶ Same operands , different story ! ] ( #-same-operands-different-story)
+ [ ▶ The out of scope variable ] ( #-the-out-of-scope-variable)
+ [ ▶ Be careful with chained operations ] ( #-be-careful-with-chained-operations)
+ [ ▶ Name resolution ignoring class scope ] ( #-name-resolution-ignoring-class-scope)
+ [ ▶ Needles in a Haystack ] ( #-needles-in-a-haystack)
+ [ ▶ Wild imports ] ( #-wild-imports)
* [ Section : The Hidden treasures ! ] ( #section-the-hidden-treasures)
+ [ ▶ Okay Python , Can you make me fly ? ] ( #-okay-python-can-you-make-me-fly)
+ [ ▶ ` goto ` , but why ? ] ( #-goto-but-why)
+ [ ▶ Brace yourself ! ] ( #-brace-yourself)
+ [ ▶ Let ' s meet Friendly Language Uncle For Life](#-lets-meet-friendly-language-uncle-for-life)
+ [ ▶ Even Python understands that love is complicated ] ( #-even-python-understands-that-love-is-complicated)
+ [ ▶ Yes , it exists ! ] ( #-yes-it-exists)
+ [ ▶ Ellipsis ] ( #-ellipsis)
+ [ ▶ Inpinity ] ( #-inpinity)
+ [ ▶ Let ' s mangle](#-lets-mangle)
* [ Section : Miscellaneous ] ( #section-miscellaneous)
+ [ ▶ ` + = ` is faster ] ( #--is-faster)
+ [ ▶ Let ' s make a giant string!](#-lets-make-a-giant-string)
+ [ ▶ Explicit typecast of strings ] ( #-explicit-typecast-of-strings)
+ [ ▶ Minor Ones ] ( #-minor-ones)
2017-08-26 20:31:11 +02:00
- [ Contributing ] ( #contributing)
- [ Acknowledgements ] ( #acknowledgements)
- [ 🎓 License ] ( #-license)
2019-07-08 18:52:04 +02:00
* [ Help ] ( #help)
* [ Surprise your geeky pythonist friends ? ] ( #surprise-your-geeky-pythonist-friends)
* [ Need a pdf version ? ] ( #need-a-pdf-version)
2017-08-26 20:31:11 +02:00
2019-07-08 18:52:04 +02:00
< ! - - tocstop - - >
2017-08-26 20:31:11 +02:00
2017-08-30 13:20:39 +02:00
# Structure of the Examples
2017-08-26 20:31:11 +02:00
2017-08-30 18:37:46 +02:00
All the examples are structured like below :
2017-08-29 21:10:21 +02:00
2019-07-08 18:31:41 +02:00
> ### ▶ Some fancy Title
2018-02-28 21:37:48 +01:00
>
2017-09-07 10:52:34 +02:00
> ` ` ` py
> # Setting up the code.
> # Preparation for the magic...
> ` ` `
2018-02-28 21:37:48 +01:00
>
2019-06-10 19:52:08 +02:00
> * * Output ( Python version ( s ) ) : * *
2017-09-07 10:52:34 +02:00
> ` ` ` py
> >> > triggering_statement
2019-06-10 19:52:08 +02:00
> Some unexpected output
2017-09-07 10:52:34 +02:00
> ` ` `
> ( Optional ) : One line describing the unexpected output .
2018-02-28 21:37:48 +01:00
>
>
2017-09-07 10:52:34 +02:00
> #### 💡 Explanation:
2018-02-28 21:37:48 +01:00
>
2017-09-07 10:52:34 +02:00
> * Brief explanation of what ' s happening and why is it happening.
> ` ` ` py
> Setting up examples for clarification ( if necessary )
> ` ` `
2019-06-10 19:52:08 +02:00
> * * Output Output ( Python version ( s ) ) : * *
2017-09-07 10:52:34 +02:00
> ` ` ` py
> >> > trigger # some example that makes it easy to unveil the magic
> # some justified output
> ` ` `
2017-08-30 13:20:39 +02:00
2018-01-22 13:04:02 +01:00
* * Note : * * All the examples are tested on Python 3.5 .2 interactive interpreter , and they should work for all the Python versions unless explicitly specified in the description .
2017-08-26 20:31:11 +02:00
2017-08-30 13:20:39 +02:00
# Usage
2017-08-26 20:31:11 +02:00
2019-06-10 19:52:08 +02:00
A nice way to get the most out of these examples , in my opinion , will be to just read them chronologically , and for every example :
2018-01-22 10:05:42 +01:00
- Carefully read the initial code for setting up the example . If you ' re an experienced Python programmer, most of the times you will successfully anticipate what ' s going to happen next .
2018-01-19 14:33:37 +01:00
- Read the output snippets and ,
2017-08-30 18:37:46 +02:00
+ Check if the outputs are the same as you ' d expect.
2018-01-22 10:05:42 +01:00
+ Make sure if you know the exact reason behind the output being the way it is .
2019-06-10 19:52:08 +02:00
- If the answer is no ( which is perfectly okay ) , take a deep breath , and read the explanation ( and if you still don ' t understand, shout out! and create an issue [here](https://github.com/satwikkansal/wtfPython)).
2017-08-30 20:29:05 +02:00
- If yes , give a gentle pat on your back , and you may skip to the next example .
2017-08-30 18:37:46 +02:00
2019-06-10 19:52:08 +02:00
PS : You can also read WTFPython at the command line . There ' s a pypi package and an npm package (which supports colored formatting) for the same.
2018-01-22 06:55:49 +01:00
To install the npm package [ ` wtfpython ` ] ( https : / / www . npmjs . com / package / wtfpython )
2017-08-30 23:58:21 +02:00
` ` ` sh
$ npm install - g wtfpython
` ` `
2018-01-22 06:55:49 +01:00
Alternatively , to install the pypi package [ ` wtfpython ` ] ( https : / / pypi . python . org / pypi / wtfpython )
` ` ` sh
$ pip install wtfpython - U
` ` `
2017-08-30 23:58:21 +02:00
Now , just run ` wtfpython ` at the command line which will open this collection in your selected ` $ PAGER ` .
2018-01-22 14:07:35 +01:00
- - -
2017-08-30 13:20:39 +02:00
# 👀 Examples
2017-08-26 20:31:11 +02:00
2018-01-22 14:07:35 +01:00
2018-01-22 15:05:37 +01:00
## Section: Strain your brain!
2018-01-22 14:07:35 +01:00
2019-07-08 18:31:41 +02:00
### ▶ Strings can be tricky sometimes
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 30 f1d3fc - e267 - 4 b30 - 84 ef - 4 d9e7091ac1a - - - >
2018-01-11 17:46:27 +01:00
1 \.
2017-10-11 15:05:15 +02:00
` ` ` py
>> > a = " some_string "
>> > id ( a )
140420665652016
>> > id ( " some " + " _ " + " string " ) # Notice that both the ids are same.
140420665652016
2018-01-11 17:46:27 +01:00
` ` `
2 \.
` ` ` py
>> > a = " wtf "
>> > b = " wtf "
>> > a is b
True
>> > a = " wtf! "
>> > b = " wtf! "
>> > a is b
False
` ` `
3 \.
2019-06-09 14:26:36 +02:00
* * Output ( < Python3 .7 ) * *
2018-01-11 17:46:27 +01:00
` ` ` py
>> > ' a ' * 20 is ' aaaaaaaaaaaaaaaaaaaa '
True
>> > ' a ' * 21 is ' aaaaaaaaaaaaaaaaaaaaa '
2018-01-14 16:41:05 +01:00
False
2018-01-11 17:46:27 +01:00
` ` `
Makes sense , right ?
#### 💡 Explanation:
+ Such behavior is due to CPython optimization ( called string interning ) that tries to use existing immutable objects in some cases rather than creating a new object every time .
+ After being interned , many variables may point to the same string object in memory ( thereby saving memory ) .
2018-01-22 10:05:42 +01:00
+ In the snippets above , strings are implicitly interned . The decision of when to implicitly intern a string is implementation dependent . There are some facts that can be used to guess if a string will be interned or not :
2018-01-11 17:46:27 +01:00
* All length 0 and length 1 strings are interned .
* Strings are interned at compile time ( ` ' wtf ' ` will be interned but ` ' ' . join ( [ ' w ' , ' t ' , ' f ' ] ` will not be interned )
2018-05-29 20:59:56 +02:00
* Strings that are not composed of ASCII letters , digits or underscores , are not interned . This explains why ` ' wtf! ' ` was not interned due to ` ! ` . Cpython implementation of this rule can be found [ here ] ( https : / / github . com / python / cpython / blob / 3.6 / Objects / codeobject . c #L19)
2019-06-07 20:18:07 +02:00
< img src = " images/string-intern/string_intern.png " alt = " " >
2019-06-09 14:26:36 +02:00
+ Constant folding is a technique for [ peephole optimization ] ( https : / / en . wikipedia . org / wiki / Peephole_optimization ) in Python . This means the expression ` ' a ' * 20 ` is replaced by ` ' aaaaaaaaaaaaaaaaaaaa ' ` during compilation to reduce few clock cycles during runtime . Constant folding only occurs for strings having length less than 20. ( Why ? Imagine the size of ` . pyc ` file generated as a result of the expression ` ' a ' * 10 * * 10 ` ) . [ Here ' s](https://github.com/python/cpython/blob/3.6/Python/peephole.c#L288) the implementation source for the same.
+ Note : In Python 3.7 , Constant folding was moved out from peephole optimizer to the new AST optimizer with some change in logic as well , so the third snippet doesn ' t work for Python 3.7. You can read more about the change [here](https://bugs.python.org/issue11549).
>> >> >> > String can be tricky : Remove outdated snippet
2018-02-25 19:00:28 +01:00
2018-01-11 17:46:27 +01:00
- - -
2019-07-08 18:31:41 +02:00
### ▶ Splitsies
2019-07-07 23:30:33 +02:00
< ! - - Example ID : ec3168ba - a81a - 4482 - afb0 - 691 f1cc8d65a - - - >
2019-06-08 10:51:19 +02:00
` ` ` py
>> > ' a ' . split ( )
[ ' a ' ]
# is same as
>> > ' a ' . split ( ' ' )
[ ' a ' ]
# but
>> > len ( ' ' . split ( ) )
0
# isn't the same as
>> > len ( ' ' . split ( ' ' ) )
1
` ` `
#### 💡 Explanation:
- It might appear at first that the default seperator for split is a single space ` ' ' ` , but as per the [ docs ] ( https : / / docs . python . org / 2.7 / library / stdtypes . html #str.split),
> If sep is not specified or is None , a different splitting algorithm is applied : runs of consecutive whitespace are regarded as a single separator , and the result will contain no empty strings at the start or end if the string has leading or trailing whitespace . Consequently , splitting an empty string or a string consisting of just whitespace with a None separator returns ` [ ] ` .
> If sep is given , consecutive delimiters are not grouped together and are deemed to delimit empty strings ( for example , ` ' 1,,2 ' . split ( ' , ' ) ` returns ` [ ' 1 ' , ' ' , ' 2 ' ] ` ) . Splitting an empty string with a specified separator returns ` [ ' ' ] ` .
- Noticing how the leading and trailing whitespaces are handled in the following snippet will make things clear ,
` ` ` py
>> > ' a ' . split ( ' ' )
[ ' ' , ' a ' , ' ' ]
>> > ' a ' . split ( )
[ ' a ' ]
>> > ' ' . split ( ' ' )
[ ' ' ]
` ` `
- - -
2018-01-22 14:07:35 +01:00
### ▶ Time for some hash brownies!
2019-07-07 23:30:33 +02:00
< ! - - Example ID : eb17db53 - 49 fd - 4 b61 - 85 d6 - 345 c5ca213ff - - - >
2018-01-22 14:07:35 +01:00
1 \.
2017-08-30 20:56:48 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
some_dict = { }
some_dict [ 5.5 ] = " Ruby "
some_dict [ 5.0 ] = " JavaScript "
some_dict [ 5 ] = " Python "
2017-08-30 20:56:48 +02:00
` ` `
* * Output : * *
` ` ` py
2018-01-22 14:07:35 +01:00
>> > some_dict [ 5.5 ]
" Ruby "
2019-07-13 19:11:27 +02:00
>> > some_dict [ 5.0 ] # "Python" destroyed the existence of "JavaScript"?
2018-01-22 14:07:35 +01:00
" Python "
2019-07-13 19:11:27 +02:00
>> > some_dict [ 5 ]
2019-06-09 20:44:01 +02:00
" Python "
>> > complex_five = 5 + 0 j
>> > type ( complex_five )
complex
>> > some_dict [ complex_five ]
2018-01-22 14:07:35 +01:00
" Python "
2017-08-30 20:56:48 +02:00
` ` `
2019-06-09 20:44:01 +02:00
So , why is Python all over the place ?
2017-08-30 20:56:48 +02:00
2017-10-11 15:05:15 +02:00
#### 💡 Explanation
2018-01-22 14:07:35 +01:00
* Python dictionaries check for equality and compare the hash value to determine if two keys are the same .
* Immutable objects with same value always have the same hash in Python .
` ` ` py
2019-06-09 20:44:01 +02:00
>> > 5 == 5.0 == 5 + 0 j
2018-01-22 14:07:35 +01:00
True
2019-06-09 20:44:01 +02:00
>> > hash ( 5 ) == hash ( 5.0 ) == hash ( 5 + 0 j )
2018-01-22 14:07:35 +01:00
True
2017-10-11 15:05:15 +02:00
` ` `
2018-01-22 14:07:35 +01:00
* * Note : * * Objects with different values may also have same hash ( known as hash collision ) .
2018-02-28 21:37:48 +01:00
* When the statement ` some_dict [ 5 ] = " Python " ` is executed , the existing value " JavaScript " is overwritten with " Python " because Python recognizes ` 5 ` and ` 5.0 ` as the same keys of the dictionary ` some_dict ` .
2018-01-22 14:07:35 +01:00
* This StackOverflow [ answer ] ( https : / / stackoverflow . com / a / 32211042 / 4354153 ) explains beautifully the rationale behind it .
2017-08-30 20:56:48 +02:00
2019-06-09 20:44:01 +02:00
- - -
2019-07-08 18:31:41 +02:00
### ▶ The disorder within order
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 91 bff1f8 - 541 d - 455 a - 9 de4 - 6 cd8ff00ea66 - - - >
2019-06-09 20:44:01 +02:00
` ` ` py
from collections import OrderedDict
dictionary = dict ( )
dictionary [ 1 ] = ' a ' ; dictionary [ 2 ] = ' b ' ;
ordered_dict = OrderedDict ( )
ordered_dict [ 1 ] = ' a ' ; ordered_dict [ 2 ] = ' b ' ;
another_ordered_dict = OrderedDict ( )
another_ordered_dict [ 2 ] = ' b ' ; another_ordered_dict [ 1 ] = ' a ' ;
class DictWithHash ( dict ) :
"""
A dict that also implements __hash__ magic .
"""
__hash__ = lambda self : 0
class OrderedDictWithHash ( OrderedDict ) :
"""
A dict that also implements __hash__ magic .
"""
__hash__ = lambda self : 0
` ` `
* * Output * *
` ` ` py
>> > dictionary == ordered_dict # If a == b
True
>> > dictionary == another_ordered_dict # and b == c
True
>> > ordered_dict == another_ordered_dict # the why isn't c == a ??
False
# We all know that a set consists of only unique elements,
# let's try making a set of these dictionaries and see what happens...
>> > len ( { dictionary , ordered_dict , another_ordered_dict } )
Traceback ( most recent call last ) :
File " <stdin> " , line 1 , in < module >
TypeError : unhashable type : ' dict '
# Makes sense since dict don't have __hash__ implemented, let's use
# our wrapper classes.
>> > dictionary = DictWithHash ( )
>> > dictionary [ 1 ] = ' a ' ; dictionary [ 2 ] = ' b ' ;
>> > ordered_dict = OrderedDictWithHash ( )
>> > ordered_dict [ 1 ] = ' a ' ; ordered_dict [ 2 ] = ' b ' ;
>> > another_ordered_dict = OrderedDictWithHash ( )
>> > another_ordered_dict [ 2 ] = ' b ' ; another_ordered_dict [ 1 ] = ' a ' ;
>> > len ( { dictionary , ordered_dict , another_ordered_dict } )
1
>> > len ( { ordered_dict , another_ordered_dict , dictionary } ) # changing the order
2
` ` `
What is going on here ?
#### 💡 Explanation:
- The reason why intransitive equality didn ' t hold among `dictionary`, `ordered_dict` and `another_ordered_dict` is because of the way `__eq__` method is implemented in `OrderedDict` class. From the [docs](https://docs.python.org/3/library/collections.html#ordereddict-objects)
> Equality tests between OrderedDict objects are order - sensitive and are implemented as ` list ( od1 . items ( ) ) == list ( od2 . items ( ) ) ` . Equality tests between ` OrderedDict ` objects and other Mapping objects are order - insensitive like regular dictionaries .
- The reason for this equality is behavior is that it allows ` OrderedDict ` objects to be directly substituted anywhere a regular dictionary is used .
- Okay , so why did changing the order affect the lenght of the generated ` set ` object ? The answer is the lack of intransitive equality only . Since sets are " unordered " collections of unique elements , the order in which elements are inserted shouldn ' t matter. But in this case, it does matter. Let ' s break it down a bit ,
` ` ` py
>> > some_set = set ( )
>> > some_set . add ( dictionary ) # these are the mapping objects from the snippets above
>> > ordered_dict in some_set
True
>> > some_set . add ( ordered_dict )
>> > len ( some_set )
1
>> > another_ordered_dict in some_set
True
>> > some_set . add ( another_ordered_dict )
>> > len ( some_set )
1
>> > another_set = set ( )
>> > another_set . add ( ordered_dict )
>> > another_ordered_dict in another_set
False
>> > another_set . add ( another_ordered_dict )
>> > len ( another_set )
2
>> > dictionary in another_set
True
>> > another_set . add ( another_ordered_dict )
>> > len ( another_set )
2
` ` `
So the inconsistency is due to ` another_ordered_dict in another_set ` being False because ` ordered_dict ` was already present in ` another_set ` and as observed before , ` ordered_dict == another_ordered_dict ` is ` False ` .
2017-08-30 20:56:48 +02:00
- - -
2019-07-08 18:31:41 +02:00
### ▶ Keep trying?
2019-07-07 23:30:33 +02:00
< ! - - Example ID : b4349443 - e89f - 4 d25 - a109 - 82616 be9d41a - - - >
2017-08-30 20:56:48 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
def some_func ( ) :
try :
return ' from_try '
finally :
return ' from_finally '
2019-06-07 20:18:07 +02:00
def another_func ( ) :
for _ in range ( 3 ) :
try :
continue
finally :
print ( " Finally! " )
def one_more_func ( ) : # A gotcha!
try :
for i in range ( 3 ) :
try :
1 / i
except ZeroDivisionError :
# Let's throw it here and handle it outside for loop
raise ZeroDivisionError ( " A trivial divide by zero error " )
finally :
print ( " Iteration " , i )
break
except ZeroDivisionError as e :
print ( " Zero division error ocurred " , e )
2017-08-30 20:56:48 +02:00
` ` `
* * Output : * *
2019-06-07 20:18:07 +02:00
2017-08-30 20:56:48 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
>> > some_func ( )
' from_finally '
2019-06-07 20:18:07 +02:00
>> > another_func ( )
Finally !
Finally !
Finally !
>> > 1 / 0
Traceback ( most recent call last ) :
File " <stdin> " , line 1 , in < module >
ZeroDivisionError : division by zero
>> > one_more_func ( )
Iteration 0
2017-08-30 20:56:48 +02:00
` ` `
2018-01-22 14:07:35 +01:00
#### 💡 Explanation:
- When a ` return ` , ` break ` or ` continue ` statement is executed in the ` try ` suite of a " try…finally " statement , the ` finally ` clause is also executed ‘ on the way out .
- The return value of a function is determined by the last ` return ` statement executed . Since the ` finally ` clause always executes , a ` return ` statement executed in the ` finally ` clause will always be the last one executed .
2019-06-07 20:18:07 +02:00
- The caveat here is , if the finally clause executes a ` return ` or ` break ` statement , the temporarily saved exception is discarded .
2018-01-22 14:07:35 +01:00
- - -
2019-07-08 18:31:41 +02:00
### ▶ Deep down, we're all the same
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 8 f99a35f - 1736 - 43e2 - 920 d - 3 b78ec35da9b - - - >
2017-08-30 20:56:48 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
class WTF :
pass
2017-08-30 20:56:48 +02:00
` ` `
* * Output : * *
` ` ` py
2018-01-22 14:07:35 +01:00
>> > WTF ( ) == WTF ( ) # two different instances can't be equal
False
>> > WTF ( ) is WTF ( ) # identities are also different
False
>> > hash ( WTF ( ) ) == hash ( WTF ( ) ) # hashes _should_ be different as well
True
>> > id ( WTF ( ) ) == id ( WTF ( ) )
True
2017-08-30 20:56:48 +02:00
` ` `
#### 💡 Explanation:
2018-01-22 14:07:35 +01:00
* When ` id ` was called , Python created a ` WTF ` class object and passed it to the ` id ` function . The ` id ` function takes its ` id ` ( its memory location ) , and throws away the object . The object is destroyed .
* When we do this twice in succession , Python allocates the same memory location to this second object as well . Since ( in CPython ) ` id ` uses the memory location as the object id , the id of the two objects is the same .
* So , object ' s id is unique only for the lifetime of the object. After the object is destroyed, or before it is created, something else can have the same id.
* But why did the ` is ` operator evaluated to ` False ` ? Let ' s see with this snippet.
` ` ` py
class WTF ( object ) :
2018-08-28 10:54:54 +02:00
def __init__ ( self ) : print ( " I " )
def __del__ ( self ) : print ( " D " )
2018-01-22 14:07:35 +01:00
` ` `
* * Output : * *
` ` ` py
>> > WTF ( ) is WTF ( )
2018-08-28 10:54:54 +02:00
I
I
D
D
False
2018-01-22 14:07:35 +01:00
>> > id ( WTF ( ) ) == id ( WTF ( ) )
2018-08-28 10:54:54 +02:00
I
D
I
D
True
2018-01-22 14:07:35 +01:00
` ` `
As you may observe , the order in which the objects are destroyed is what made all the difference here .
2017-08-30 20:56:48 +02:00
2018-01-22 14:07:35 +01:00
- - -
2017-08-27 19:39:10 +02:00
2018-01-22 14:07:35 +01:00
### ▶ For what?
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 64 a9dccf - 5083 - 4 bc9 - 98 aa - 8 aeecde4f210 - - - >
2017-08-27 19:39:10 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
some_string = " wtf "
some_dict = { }
for i , some_dict [ i ] in enumerate ( some_string ) :
pass
` ` `
* * Output : * *
` ` ` py
>> > some_dict # An indexed dict is created.
{ 0 : ' w ' , 1 : ' t ' , 2 : ' f ' }
` ` `
#### 💡 Explanation:
* A ` for ` statement is defined in the [ Python grammar ] ( https : / / docs . python . org / 3 / reference / grammar . html ) as :
` ` `
for_stmt : ' for ' exprlist ' in ' testlist ' : ' suite [ ' else ' ' : ' suite ]
` ` `
Where ` exprlist ` is the assignment target . This means that the equivalent of ` { exprlist } = { next_value } ` is * * executed for each item * * in the iterable .
An interesting example that illustrates this :
` ` ` py
for i in range ( 4 ) :
print ( i )
i = 10
` ` `
* * Output : * *
` ` `
0
1
2
3
` ` `
Did you expect the loop to run just once ?
* * 💡 Explanation : * *
- The assignment statement ` i = 10 ` never affects the iterations of the loop because of the way for loops work in Python . Before the beginning of every iteration , the next item provided by the iterator ( ` range ( 4 ) ` this case ) is unpacked and assigned the target list variables ( ` i ` in this case ) .
* The ` enumerate ( some_string ) ` function yields a new value ` i ` ( A counter going up ) and a character from the ` some_string ` in each iteration . It then sets the ( just assigned ) ` i ` key of the dictionary ` some_dict ` to that character . The unrolling of the loop can be simplified as :
` ` ` py
>> > i , some_dict [ i ] = ( 0 , ' w ' )
>> > i , some_dict [ i ] = ( 1 , ' t ' )
>> > i , some_dict [ i ] = ( 2 , ' f ' )
>> > some_dict
` ` `
- - -
2019-07-08 18:31:41 +02:00
### ▶ Evaluation time discrepancy
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 6 aa11a4b - 4 cf1 - 467 a - b43a - 810731517e98 - - - >
2018-02-25 20:04:22 +01:00
1 \.
2018-01-22 14:07:35 +01:00
` ` ` py
array = [ 1 , 8 , 15 ]
g = ( x for x in array if array . count ( x ) > 0 )
array = [ 2 , 8 , 22 ]
` ` `
* * Output : * *
` ` ` py
>> > print ( list ( g ) )
[ 8 ]
` ` `
2018-02-25 20:04:22 +01:00
2 \.
2018-02-25 19:48:56 +01:00
` ` ` py
2018-02-25 20:04:22 +01:00
array_1 = [ 1 , 2 , 3 , 4 ]
g1 = ( x for x in array_1 )
array_1 = [ 1 , 2 , 3 , 4 , 5 ]
2018-02-25 19:48:56 +01:00
2018-02-25 20:04:22 +01:00
array_2 = [ 1 , 2 , 3 , 4 ]
g2 = ( x for x in array_2 )
array_2 [ : ] = [ 1 , 2 , 3 , 4 , 5 ]
2018-02-25 19:48:56 +01:00
` ` `
* * Output : * *
` ` ` py
2018-02-25 20:04:22 +01:00
>> > print ( list ( g1 ) )
2018-02-25 19:48:56 +01:00
[ 1 , 2 , 3 , 4 ]
>> > print ( list ( g2 ) )
[ 1 , 2 , 3 , 4 , 5 ]
` ` `
2019-06-07 20:53:33 +02:00
3 \.
` ` ` py
array_3 = [ 1 , 2 , 3 ]
array_4 = [ 10 , 20 , 30 ]
g = ( i + j for i in array_3 for j in array_4 )
array_3 = [ 4 , 5 , 6 ]
array_4 = [ 400 , 500 , 600 ]
` ` `
* * Output : * *
` ` ` py
>> > print ( list ( g ) )
[ 401 , 501 , 601 , 402 , 502 , 602 , 403 , 503 , 603 ]
` ` `
2018-02-25 19:48:56 +01:00
#### 💡 Explanation
2018-02-25 20:04:22 +01:00
- In a [ generator ] ( https : / / wiki . python . org / moin / Generators ) expression , the ` in ` clause is evaluated at declaration time , but the conditional clause is evaluated at runtime .
- So before runtime , ` array ` is re - assigned to the list ` [ 2 , 8 , 22 ] ` , and since out of ` 1 ` , ` 8 ` and ` 15 ` , only the count of ` 8 ` is greater than ` 0 ` , the generator only yields ` 8 ` .
- The differences in the output of ` g1 ` and ` g2 ` in the second part is due the way variables ` array_1 ` and ` array_2 ` are re - assigned values .
- In the first case , ` array_1 ` is binded to the new object ` [ 1 , 2 , 3 , 4 , 5 ] ` and since the ` in ` clause is evaluated at the declaration time it still refers to the old object ` [ 1 , 2 , 3 , 4 ] ` ( which is not destroyed ) .
- In the second case , the slice assignment to ` array_2 ` updates the same old object ` [ 1 , 2 , 3 , 4 ] ` to ` [ 1 , 2 , 3 , 4 , 5 ] ` . Hence both the ` g2 ` and ` array_2 ` still have reference to the same object ( which has now been updated to ` [ 1 , 2 , 3 , 4 , 5 ] ` ) .
2019-06-07 20:53:33 +02:00
- Okay , going by the logic discussed so far , shouldn ' t be the value of `list(g)` in the third snippet be `[11, 21, 31, 12, 22, 32, 13, 23, 33]`? (because `array_3` and `array_4` are going to behave just like `array_1`). The reason why (only) `array_4` values got updated is explained in [PEP-289](https://www.python.org/dev/peps/pep-0289/#the-details)
2019-06-07 23:23:53 +02:00
2019-06-07 20:53:33 +02:00
> Only the outermost for - expression is evaluated immediately , the other expressions are deferred until the generator is run .
2018-02-25 19:48:56 +01:00
- - -
2019-07-08 18:52:04 +02:00
### ▶ Messing around with `is` operator
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 230 fa2ac - ab36 - 4 ad1 - b675 - 5 f5a1c1a6217 - - - >
2018-01-22 14:07:35 +01:00
The following is a very famous example present all over the internet .
2019-06-08 19:55:37 +02:00
1 \.
2018-01-22 14:07:35 +01:00
` ` ` py
>> > a = 256
2017-08-27 19:39:10 +02:00
>> > b = 256
>> > a is b
True
>> > a = 257
>> > b = 257
>> > a is b
False
2019-06-08 19:55:37 +02:00
` ` `
2 \.
` ` ` py
>> > a = [ ]
>> > b = [ ]
>> > a is b
False
>> > a = tuple ( )
>> > b = tuple ( )
>> > a is b
True
` ` `
3 \.
* * Output ( < Python 3.7 ) * *
` ` `
>> > a , b = 257 , 257
True
>> > a = 257 ; b = 257
>> > a is b
True
` ` `
* * Output ( Python 3.7 ) * *
` ` `
>> > a , b = 257 , 257
False
2017-08-27 19:39:10 +02:00
>> > a = 257 ; b = 257
>> > a is b
True
` ` `
2017-08-30 20:29:05 +02:00
#### 💡 Explanation:
2017-08-27 19:39:10 +02:00
* * The difference between ` is ` and ` == ` * *
2018-01-22 10:05:42 +01:00
* ` is ` operator checks if both the operands refer to the same object ( i . e . , it checks if the identity of the operands matches or not ) .
2017-08-27 19:39:10 +02:00
* ` == ` operator compares the values of both the operands and checks if they are the same .
2017-09-01 10:13:44 +02:00
* So ` is ` is for reference equality and ` == ` is for value equality . An example to clear things up ,
` ` ` py
2019-06-08 19:55:37 +02:00
>> > class A : pass
>> > A ( ) is A ( ) # These are two empty objects at two different memory locations.
2017-09-01 10:13:44 +02:00
False
` ` `
2017-08-27 19:39:10 +02:00
* * ` 256 ` is an existing object but ` 257 ` isn ' t**
2017-08-30 20:29:05 +02:00
When you start up python the numbers from ` - 5 ` to ` 256 ` will be allocated . These numbers are used a lot , so it makes sense just to have them ready .
2017-08-27 19:39:10 +02:00
Quoting from https : / / docs . python . org / 3 / c - api / long . html
2017-08-30 20:29:05 +02:00
> The current implementation keeps an array of integer objects for all integers between - 5 and 256 , when you create an int in that range you just get back a reference to the existing object . So it should be possible to change the value of 1. I suspect the behavior of Python , in this case , is undefined . : - )
2017-08-27 19:39:10 +02:00
` ` ` py
>> > id ( 256 )
10922528
>> > a = 256
>> > b = 256
>> > id ( a )
10922528
>> > id ( b )
10922528
>> > id ( 257 )
140084850247312
>> > x = 257
>> > y = 257
>> > id ( x )
140084850247440
>> > id ( y )
140084850247344
` ` `
2018-01-22 10:05:42 +01:00
Here the interpreter isn ' t smart enough while executing `y = 257` to recognize that we ' ve already created an integer of the value ` 257 , ` and so it goes on to create another object in the memory .
2017-08-27 20:47:59 +02:00
2019-06-08 19:55:37 +02:00
Similar optimization applies to other * * immutable * * objects like empty tuples as well . Since lists are mutable , that ' s why `[] is []` will return `False` and `() is ()` will return `True`. This explains our second snippet. Let ' s move on to the third one ,
2018-01-22 10:05:42 +01:00
* * Both ` a ` and ` b ` refer to the same object when initialized with same value in the same line . * *
2017-08-27 19:39:10 +02:00
2019-06-08 19:55:37 +02:00
* * Output ( < Python 3.7 ) * *
2017-08-27 19:39:10 +02:00
` ` ` py
>> > a , b = 257 , 257
>> > id ( a )
140640774013296
>> > id ( b )
140640774013296
>> > a = 257
>> > b = 257
>> > id ( a )
140640774013392
>> > id ( b )
140640774013488
` ` `
2017-08-30 20:29:05 +02:00
* When a and b are set to ` 257 ` in the same line , the Python interpreter creates a new object , then references the second variable at the same time . If you do it on separate lines , it doesn ' t " know " that there ' s already ` 257 ` as an object .
* It ' s a compiler optimization and specifically applies to the interactive environment. When you enter two lines in a live interpreter, they ' re compiled separately , therefore optimized separately . If you were to try this example in a ` . py ` file , you would not see the same behavior , because the file is compiled all at once .
2019-06-08 19:55:37 +02:00
* Why didn ' t this work for Python 3.7? The abstract reason is because such compiler optimizations are implementation specific (i.e. may change with version, OS, etc). I ' m still figuring out what exact implementation change cause the issue , you can check out this [ issue ] ( https : / / github . com / satwikkansal / wtfpython / issues / 100 ) for updates .
2017-08-30 19:27:10 +02:00
2017-08-30 19:40:42 +02:00
- - -
2017-08-30 20:56:48 +02:00
2018-01-22 14:07:35 +01:00
### ▶ A tic-tac-toe where X wins in the first attempt!
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 69329249 - bdcb - 424 f - bd09 - cca2e6705a7a - - - >
2017-08-30 20:56:48 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
# Let's initialize a row
row = [ " " ] * 3 #row i['', '', '']
# Let's make a board
board = [ row ] * 3
2017-08-30 20:56:48 +02:00
` ` `
2018-01-22 14:07:35 +01:00
* * Output : * *
` ` ` py
>> > board
[ [ ' ' , ' ' , ' ' ] , [ ' ' , ' ' , ' ' ] , [ ' ' , ' ' , ' ' ] ]
>> > board [ 0 ]
[ ' ' , ' ' , ' ' ]
>> > board [ 0 ] [ 0 ]
' '
>> > board [ 0 ] [ 0 ] = " X "
>> > board
[ [ ' X ' , ' ' , ' ' ] , [ ' X ' , ' ' , ' ' ] , [ ' X ' , ' ' , ' ' ] ]
` ` `
2017-08-30 20:56:48 +02:00
2018-01-22 14:07:35 +01:00
We didn ' t assign 3 " X " s or did we?
#### 💡 Explanation:
When we initialize ` row ` variable , this visualization explains what happens in the memory
2019-06-07 20:18:07 +02:00
! [ image ] ( images / tic - tac - toe / after_row_initialized . png )
2018-01-22 14:07:35 +01:00
And when the ` board ` is initialized by multiplying the ` row ` , this is what happens inside the memory ( each of the elements ` board [ 0 ] ` , ` board [ 1 ] ` and ` board [ 2 ] ` is a reference to the same list referred by ` row ` )
2019-06-07 20:18:07 +02:00
! [ image ] ( images / tic - tac - toe / after_board_initialized . png )
2017-08-30 20:56:48 +02:00
2018-02-25 19:33:37 +01:00
We can avoid this scenario here by not using ` row ` variable to generate ` board ` . ( Asked in [ this ] ( https : / / github . com / satwikkansal / wtfpython / issues / 68 ) issue ) .
` ` ` py
2018-05-23 14:43:50 +02:00
>> > board = [ [ ' ' ] * 3 for _ in range ( 3 ) ]
2018-02-25 19:33:37 +01:00
>> > board [ 0 ] [ 0 ] = " X "
>> > board
[ [ ' X ' , ' ' , ' ' ] , [ ' ' , ' ' , ' ' ] , [ ' ' , ' ' , ' ' ] ]
` ` `
2017-08-30 20:56:48 +02:00
- - -
2018-01-22 14:07:35 +01:00
### ▶ The sticky output function
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 4 dc42f77 - 94 cb - 4 eb5 - a120 - 8203 d3ed7604 - - - >
2017-08-27 20:04:18 +02:00
` ` ` py
funcs = [ ]
results = [ ]
for x in range ( 7 ) :
def some_func ( ) :
return x
funcs . append ( some_func )
2018-12-05 20:45:59 +01:00
results . append ( some_func ( ) ) # note the function call here
2017-08-27 20:04:18 +02:00
funcs_results = [ func ( ) for func in funcs ]
` ` `
* * Output : * *
` ` ` py
>> > results
[ 0 , 1 , 2 , 3 , 4 , 5 , 6 ]
>> > funcs_results
[ 6 , 6 , 6 , 6 , 6 , 6 , 6 ]
` ` `
2017-08-30 19:27:10 +02:00
Even when the values of ` x ` were different in every iteration prior to appending ` some_func ` to ` funcs ` , all the functions return 6.
2017-08-27 20:04:18 +02:00
/ / OR
` ` ` py
>> > powers_of_x = [ lambda x : x * * i for i in range ( 10 ) ]
>> > [ f ( 2 ) for f in powers_of_x ]
[ 512 , 512 , 512 , 512 , 512 , 512 , 512 , 512 , 512 , 512 ]
` ` `
2017-09-02 21:57:23 +02:00
#### 💡 Explanation
2017-08-27 20:04:18 +02:00
2017-08-30 19:27:10 +02:00
- When defining a function inside a loop that uses the loop variable in its body , the loop function ' s closure is bound to the variable, not its value. So all of the functions use the latest value assigned to the variable for computation.
2017-08-27 20:04:18 +02:00
2017-08-31 13:31:41 +02:00
- To get the desired behavior you can pass in the loop variable as a named variable to the function . * * Why this works ? * * Because this will define the variable again within the function ' s scope.
2017-08-27 20:04:18 +02:00
2017-08-30 19:27:10 +02:00
` ` ` py
funcs = [ ]
for x in range ( 7 ) :
def some_func ( x = x ) :
return x
funcs . append ( some_func )
` ` `
2017-08-27 20:04:18 +02:00
2017-08-30 19:27:10 +02:00
* * Output : * *
` ` ` py
>> > funcs_results = [ func ( ) for func in funcs ]
>> > funcs_results
[ 0 , 1 , 2 , 3 , 4 , 5 , 6 ]
` ` `
2017-08-27 20:04:18 +02:00
2017-08-30 19:40:42 +02:00
- - -
2017-08-30 20:56:48 +02:00
2019-07-08 18:31:41 +02:00
### ▶ The chicken-egg problem
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 60730 dc2 - 0 d79 - 4416 - 8568 - 2 a63323b3ce8 - - - >
2019-06-08 20:37:02 +02:00
1 \.
` ` ` py
>> > isinstance ( 3 , int )
True
>> > isinstance ( type , object )
True
>> > isinstance ( object , type )
True
` ` `
2 \. So which is the ultimate , base class ? And wait , there ' s more to the confusion
` ` ` py
>> > class A : pass
>> > isinstance ( A , A )
False
>> > isinstance ( type , type )
True
>> > isinstance ( object , object )
True
` ` `
3 \.
` ` ` py
>> > issubclass ( int , object )
True
>> > issubclass ( type , object )
True
>> > issubclass ( object , type )
False
` ` `
#### 💡 Explanation
- ` type ` is a [ metaclass ] ( https : / / realpython . com / python - metaclasses / ) in Python .
- * * Everything * * is an ` object ` in Python , which includes classes as well as their objects ( instances ) .
- class ` type ` is the metaclass of class ` object ` , and every class ( including ` type ` ) has inherited directly or indirectly from ` object ` .
- There is no real base class among ` object ` and ` type ` . The confusion in the above snippets is arising because we ' re thinking these relationships (`issubclass` and `isinstance`) in terms of Python classes. The relationship between `object` and `type` can ' t be reproduced in pure python . To be more precise the following relationships can ' t be reproduced in pure Python,
+ class A is instance of class B , and class B is an instance of class A .
+ class A is an instance of itself .
- These relationships between ` object ` and ` type ` ( both being instances of eachother as well as themselves ) exist in Python because of " cheating " at implementation level .
- - -
2018-01-22 14:07:35 +01:00
### ▶ `is not ...` is not `is (not ...)`
2019-07-07 23:30:33 +02:00
< ! - - Example ID : b26fb1ed - 0 c7d - 4 b9c - 8 c6d - 94 a58a055c0d - - - >
2017-08-27 22:05:30 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
>> > ' something ' is not None
True
>> > ' something ' is ( not None )
False
2017-08-27 22:05:30 +02:00
` ` `
2018-01-22 14:07:35 +01:00
#### 💡 Explanation
2017-08-27 22:05:30 +02:00
2018-01-22 14:07:35 +01:00
- ` is not ` is a single binary operator , and has behavior different than using ` is ` and ` not ` separated .
- ` is not ` evaluates to ` False ` if the variables on either side of the operator point to the same object and ` True ` otherwise .
2017-08-27 22:05:30 +02:00
2018-01-22 14:07:35 +01:00
- - -
### ▶ The surprising comma
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 31 a819c8 - ed73 - 4 dcc - 84 eb - 91 bedbb51e58 - - - >
2017-08-27 22:05:30 +02:00
* * Output : * *
` ` ` py
2018-01-22 14:07:35 +01:00
>> > def f ( x , y , ) :
. . . print ( x , y )
. . .
>> > def g ( x = 4 , y = 5 , ) :
. . . print ( x , y )
. . .
>> > def h ( x , * * kwargs , ) :
File " <stdin> " , line 1
def h ( x , * * kwargs , ) :
^
SyntaxError : invalid syntax
>> > def h ( * args , ) :
File " <stdin> " , line 1
def h ( * args , ) :
^
SyntaxError : invalid syntax
2017-08-27 22:05:30 +02:00
` ` `
2018-01-22 14:07:35 +01:00
#### 💡 Explanation:
2017-08-27 22:05:30 +02:00
2018-01-22 14:07:35 +01:00
- Trailing comma is not always legal in formal parameters list of a Python function .
- In Python , the argument list is defined partially with leading commas and partially with trailing commas . This conflict causes situations where a comma is trapped in the middle , and no rule accepts it .
- * * Note : * * The trailing comma problem is [ fixed in Python 3.6 ] ( https : / / bugs . python . org / issue9232 ) . The remarks in [ this ] ( https : / / bugs . python . org / issue9232 #msg248399) post discuss in brief different usages of trailing commas in Python.
2017-08-27 22:05:30 +02:00
2018-01-22 14:07:35 +01:00
- - -
2017-08-27 22:05:30 +02:00
2019-07-08 18:31:41 +02:00
### ▶ Strings and the backslashes
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 6 ae622c3 - 6 d99 - 4041 - 9 b33 - 507 bd1a4407b - - - >
2018-01-22 14:07:35 +01:00
* * Output : * *
2019-06-07 23:23:53 +02:00
` ` ` py
>> > print ( " \" " )
"
2017-08-27 22:05:30 +02:00
2019-06-07 23:23:53 +02:00
>> > print ( r " \" " )
\"
>> > print ( r " \" )
File " <stdin> " , line 1
print ( r " \" )
^
2018-01-22 14:07:35 +01:00
SyntaxError : EOL while scanning string literal
2019-06-07 23:23:53 +02:00
>> > r ' \' ' == " \\ ' "
True
2018-01-22 14:07:35 +01:00
` ` `
2017-08-27 22:05:30 +02:00
2018-01-22 14:07:35 +01:00
#### 💡 Explanation
2017-08-27 22:05:30 +02:00
2019-06-07 23:23:53 +02:00
- In a normal python string , the backslash is used to escape characters that may have special meaning ( like single - quote , double - quote and the backslash itself ) .
` ` ` py
>> > ' wt \" f '
' wt " f '
` ` `
- In a raw string literal ( as indicated by the prefix ` r ` ) , the backslashes pass themselves as is along with the behavior of escaping the following character .
` ` ` py
>> > r ' wt \ " f ' == ' wt \\ " f '
True
>> > print ( repr ( r ' wt \ " f ' )
' wt \\ " f '
>> > print ( " \n " )
>> > print ( r " \\ n " )
' \\ \\ n '
` ` `
- This means when a parser encounters a backslash in a raw string , it expects another character following it . And in our case ( ` print ( r " \" )`), the backslash escaped the trailing quote, leaving the parser without a terminating quote (hence the `SyntaxError`). That ' s why backslashes don ' t work at the end of a raw string.
2017-08-27 22:05:30 +02:00
2017-08-30 19:40:42 +02:00
- - -
2017-08-30 20:56:48 +02:00
2018-01-22 14:07:35 +01:00
### ▶ not knot!
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 7034 deb1 - 7443 - 417 d - 94 ee - 29 a800524de8 - - - >
2017-08-27 20:47:59 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
x = True
y = False
2017-08-27 20:47:59 +02:00
` ` `
* * Output : * *
` ` ` py
2018-01-22 14:07:35 +01:00
>> > not x == y
True
>> > x == not y
File " <input> " , line 1
x == not y
^
SyntaxError : invalid syntax
2017-08-27 20:47:59 +02:00
` ` `
2017-08-30 20:29:05 +02:00
#### 💡 Explanation:
2017-08-27 20:47:59 +02:00
2018-01-22 14:07:35 +01:00
* Operator precedence affects how an expression is evaluated , and ` == ` operator has higher precedence than ` not ` operator in Python .
* So ` not x == y ` is equivalent to ` not ( x == y ) ` which is equivalent to ` not ( True == False ) ` finally evaluating to ` True ` .
* But ` x == not y ` raises a ` SyntaxError ` because it can be thought of being equivalent to ` ( x == not ) y ` and not ` x == ( not y ) ` which you might have expected at first sight .
* The parser expected the ` not ` token to be a part of the ` not in ` operator ( because both ` == ` and ` not in ` operators have the same precedence ) , but after not being able to find an ` in ` token following the ` not ` token , it raises a ` SyntaxError ` .
2017-08-27 20:47:59 +02:00
2017-08-30 19:40:42 +02:00
- - -
2017-08-30 20:56:48 +02:00
2018-01-22 14:07:35 +01:00
### ▶ Half triple-quoted strings
2019-07-07 23:30:33 +02:00
< ! - - Example ID : c55da3e2 - 1034 - 43 b9 - abeb - a7a970a2ad9e - - - >
2018-01-22 14:07:35 +01:00
* * Output : * *
2017-08-27 20:47:59 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
>> > print ( ' wtfpython ' ' ' )
wtfpython
>> > print ( " wtfpython " " " )
wtfpython
>> > # The following statements raise `SyntaxError`
>> > # print('''wtfpython')
>> > # print("""wtfpython")
2017-08-27 20:47:59 +02:00
` ` `
2018-01-22 14:07:35 +01:00
#### 💡 Explanation:
+ Python supports implicit [ string literal concatenation ] ( https : / / docs . python . org / 2 / reference / lexical_analysis . html #string-literal-concatenation), Example,
` ` `
>> > print ( " wtf " " python " )
wtfpython
>> > print ( " wtf " " " ) # or "wtf"""
wtf
` ` `
+ ` ''' ` and ` " " " ` are also string delimiters in Python which causes a SyntaxError because the Python interpreter was expecting a terminating triple quote as delimiter while scanning the currently encountered triple quoted string literal.
- - -
### ▶ Midnight time doesn't exist?
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 1 bce8294 - 5619 - 4 d70 - 8 ce3 - fe0bade690d1 - - - >
2018-01-22 14:07:35 +01:00
` ` ` py
from datetime import datetime
midnight = datetime ( 2018 , 1 , 1 , 0 , 0 )
midnight_time = midnight . time ( )
noon = datetime ( 2018 , 1 , 1 , 12 , 0 )
noon_time = noon . time ( )
if midnight_time :
print ( " Time at midnight is " , midnight_time )
if noon_time :
print ( " Time at noon is " , noon_time )
` ` `
* * Output : * *
` ` ` sh
( ' Time at noon is ' , datetime . time ( 12 , 0 ) )
2017-08-27 21:17:07 +02:00
` ` `
2018-01-22 14:07:35 +01:00
The midnight time is not printed .
2017-08-27 21:17:07 +02:00
2017-08-30 20:29:05 +02:00
#### 💡 Explanation:
2017-08-27 21:17:07 +02:00
2018-01-22 14:07:35 +01:00
Before Python 3.5 , the boolean value for ` datetime . time ` object was considered to be ` False ` if it represented midnight in UTC . It is error - prone when using the ` if obj : ` syntax to check if the ` obj ` is null or some equivalent of " empty. "
2017-08-27 21:17:07 +02:00
2018-01-22 14:07:35 +01:00
- - -
2017-08-27 20:47:59 +02:00
2018-01-22 14:07:35 +01:00
### ▶ What's wrong with booleans?
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 0 bba5fa7 - 9e6 d - 4 cd2 - 8 b94 - 952 d061af5dd - - - >
2018-01-22 14:07:35 +01:00
1 \.
` ` ` py
2019-07-07 18:47:33 +02:00
# A simple example to count the number of booleans and
2018-01-22 14:07:35 +01:00
# integers in an iterable of mixed data types.
mixed_list = [ False , 1.0 , " some_string " , 3 , True , [ ] , False ]
integers_found_so_far = 0
booleans_found_so_far = 0
2017-08-27 21:28:58 +02:00
2018-01-22 14:07:35 +01:00
for item in mixed_list :
if isinstance ( item , int ) :
integers_found_so_far + = 1
elif isinstance ( item , bool ) :
booleans_found_so_far + = 1
` ` `
* * Output : * *
` ` ` py
>> > integers_found_so_far
4
2018-12-05 20:50:17 +01:00
>> > booleans_found_so_far
0
2018-01-22 14:07:35 +01:00
` ` `
2019-07-07 18:47:33 +02:00
2 \.
2018-01-22 14:07:35 +01:00
` ` ` py
>> > some_bool = True
2019-07-07 18:47:33 +02:00
>> > " wtf " * some_bool
2018-01-22 14:07:35 +01:00
' wtf '
2018-01-26 06:50:39 +01:00
>> > some_bool = False
2019-07-07 18:47:33 +02:00
>> > " wtf " * some_bool
2018-01-22 14:07:35 +01:00
' '
` ` `
#### 💡 Explanation:
2019-07-07 18:47:33 +02:00
* ` bool ` is a subclass of ` int ` in Python
` ` ` py
>> > issubclass ( bool , int )
True
>> > issubclass ( int , bool )
False
` ` `
* And thus , ` True ` and ` False ` are instances of ` int `
2018-01-22 14:07:35 +01:00
` ` ` py
>> > isinstance ( True , int )
True
>> > isinstance ( False , int )
True
` ` `
* The integer value of ` True ` is ` 1 ` and that of ` False ` is ` 0 ` .
` ` ` py
2019-07-07 18:47:33 +02:00
>> > int ( True )
1
>> > int ( False )
0
2018-01-22 14:07:35 +01:00
` ` `
* See this StackOverflow [ answer ] ( https : / / stackoverflow . com / a / 8169049 / 4354153 ) for the rationale behind it .
2017-08-27 21:28:58 +02:00
2017-08-30 19:40:42 +02:00
- - -
2017-08-30 20:56:48 +02:00
2018-01-22 14:07:35 +01:00
### ▶ Class attributes and instance attributes
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 6 f332208 - 33 bd - 482 d - 8106 - 42863 b739ed9 - - - >
2017-08-31 15:06:48 +02:00
1 \.
` ` ` py
2018-01-22 14:07:35 +01:00
class A :
x = 1
class B ( A ) :
pass
class C ( A ) :
pass
2017-08-31 15:06:48 +02:00
` ` `
2018-12-06 19:54:45 +01:00
* * Output : * *
2017-08-31 15:06:48 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
>> > A . x , B . x , C . x
( 1 , 1 , 1 )
>> > B . x = 2
>> > A . x , B . x , C . x
( 1 , 2 , 1 )
>> > A . x = 3
>> > A . x , B . x , C . x
( 3 , 2 , 3 )
>> > a = A ( )
>> > a . x , A . x
( 3 , 3 )
>> > a . x + = 1
>> > a . x , A . x
( 4 , 3 )
2017-08-31 15:06:48 +02:00
` ` `
2017-08-27 21:17:07 +02:00
2017-08-31 15:06:48 +02:00
2 \.
` ` ` py
2018-01-22 14:07:35 +01:00
class SomeClass :
some_var = 15
some_list = [ 5 ]
another_list = [ 5 ]
def __init__ ( self , x ) :
self . some_var = x + 1
self . some_list = self . some_list + [ x ]
self . another_list + = [ x ]
2017-08-31 15:06:48 +02:00
` ` `
* * Output : * *
2018-01-22 14:07:35 +01:00
2017-08-31 15:06:48 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
>> > some_obj = SomeClass ( 420 )
>> > some_obj . some_list
[ 5 , 420 ]
>> > some_obj . another_list
[ 5 , 420 ]
>> > another_obj = SomeClass ( 111 )
>> > another_obj . some_list
[ 5 , 111 ]
>> > another_obj . another_list
[ 5 , 420 , 111 ]
>> > another_obj . another_list is SomeClass . another_list
True
>> > another_obj . another_list is some_obj . another_list
True
2017-08-31 15:06:48 +02:00
` ` `
#### 💡 Explanation:
2018-01-22 14:07:35 +01:00
* Class variables and variables in class instances are internally handled as dictionaries of a class object . If a variable name is not found in the dictionary of the current class , the parent classes are searched for it .
* The ` + = ` operator modifies the mutable object in - place without creating a new object . So changing the attribute of one instance affects the other instances and the class attribute as well .
2017-08-31 15:06:48 +02:00
2018-01-22 14:07:35 +01:00
- - -
2017-08-31 15:06:48 +02:00
2018-01-22 14:07:35 +01:00
### ▶ yielding None
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 5 a40c241 - 2 c30 - 40 d0 - 8 ba9 - cf7e097b3b53 - - - >
2018-01-22 14:07:35 +01:00
` ` ` py
some_iterable = ( ' a ' , ' b ' )
def some_func ( val ) :
return " something "
` ` `
* * Output : * *
` ` ` py
>> > [ x for x in some_iterable ]
[ ' a ' , ' b ' ]
>> > [ ( yield x ) for x in some_iterable ]
< generator object < listcomp > at 0x7f70b0a4ad58 >
>> > list ( [ ( yield x ) for x in some_iterable ] )
[ ' a ' , ' b ' ]
>> > list ( ( yield x ) for x in some_iterable )
[ ' a ' , None , ' b ' , None ]
>> > list ( some_func ( ( yield x ) ) for x in some_iterable )
[ ' a ' , ' something ' , ' b ' , ' something ' ]
` ` `
#### 💡 Explanation:
- Source and explanation can be found here : https : / / stackoverflow . com / questions / 32139885 / yield - in - list - comprehensions - and - generator - expressions
- Related bug report : http : / / bugs . python . org / issue10544
2017-08-31 15:06:48 +02:00
- - -
2018-01-22 14:07:35 +01:00
### ▶ Mutating the immutable!
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 15 a9e782 - 1695 - 43 ea - 817 a - a9208f6bb33d - - - >
2017-08-27 21:17:07 +02:00
` ` ` py
some_tuple = ( " A " , " tuple " , " with " , " values " )
another_tuple = ( [ 1 , 2 ] , [ 3 , 4 ] , [ 5 , 6 ] )
` ` `
* * Output : * *
` ` ` py
>> > some_tuple [ 2 ] = " change this "
TypeError : ' tuple ' object does not support item assignment
>> > another_tuple [ 2 ] . append ( 1000 ) #This throws no error
>> > another_tuple
( [ 1 , 2 ] , [ 3 , 4 ] , [ 5 , 6 , 1000 ] )
2017-08-30 09:14:41 +02:00
>> > another_tuple [ 2 ] + = [ 99 , 999 ]
TypeError : ' tuple ' object does not support item assignment
>> > another_tuple
( [ 1 , 2 ] , [ 3 , 4 ] , [ 5 , 6 , 1000 , 99 , 999 ] )
2017-08-27 21:17:07 +02:00
` ` `
2017-08-31 15:06:48 +02:00
But I thought tuples were immutable . . .
2017-08-30 20:29:05 +02:00
#### 💡 Explanation:
2017-08-27 21:17:07 +02:00
2017-08-30 09:14:41 +02:00
* Quoting from https : / / docs . python . org / 2 / reference / datamodel . html
2017-08-27 20:47:59 +02:00
2017-08-30 19:27:10 +02:00
> Immutable sequences
2017-08-30 20:29:05 +02:00
An object of an immutable sequence type cannot change once it is created . ( If the object contains references to other objects , these other objects may be mutable and may be modified ; however , the collection of objects directly referenced by an immutable object cannot change . )
2017-08-27 20:47:59 +02:00
2017-08-30 09:14:41 +02:00
* ` + = ` operator changes the list in - place . The item assignment doesn ' t work, but when the exception occurs, the item has already been changed in place.
2017-08-30 19:40:42 +02:00
- - -
2017-08-30 20:56:48 +02:00
2018-01-22 14:07:35 +01:00
### ▶ The disappearing variable from outer scope
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 7 f1e71b6 - cb3e - 44 fb - aa47 - 87 ef1b7decc8 - - - >
2017-08-27 22:05:30 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
e = 7
try :
raise Exception ( )
except Exception as e :
pass
` ` `
2017-08-27 22:05:30 +02:00
2018-01-22 14:07:35 +01:00
* * Output ( Python 2. x ) : * *
` ` ` py
>> > print ( e )
# prints nothing
2017-08-27 22:05:30 +02:00
` ` `
2018-01-22 14:07:35 +01:00
* * Output ( Python 3. x ) : * *
2017-08-27 22:05:30 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
>> > print ( e )
NameError : name ' e ' is not defined
2017-08-27 22:05:30 +02:00
` ` `
2017-08-30 20:29:05 +02:00
#### 💡 Explanation:
2018-01-22 14:07:35 +01:00
* Source : https : / / docs . python . org / 3 / reference / compound_stmts . html #except
When an exception has been assigned using ` as ` target , it is cleared at the end of the except clause . This is as if
2017-08-28 18:07:06 +02:00
` ` ` py
except E as N :
foo
` ` `
2018-01-22 10:05:42 +01:00
was translated into
2017-08-28 18:07:06 +02:00
` ` ` py
except E as N :
try :
foo
finally :
del N
` ` `
2017-08-30 20:29:05 +02:00
This means the exception must be assigned to a different name to be able to refer to it after the except clause . Exceptions are cleared because , with the traceback attached to them , they form a reference cycle with the stack frame , keeping all locals in that frame alive until the next garbage collection occurs .
2017-08-28 18:07:06 +02:00
2017-08-30 20:29:05 +02:00
* The clauses are not scoped in Python . Everything in the example is present in the same scope , and the variable ` e ` got removed due to the execution of the ` except ` clause . The same is not the case with functions which have their separate inner - scopes . The example below illustrates this :
2017-08-28 18:07:06 +02:00
2017-08-30 19:27:10 +02:00
` ` ` py
def f ( x ) :
del ( x )
print ( x )
2017-08-30 20:56:48 +02:00
2017-08-30 19:27:10 +02:00
x = 5
y = [ 5 , 4 , 3 ]
` ` `
* * Output : * *
` ` ` py
>> > f ( x )
UnboundLocalError : local variable ' x ' referenced before assignment
>> > f ( y )
UnboundLocalError : local variable ' x ' referenced before assignment
>> > x
5
>> > y
[ 5 , 4 , 3 ]
` ` `
2017-08-28 18:07:06 +02:00
* In Python 2. x the variable name ` e ` gets assigned to ` Exception ( ) ` instance , so when you try to print , it prints nothing .
2017-08-30 19:27:10 +02:00
* * Output ( Python 2. x ) : * *
` ` ` py
>> > e
Exception ( )
>> > print e
# Nothing is printed!
` ` `
2017-08-28 18:07:06 +02:00
2017-08-30 19:40:42 +02:00
- - -
2017-08-31 13:31:41 +02:00
2018-01-22 14:07:35 +01:00
### ▶ When True is actually False
2019-07-07 23:30:33 +02:00
< ! - - Example ID : c8317047 - 48 ae - 4306 - af5a - 04 c6d8b7c2b9 - - - >
2017-08-28 19:33:10 +02:00
` ` ` py
2017-08-31 19:40:47 +02:00
True = False
2017-08-28 19:33:10 +02:00
if True == False :
print ( " I ' ve lost faith in truth! " )
` ` `
* * Output : * *
` ` `
I ' ve lost faith in truth!
` ` `
2017-08-30 20:29:05 +02:00
#### 💡 Explanation:
2017-08-28 19:33:10 +02:00
2017-08-30 20:29:05 +02:00
- Initially , Python used to have no ` bool ` type ( people used 0 for false and non - zero value like 1 for true ) . Then they added ` True ` , ` False ` , and a ` bool ` type , but , for backward compatibility , they couldn ' t make `True` and `False` constants- they just were built-in variables.
2018-01-22 10:05:42 +01:00
- Python 3 was backward - incompatible , so it was now finally possible to fix that , and so this example won ' t work with Python 3.x!
2017-08-28 19:33:10 +02:00
2017-08-30 19:40:42 +02:00
- - -
2017-08-28 22:19:09 +02:00
2019-06-09 16:15:13 +02:00
### ▶ Lossy zip of iterators
2019-07-07 23:30:33 +02:00
< ! - - Example ID : c28ed154 - e59f - 4070 - 8 eb6 - 8967 a4acac6d - - - >
2019-06-09 16:15:13 +02:00
` ` ` py
>> > numbers = list ( range ( 7 ) )
>> > numbers
[ 0 , 1 , 2 , 3 , 4 , 5 , 6 ]
>> > first_three , remaining = numbers [ : 3 ] , numbers [ 3 : ]
>> > first_three , remaining
( [ 0 , 1 , 2 ] , [ 3 , 4 , 5 , 6 ] )
>> > numbers_iter = iter ( numbers )
>> > list ( zip ( numbers_iter , first_three ) )
[ ( 0 , 0 ) , ( 1 , 1 ) , ( 2 , 2 ) ]
# so far so good, let's zip the remaining
>> > list ( zip ( numbers_iter , remaining ) )
[ ( 4 , 3 ) , ( 5 , 4 ) , ( 6 , 5 ) ]
` ` `
Where did element ` 3 ` go from the ` numbers ` list ?
#### 💡 Explanation:
- From Python [ docs ] ( https : / / docs . python . org / 3.3 / library / functions . html #zip), here's an approximate implementation of zip function,
` ` ` py
def zip ( * iterables ) :
sentinel = object ( )
iterators = [ iter ( it ) for it in iterables ]
while iterators :
result = [ ]
for it in iterators :
elem = next ( it , sentinel )
if elem is sentinel :
return
result . append ( elem )
yield tuple ( result )
` ` `
- So the function takes in arbitrary number of itreable objects , adds each of their items to the ` result ` list by calling the ` next ` function on them , and stops whenever any of the iterable is exhausted .
- The caveat here is when any iterable is exhausted , the existing elements in the ` result ` list are discarded . That ' s what happened with `3` in the `numbers_iter`.
- The correct way to do the above using ` zip ` would be ,
` ` ` py
>> > numbers = list ( range ( 7 ) )
>> > numbers_iter = iter ( numbers )
>> > list ( zip ( first_three , numbers_iter ) )
[ ( 0 , 0 ) , ( 1 , 1 ) , ( 2 , 2 ) ]
>> > list ( zip ( remaining , numbers_iter ) )
[ ( 3 , 3 ) , ( 4 , 4 ) , ( 5 , 5 ) , ( 6 , 6 ) ]
` ` `
The first argument of zip should be the one with fewest elements .
- - -
2018-01-22 14:07:35 +01:00
### ▶ From filled to None in one instruction...
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 9 a0d5335 - efe5 - 4 eae - af44 - 584 d15233066 - - - >
2017-08-29 21:10:21 +02:00
` ` ` py
some_list = [ 1 , 2 , 3 ]
some_dict = {
" key_1 " : 1 ,
" key_2 " : 2 ,
" key_3 " : 3
}
some_list = some_list . append ( 4 )
some_dict = some_dict . update ( { " key_4 " : 4 } )
` ` `
* * Output : * *
` ` ` py
>> > print ( some_list )
None
>> > print ( some_dict )
None
` ` `
2017-08-30 20:29:05 +02:00
#### 💡 Explanation
2017-08-29 21:10:21 +02:00
2017-08-30 20:29:05 +02:00
Most methods that modify the items of sequence / mapping objects like ` list . append ` , ` dict . update ` , ` list . sort ` , etc . modify the objects in - place and return ` None ` . The rationale behind this is to improve performance by avoiding making a copy of the object if the operation can be done in - place ( Referred from [ here ] ( http : / / docs . python . org / 2 / faq / design . html #why-doesn-t-list-sort-return-the-sorted-list))
2017-08-29 21:10:21 +02:00
2017-08-30 19:40:42 +02:00
- - -
2017-08-29 21:10:21 +02:00
2019-07-08 18:31:41 +02:00
### ▶ Subclass relationships
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 9 f6d8cf0 - e1b5 - 42 d0 - 84 a0 - 4 cfab25a0bc0 - - - >
2017-08-29 23:42:17 +02:00
* * Output : * *
` ` ` py
2018-01-22 14:07:35 +01:00
>> > from collections import Hashable
>> > issubclass ( list , object )
2017-08-29 23:42:17 +02:00
True
2018-01-22 14:07:35 +01:00
>> > issubclass ( object , Hashable )
2017-09-01 10:22:18 +02:00
True
2018-01-22 14:07:35 +01:00
>> > issubclass ( list , Hashable )
2017-08-29 23:42:17 +02:00
False
` ` `
2018-01-22 14:07:35 +01:00
The Subclass relationships were expected to be transitive , right ? ( i . e . , if ` A ` is a subclass of ` B ` , and ` B ` is a subclass of ` C ` , the ` A ` _should_ a subclass of ` C ` )
2017-08-30 20:29:05 +02:00
#### 💡 Explanation:
2017-08-29 23:42:17 +02:00
2018-01-22 14:07:35 +01:00
* Subclass relationships are not necessarily transitive in Python . Anyone is allowed to define their own , arbitrary ` __subclasscheck__ ` in a metaclass .
* When ` issubclass ( cls , Hashable ) ` is called , it simply looks for non - Falsey " `__hash__` " method in ` cls ` or anything it inherits from .
* Since ` object ` is hashable , but ` list ` is non - hashable , it breaks the transitivity relation .
* More detailed explanation can be found [ here ] ( https : / / www . naftaliharris . com / blog / python - subclass - intransitivity / ) .
2017-08-29 23:42:17 +02:00
2017-08-30 19:40:42 +02:00
- - -
2017-08-29 21:10:21 +02:00
2019-07-08 18:31:41 +02:00
### ▶ The mysterious key type conversion
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 00 f42dd0 - b9ef - 408 d - 9e39 - 1 bc209ce3f36 - - - >
2017-08-30 11:22:11 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
class SomeClass ( str ) :
2017-08-30 11:22:11 +02:00
pass
2018-01-22 14:07:35 +01:00
some_dict = { ' s ' : 42 }
2017-08-30 11:22:11 +02:00
` ` `
* * Output : * *
` ` ` py
2018-01-22 14:07:35 +01:00
>> > type ( list ( some_dict . keys ( ) ) [ 0 ] )
str
>> > s = SomeClass ( ' s ' )
>> > some_dict [ s ] = 40
>> > some_dict # expected: Two different keys-value pairs
{ ' s ' : 40 }
>> > type ( list ( some_dict . keys ( ) ) [ 0 ] )
str
2017-08-30 11:22:11 +02:00
` ` `
2017-09-02 21:57:23 +02:00
#### 💡 Explanation:
2017-08-30 11:22:11 +02:00
2018-01-22 14:07:35 +01:00
* Both the object ` s ` and the string ` " s " ` hash to the same value because ` SomeClass ` inherits the ` __hash__ ` method of ` str ` class .
* ` SomeClass ( " s " ) == " s " ` evaluates to ` True ` because ` SomeClass ` also inherits ` __eq__ ` method from ` str ` class .
* Since both the objects hash to the same value and are equal , they are represented by the same key in the dictionary .
* For the desired behavior , we can redefine the ` __eq__ ` method in ` SomeClass `
` ` ` py
class SomeClass ( str ) :
def __eq__ ( self , other ) :
return (
type ( self ) is SomeClass
and type ( other ) is SomeClass
and super ( ) . __eq__ ( other )
)
2018-02-28 21:37:48 +01:00
2018-01-22 14:07:35 +01:00
# When we define a custom __eq__, Python stops automatically inheriting the
# __hash__ method, so we need to define it as well
__hash__ = str . __hash__
2017-08-30 09:02:13 +02:00
2018-01-22 14:07:35 +01:00
some_dict = { ' s ' : 42 }
` ` `
2017-08-30 09:02:13 +02:00
2018-01-22 14:07:35 +01:00
* * Output : * *
` ` ` py
>> > s = SomeClass ( ' s ' )
>> > some_dict [ s ] = 40
>> > some_dict
2018-06-14 21:18:50 +02:00
{ ' s ' : 40 , ' s ' : 42 }
2018-01-22 14:07:35 +01:00
>> > keys = list ( some_dict . keys ( ) )
>> > type ( keys [ 0 ] ) , type ( keys [ 1 ] )
( __main__ . SomeClass , str )
` ` `
2017-08-30 09:02:13 +02:00
2018-01-22 14:07:35 +01:00
- - -
### ▶ Let's see if you can guess this?
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 81 aa9fbe - bd63 - 4283 - b56d - 6 fdd14c9105e - - - >
2018-01-22 14:07:35 +01:00
` ` ` py
a , b = a [ b ] = { } , 5
2017-08-30 09:02:13 +02:00
` ` `
2018-01-22 14:07:35 +01:00
* * Output : * *
2017-08-30 09:02:13 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
>> > a
{ 5 : ( { . . . } , 5 ) }
2017-08-30 09:02:13 +02:00
` ` `
2018-01-22 14:07:35 +01:00
#### 💡 Explanation:
2017-08-30 09:02:13 +02:00
2018-01-22 14:07:35 +01:00
* According to [ Python language reference ] ( https : / / docs . python . org / 2 / reference / simple_stmts . html #assignment-statements), assignment statements have the form
2017-08-30 09:02:13 +02:00
` ` `
2018-01-22 14:07:35 +01:00
( target_list " = " ) + ( expression_list | yield_expression )
2017-08-30 09:02:13 +02:00
` ` `
2018-01-22 14:07:35 +01:00
and
2019-06-07 20:18:07 +02:00
> An assignment statement evaluates the expression list ( remember that this can be a single expression or a comma - separated list , the latter yielding a tuple ) and assigns the single resulting object to each of the target lists , from left to right .
2019-06-07 20:53:33 +02:00
2018-01-22 14:07:35 +01:00
* The ` + ` in ` ( target_list " = " ) + ` means there can be * * one or more * * target lists . In this case , target lists are ` a , b ` and ` a [ b ] ` ( note the expression list is exactly one , which in our case is ` { } , 5 ` ) .
* After the expression list is evaluated , it ' s value is unpacked to the target lists from **left to right**. So, in our case, first the ` {} , 5` tuple is unpacked to `a, b` and we now have `a = {} ` and `b = 5`.
* ` a ` is now assigned to ` { } ` which is a mutable object .
* The second target list is ` a [ b ] ` ( you may expect this to throw an error because both ` a ` and ` b ` have not been defined in the statements before . But remember , we just assigned ` a ` to ` { } ` and ` b ` to ` 5 ` ) .
* Now , we are setting the key ` 5 ` in the dictionary to the tuple ` ( { } , 5 ) ` creating a circular reference ( the ` { . . . } ` in the output refers to the same object that ` a ` is already referencing ) . Another simpler example of circular reference could be
2017-08-30 09:02:13 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
>> > some_list = some_list [ 0 ] = [ 0 ]
>> > some_list
[ [ . . . ] ]
>> > some_list [ 0 ]
[ [ . . . ] ]
>> > some_list is some_list [ 0 ]
True
>> > some_list [ 0 ] [ 0 ] [ 0 ] [ 0 ] [ 0 ] [ 0 ] == some_list
True
2017-08-30 09:02:13 +02:00
` ` `
2018-01-22 14:07:35 +01:00
Similar is the case in our example ( ` a [ b ] [ 0 ] ` is the same object as ` a ` )
2017-08-30 09:02:13 +02:00
2018-01-22 14:07:35 +01:00
* So to sum it up , you can break the example down to
2017-08-30 09:02:13 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
a , b = { } , 5
a [ b ] = a , b
2017-08-30 09:02:13 +02:00
` ` `
2018-01-22 14:07:35 +01:00
And the circular reference can be justified by the fact that ` a [ b ] [ 0 ] ` is the same object as ` a `
` ` ` py
>> > a [ b ] [ 0 ] is a
True
2017-08-30 09:02:13 +02:00
` ` `
2017-08-30 19:40:42 +02:00
- - -
2017-08-30 11:22:11 +02:00
2018-01-22 14:07:35 +01:00
- - -
2017-08-30 11:22:11 +02:00
2018-01-25 18:46:33 +01:00
## Section: Appearances are deceptive!
### ▶ Skipping lines?
2019-07-07 23:30:33 +02:00
< ! - - Example ID : d50bbde1 - fb9d - 4735 - 9633 - 3444 b9d2f417 - - - >
2018-01-25 18:46:33 +01:00
* * Output : * *
` ` ` py
>> > value = 11
>> > valuе = 32
>> > value
11
` ` `
Wut ?
* * Note : * * The easiest way to reproduce this is to simply copy the statements from the above snippet and paste them into your file / shell .
#### 💡 Explanation
Some non - Western characters look identical to letters in the English alphabet but are considered distinct by the interpreter .
` ` ` py
>> > ord ( ' е ' ) # cyrillic 'e' (Ye)
1077
>> > ord ( ' e ' ) # latin 'e', as used in English and typed using standard keyboard
101
>> > ' е ' == ' e '
False
>> > value = 42 # latin e
>> > valuе = 23 # cyrillic 'e', Python 2.x interpreter would raise a `SyntaxError` here
>> > value
42
` ` `
The built - in ` ord ( ) ` function returns a character ' s Unicode [code point](https://en.wikipedia.org/wiki/Code_point), and different code positions of Cyrillic ' e ' and Latin ' e ' justify the behavior of the above example.
- - -
2019-07-08 18:31:41 +02:00
### ▶ Teleportation
2019-07-07 23:30:33 +02:00
< ! - - Example ID : edafe923 - 0 c20 - 4315 - b6e1 - 0 c31abfc38f5 - - - >
2018-01-25 18:46:33 +01:00
` ` ` py
import numpy as np
def energy_send ( x ) :
# Initializing a numpy array
np . array ( [ float ( x ) ] )
def energy_receive ( ) :
# Return an empty numpy array
return np . empty ( ( ) , dtype = np . float ) . tolist ( )
` ` `
* * Output : * *
` ` ` py
>> > energy_send ( 123.456 )
>> > energy_receive ( )
123.456
` ` `
Where ' s the Nobel Prize?
#### 💡 Explanation:
* Notice that the numpy array created in the ` energy_send ` function is not returned , so that memory space is free to reallocate .
* ` numpy . empty ( ) ` returns the next free memory slot without reinitializing it . This memory spot just happens to be the same one that was just freed ( usually , but not always ) .
- - -
### ▶ Well, something is fishy...
2019-07-07 23:30:33 +02:00
< ! - - Example ID : cb6a37c5 - 74 f7 - 44 ca - b58c - 3 b902419b362 - - - >
2018-01-25 18:46:33 +01:00
` ` ` py
def square ( x ) :
"""
A simple function to calculate the square of a number by addition .
"""
sum_so_far = 0
for counter in range ( x ) :
sum_so_far = sum_so_far + x
return sum_so_far
` ` `
* * Output ( Python 2. x ) : * *
` ` ` py
>> > square ( 10 )
10
` ` `
Shouldn ' t that be 100?
* * Note : * * If you ' re not able to reproduce this, try running the file [mixed_tabs_and_spaces.py](/mixed_tabs_and_spaces.py) via the shell.
#### 💡 Explanation
* * * Don ' t mix tabs and spaces!** The character just preceding return is a " tab " , and the code is indented by multiple of " 4 spaces " elsewhere in the example.
* This is how Python handles tabs :
2019-06-07 20:18:07 +02:00
2018-01-25 18:46:33 +01:00
> First , tabs are replaced ( from left to right ) by one to eight spaces such that the total number of characters up to and including the replacement is a multiple of eight < . . . >
* So the " tab " at the last line of ` square ` function is replaced with eight spaces , and it gets into the loop .
* Python 3 is kind enough to throw an error for such cases automatically .
2018-02-28 21:37:48 +01:00
2018-01-25 18:46:33 +01:00
* * Output ( Python 3. x ) : * *
` ` ` py
TabError : inconsistent use of tabs and spaces in indentation
` ` `
- - -
- - -
2018-01-22 15:05:37 +01:00
## Section: Watch out for the landmines!
2017-08-30 11:22:11 +02:00
2018-01-22 14:07:35 +01:00
### ▶ Modifying a dictionary while iterating over it
2019-07-07 23:30:33 +02:00
< ! - - Example ID : b4e5cdfb - c3a8 - 4112 - bd38 - e2356d801c41 - - - >
2018-01-22 14:07:35 +01:00
` ` ` py
x = { 0 : None }
2017-08-30 20:56:48 +02:00
2018-01-22 14:07:35 +01:00
for i in x :
del x [ i ]
x [ i + 1 ] = None
print ( i )
2017-08-30 11:22:11 +02:00
` ` `
2018-01-22 14:07:35 +01:00
* * Output ( Python 2.7 - Python 3.5 ) : * *
2017-08-30 20:56:48 +02:00
` ` `
2018-01-22 14:07:35 +01:00
0
1
2
3
4
5
6
7
` ` `
Yes , it runs for exactly * * eight * * times and stops .
2017-08-30 20:56:48 +02:00
#### 💡 Explanation:
2018-01-22 14:07:35 +01:00
* Iteration over a dictionary that you edit at the same time is not supported .
* It runs eight times because that ' s the point at which the dictionary resizes to hold more keys (we have eight deletion entries, so a resize is needed). This is actually an implementation detail.
* How deleted keys are handled and when the resize occurs might be different for different Python implementations .
2019-06-07 21:11:58 +02:00
* So for Python versions other than Python 2.7 - Python 3.5 the count might be different from 8 ( but whatever the count is , it ' s going to be the same every time you run it). You can find some discussion around this [here](https://github.com/satwikkansal/wtfpython/issues/53) or in [this](https://stackoverflow.com/questions/44763802/bug-in-python-dict) StackOverflow thread.
2017-08-30 20:56:48 +02:00
- - -
2019-07-08 18:31:41 +02:00
### ▶ Stubborn `del` operation
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 777 ed4fd - 3 a2d - 466 f - 95e7 - c4058e61d78e - - - >
2017-09-01 19:17:05 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
class SomeClass :
def __del__ ( self ) :
print ( " Deleted! " )
2017-09-01 19:17:05 +02:00
` ` `
2017-10-17 10:24:59 +02:00
* * Output : * *
2018-01-22 14:07:35 +01:00
1 \.
2017-09-01 19:17:05 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
>> > x = SomeClass ( )
>> > y = x
>> > del x # this should print "Deleted!"
>> > del y
Deleted !
2017-09-01 19:17:05 +02:00
` ` `
2018-01-22 14:07:35 +01:00
Phew , deleted at last . You might have guessed what saved from ` __del__ ` being called in our first attempt to delete ` x ` . Let ' s add more twist to the example.
2017-09-04 19:46:31 +02:00
2 \.
` ` ` py
2018-01-22 14:07:35 +01:00
>> > x = SomeClass ( )
>> > y = x
>> > del x
>> > y # check if y exists
< __main__ . SomeClass instance at 0x7f98a1a67fc8 >
>> > del y # Like previously, this should print "Deleted!"
>> > globals ( ) # oh, it didn't. Let's check all our global variables and confirm
Deleted !
{ ' __builtins__ ' : < module ' __builtin__ ' ( built - in ) > , ' SomeClass ' : < class __main__ . SomeClass at 0x7f98a1a5f668 > , ' __package__ ' : None , ' __name__ ' : ' __main__ ' , ' __doc__ ' : None }
2017-09-04 19:46:31 +02:00
` ` `
2018-01-22 14:07:35 +01:00
Okay , now it ' s deleted :confused:
#### 💡 Explanation:
+ ` del x ` doesn ’ t directly call ` x . __del__ ( ) ` .
+ Whenever ` del x ` is encountered , Python decrements the reference count for ` x ` by one , and ` x . __del__ ( ) ` when x ’ s reference count reaches zero .
+ In the second output snippet , ` y . __del__ ( ) ` was not called because the previous statement ( ` >> > y ` ) in the interactive interpreter created another reference to the same object , thus preventing the reference count to reach zero when ` del y ` was encountered .
+ Calling ` globals ` caused the existing reference to be destroyed and hence we can see " Deleted! " being printed ( finally ! ) .
- - -
### ▶ Deleting a list item while iterating
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 4 cc52d4e - d42b - 4e09 - b25f - fbf5699b7d4e - - - >
2017-09-04 19:46:31 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
list_1 = [ 1 , 2 , 3 , 4 ]
list_2 = [ 1 , 2 , 3 , 4 ]
list_3 = [ 1 , 2 , 3 , 4 ]
list_4 = [ 1 , 2 , 3 , 4 ]
for idx , item in enumerate ( list_1 ) :
del item
for idx , item in enumerate ( list_2 ) :
list_2 . remove ( item )
for idx , item in enumerate ( list_3 [ : ] ) :
list_3 . remove ( item )
for idx , item in enumerate ( list_4 ) :
list_4 . pop ( idx )
2017-09-04 19:46:31 +02:00
` ` `
2018-01-22 14:07:35 +01:00
* * Output : * *
2018-01-17 14:53:13 +01:00
` ` ` py
2018-01-22 14:07:35 +01:00
>> > list_1
[ 1 , 2 , 3 , 4 ]
>> > list_2
[ 2 , 4 ]
>> > list_3
[ ]
>> > list_4
[ 2 , 4 ]
2018-01-17 14:53:13 +01:00
` ` `
2017-09-04 19:46:31 +02:00
2018-01-22 14:07:35 +01:00
Can you guess why the output is ` [ 2 , 4 ] ` ?
2017-09-01 19:17:05 +02:00
#### 💡 Explanation:
2018-01-22 14:07:35 +01:00
* It ' s never a good idea to change the object you ' re iterating over . The correct way to do so is to iterate over a copy of the object instead , and ` list_3 [ : ] ` does just that .
2017-09-01 19:17:05 +02:00
2018-01-22 14:07:35 +01:00
` ` ` py
>> > some_list = [ 1 , 2 , 3 , 4 ]
>> > id ( some_list )
139798789457608
>> > id ( some_list [ : ] ) # Notice that python creates new object for sliced list.
139798779601192
` ` `
2017-09-04 19:46:31 +02:00
2018-01-22 14:07:35 +01:00
* * Difference between ` del ` , ` remove ` , and ` pop ` : * *
* ` del var_name ` just removes the binding of the ` var_name ` from the local or global namespace ( That ' s why the `list_1` is unaffected).
* ` remove ` removes the first matching value , not a specific index , raises ` ValueError ` if the value is not found .
* ` pop ` removes the element at a specific index and returns it , raises ` IndexError ` if an invalid index is specified .
* * Why the output is ` [ 2 , 4 ] ` ? * *
- The list iteration is done index by index , and when we remove ` 1 ` from ` list_2 ` or ` list_4 ` , the contents of the lists are now ` [ 2 , 3 , 4 ] ` . The remaining elements are shifted down , i . e . , ` 2 ` is at index 0 , and ` 3 ` is at index 1. Since the next iteration is going to look at index 1 ( which is the ` 3 ` ) , the ` 2 ` gets skipped entirely . A similar thing will happen with every alternate element in the list sequence .
* Refer to this StackOverflow [ thread ] ( https : / / stackoverflow . com / questions / 45946228 / what - happens - when - you - try - to - delete - a - list - element - while - iterating - over - it ) explaining the example
* See also this nice StackOverflow [ thread ] ( https : / / stackoverflow . com / questions / 45877614 / how - to - change - all - the - dictionary - keys - in - a - for - loop - with - d - items ) for a similar example related to dictionaries in Python .
2017-09-01 19:17:05 +02:00
- - -
2018-01-22 14:07:35 +01:00
### ▶ Loop variables leaking out!
2019-07-07 23:30:33 +02:00
< ! - - Example ID : ccec7bf6 - 7679 - 4963 - 907 a - 1 cd8587be9ea - - - >
2018-01-07 10:35:00 +01:00
1 \.
` ` ` py
2018-01-22 14:07:35 +01:00
for x in range ( 7 ) :
if x == 6 :
print ( x , ' : for x inside loop ' )
print ( x , ' : x in global ' )
2018-01-07 10:35:00 +01:00
` ` `
* * Output : * *
2018-01-22 14:07:35 +01:00
` ` ` py
6 : for x inside loop
6 : x in global
2018-01-07 10:35:00 +01:00
` ` `
2017-08-31 15:06:48 +02:00
2018-01-22 14:07:35 +01:00
But ` x ` was never defined outside the scope of for loop . . .
2018-01-07 10:35:00 +01:00
2 \.
2017-08-30 11:22:11 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
# This time let's initialize x first
x = - 1
for x in range ( 7 ) :
if x == 6 :
print ( x , ' : for x inside loop ' )
print ( x , ' : x in global ' )
2017-08-30 11:22:11 +02:00
` ` `
2017-08-30 20:56:48 +02:00
* * Output : * *
` ` ` py
2018-01-22 14:07:35 +01:00
6 : for x inside loop
6 : x in global
` ` `
3 \.
2017-08-30 11:22:11 +02:00
2019-06-10 21:13:32 +02:00
* * Output ( Python 2. x ) : * *
2018-01-22 14:07:35 +01:00
` ` `
2019-06-10 21:13:32 +02:00
>> > x = 1
>> > print ( [ x for x in range ( 5 ) ] )
2018-01-22 14:07:35 +01:00
[ 0 , 1 , 2 , 3 , 4 ]
2019-06-10 21:13:32 +02:00
>> > print ( x , ' : x in global ' )
2018-01-22 14:07:35 +01:00
( 4 , ' : x in global ' )
` ` `
2019-06-10 21:13:32 +02:00
* * Output ( Python 3. x ) : * *
2018-01-22 14:07:35 +01:00
` ` `
2019-06-10 21:13:32 +02:00
>> > x = 1
>> > print ( [ x for x in range ( 5 ) ] )
2018-01-22 14:07:35 +01:00
[ 0 , 1 , 2 , 3 , 4 ]
2019-06-10 21:13:32 +02:00
>> > print ( x , ' : x in global ' )
2018-01-22 14:07:35 +01:00
1 : x in global
` ` `
2017-08-29 23:42:17 +02:00
2017-08-30 20:56:48 +02:00
#### 💡 Explanation:
2018-01-22 14:07:35 +01:00
- In Python , for - loops use the scope they exist in and leave their defined loop - variable behind . This also applies if we explicitly defined the for - loop variable in the global namespace before . In this case , it will rebind the existing variable .
2017-08-30 20:56:48 +02:00
2018-01-22 14:07:35 +01:00
- The differences in the output of Python 2. x and Python 3. x interpreters for list comprehension example can be explained by following change documented in [ What ’ s New In Python 3.0 ] ( https : / / docs . python . org / 3 / whatsnew / 3.0 . html ) documentation :
2018-01-07 07:57:16 +01:00
2018-01-22 14:07:35 +01:00
> " List comprehensions no longer support the syntactic form `[... for var in item1, item2, ...]`. Use `[... for var in (item1, item2, ...)]` instead. Also, note that list comprehensions have different semantics: they are closer to syntactic sugar for a generator expression inside a `list()` constructor, and in particular the loop control variables are no longer leaked into the surrounding scope. "
2018-01-07 07:57:16 +01:00
2018-01-22 14:07:35 +01:00
- - -
2018-01-07 07:57:16 +01:00
2018-01-22 14:07:35 +01:00
### ▶ Beware of default mutable arguments!
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 7 d42dade - e20d - 4 a7b - 9 ed7 - 16 fb58505fe9 - - - >
2018-01-22 14:07:35 +01:00
` ` ` py
def some_func ( default_arg = [ ] ) :
default_arg . append ( " some_string " )
return default_arg
2018-01-07 07:57:16 +01:00
` ` `
* * Output : * *
` ` ` py
2018-01-22 14:07:35 +01:00
>> > some_func ( )
[ ' some_string ' ]
>> > some_func ( )
[ ' some_string ' , ' some_string ' ]
>> > some_func ( [ ] )
[ ' some_string ' ]
>> > some_func ( )
[ ' some_string ' , ' some_string ' , ' some_string ' ]
2018-01-07 07:57:16 +01:00
` ` `
#### 💡 Explanation:
2018-01-22 14:07:35 +01:00
- The default mutable arguments of functions in Python aren ' t really initialized every time you call the function. Instead, the recently assigned value to them is used as the default value. When we explicitly passed `[]` to `some_func` as the argument, the default value of the `default_arg` variable was not used, so the function returned as expected.
2018-01-07 07:57:16 +01:00
2018-01-22 14:07:35 +01:00
` ` ` py
def some_func ( default_arg = [ ] ) :
default_arg . append ( " some_string " )
return default_arg
` ` `
* * Output : * *
` ` ` py
>> > some_func . __defaults__ #This will show the default argument values for the function
( [ ] , )
>> > some_func ( )
>> > some_func . __defaults__
( [ ' some_string ' ] , )
>> > some_func ( )
>> > some_func . __defaults__
( [ ' some_string ' , ' some_string ' ] , )
>> > some_func ( [ ] )
>> > some_func . __defaults__
( [ ' some_string ' , ' some_string ' ] , )
` ` `
2018-01-07 07:57:16 +01:00
2018-01-22 14:07:35 +01:00
- A common practice to avoid bugs due to mutable arguments is to assign ` None ` as the default value and later check if any value is passed to the function corresponding to that argument . Example :
2017-10-11 14:12:45 +02:00
2018-01-22 14:07:35 +01:00
` ` ` py
def some_func ( default_arg = None ) :
if not default_arg :
default_arg = [ ]
default_arg . append ( " some_string " )
return default_arg
` ` `
2017-10-11 14:12:45 +02:00
2018-01-22 14:07:35 +01:00
- - -
2017-10-11 14:12:45 +02:00
2018-01-22 14:07:35 +01:00
### ▶ Catching the Exceptions
2019-07-07 23:30:33 +02:00
< ! - - Example ID : b5ca5e6a - 47 b9 - 4 f69 - 9375 - cda0f8c6755d - - - >
2017-10-11 14:12:45 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
some_list = [ 1 , 2 , 3 ]
try :
# This should raise an ``IndexError``
print ( some_list [ 4 ] )
except IndexError , ValueError :
print ( " Caught! " )
2017-10-11 14:12:45 +02:00
2018-01-22 14:07:35 +01:00
try :
# This should raise a ``ValueError``
some_list . remove ( 4 )
except IndexError , ValueError :
print ( " Caught again! " )
` ` `
2017-10-11 14:12:45 +02:00
2018-01-22 14:07:35 +01:00
* * Output ( Python 2. x ) : * *
` ` ` py
Caught !
2017-10-11 14:12:45 +02:00
2018-01-22 14:07:35 +01:00
ValueError : list . remove ( x ) : x not in list
` ` `
2017-09-06 13:10:38 +02:00
2018-01-22 14:07:35 +01:00
* * Output ( Python 3. x ) : * *
2017-09-06 13:10:38 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
File " <input> " , line 3
except IndexError , ValueError :
2017-09-06 13:10:38 +02:00
^
SyntaxError : invalid syntax
` ` `
2018-01-22 14:07:35 +01:00
#### 💡 Explanation
* To add multiple Exceptions to the except clause , you need to pass them as parenthesized tuple as the first argument . The second argument is an optional name , which when supplied will bind the Exception instance that has been raised . Example ,
` ` ` py
some_list = [ 1 , 2 , 3 ]
try :
# This should raise a ``ValueError``
some_list . remove ( 4 )
except ( IndexError , ValueError ) , e :
print ( " Caught again! " )
print ( e )
` ` `
* * Output ( Python 2. x ) : * *
` ` `
Caught again !
list . remove ( x ) : x not in list
` ` `
* * Output ( Python 3. x ) : * *
` ` ` py
File " <input> " , line 4
except ( IndexError , ValueError ) , e :
^
IndentationError : unindent does not match any outer indentation level
` ` `
* Separating the exception from the variable with a comma is deprecated and does not work in Python 3 ; the correct way is to use ` as ` . Example ,
` ` ` py
some_list = [ 1 , 2 , 3 ]
try :
some_list . remove ( 4 )
2017-09-06 13:10:38 +02:00
2018-01-22 14:07:35 +01:00
except ( IndexError , ValueError ) as e :
print ( " Caught again! " )
print ( e )
` ` `
* * Output : * *
` ` `
Caught again !
list . remove ( x ) : x not in list
` ` `
2017-09-06 13:10:38 +02:00
- - -
2018-01-22 14:07:35 +01:00
### ▶ Same operands, different story!
2019-07-07 23:30:33 +02:00
< ! - - Example ID : ca052cdf - dd2d - 4105 - b936 - 65 c28adc18a0 - - - >
2018-01-22 14:07:35 +01:00
1 \.
2017-09-04 21:22:19 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
a = [ 1 , 2 , 3 , 4 ]
b = a
a = a + [ 5 , 6 , 7 , 8 ]
2017-09-04 21:22:19 +02:00
` ` `
2017-10-17 10:24:59 +02:00
* * Output : * *
2017-09-04 21:22:19 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
>> > a
[ 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 ]
>> > b
[ 1 , 2 , 3 , 4 ]
2017-09-04 21:22:19 +02:00
` ` `
2018-01-22 14:07:35 +01:00
2 \.
` ` ` py
a = [ 1 , 2 , 3 , 4 ]
b = a
a + = [ 5 , 6 , 7 , 8 ]
` ` `
2017-09-04 21:22:19 +02:00
2018-01-22 14:07:35 +01:00
* * Output : * *
` ` ` py
>> > a
[ 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 ]
>> > b
[ 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 ]
` ` `
2017-09-04 21:22:19 +02:00
2018-01-22 14:07:35 +01:00
#### 💡 Explanation:
2017-09-04 21:22:19 +02:00
2018-01-22 14:07:35 +01:00
* ` a + = b ` doesn ' t always behave the same way as `a = a + b`. Classes *may* implement the *`op=`* operators differently, and lists do this.
2017-09-04 21:22:19 +02:00
2018-01-22 14:07:35 +01:00
* The expression ` a = a + [ 5 , 6 , 7 , 8 ] ` generates a new list and sets ` a ` ' s reference to that new list, leaving `b` unchanged.
2017-09-04 21:22:19 +02:00
2018-02-28 21:37:48 +01:00
* The expression ` a + = [ 5 , 6 , 7 , 8 ] ` is actually mapped to an " extend " function that operates on the list such that ` a ` and ` b ` still point to the same list that has been modified in - place .
2017-09-04 21:22:19 +02:00
- - -
2018-01-22 14:07:35 +01:00
### ▶ The out of scope variable
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 75 c03015 - 7 be9 - 4289 - 9e22 - 4 f5fdda056f7 - - - >
2017-09-04 19:01:18 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
a = 1
def some_func ( ) :
return a
def another_func ( ) :
a + = 1
return a
2017-09-04 19:01:18 +02:00
` ` `
* * Output : * *
` ` ` py
2018-01-22 14:07:35 +01:00
>> > some_func ( )
1
>> > another_func ( )
UnboundLocalError : local variable ' a ' referenced before assignment
2017-09-04 19:01:18 +02:00
` ` `
#### 💡 Explanation:
2018-01-22 14:07:35 +01:00
* When you make an assignment to a variable in scope , it becomes local to that scope . So ` a ` becomes local to the scope of ` another_func ` , but it has not been initialized previously in the same scope which throws an error .
* Read [ this ] ( http : / / sebastianraschka . com / Articles / 2014 _python_scope_and_namespaces . html ) short but an awesome guide to learn more about how namespaces and scope resolution works in Python .
* To modify the outer scope variable ` a ` in ` another_func ` , use ` global ` keyword .
` ` ` py
def another_func ( )
global a
a + = 1
return a
` ` `
2018-02-28 21:37:48 +01:00
2018-01-22 14:07:35 +01:00
* * Output : * *
` ` ` py
>> > another_func ( )
2
` ` `
2017-09-04 19:01:18 +02:00
- - -
2018-01-22 14:07:35 +01:00
### ▶ Be careful with chained operations
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 07974979 - 9 c86 - 4720 - 80 bd - 467 aa19470d9 - - - >
2018-01-07 06:36:47 +01:00
` ` ` py
2018-01-22 14:07:35 +01:00
>> > ( False == False ) in [ False ] # makes sense
False
>> > False == ( False in [ False ] ) # makes sense
False
>> > False == False in [ False ] # now what?
2018-01-07 06:36:47 +01:00
True
2018-01-22 14:07:35 +01:00
>> > True is False == False
False
>> > False is False is False
2018-01-07 06:36:47 +01:00
True
2018-01-22 14:07:35 +01:00
>> > 1 > 0 < 1
True
>> > ( 1 > 0 ) < 1
False
>> > 1 > ( 0 < 1 )
2018-01-07 06:36:47 +01:00
False
` ` `
#### 💡 Explanation:
2018-01-22 14:07:35 +01:00
As per https : / / docs . python . org / 2 / reference / expressions . html #not-in
2018-01-07 06:36:47 +01:00
2018-01-22 14:07:35 +01:00
> Formally , if a , b , c , . . . , y , z are expressions and op1 , op2 , . . . , opN are comparison operators , then a op1 b op2 c . . . y opN z is equivalent to a op1 b and b op2 c and . . . y opN z , except that each expression is evaluated at most once .
While such behavior might seem silly to you in the above examples , it ' s fantastic with stuff like `a == b == c` and `0 <= x <= 100`.
* ` False is False is False ` is equivalent to ` ( False is False ) and ( False is False ) `
* ` True is False == False ` is equivalent to ` True is False and False == False ` and since the first part of the statement ( ` True is False ` ) evaluates to ` False ` , the overall expression evaluates to ` False ` .
* ` 1 > 0 < 1 ` is equivalent to ` 1 > 0 and 0 < 1 ` which evaluates to ` True ` .
* The expression ` ( 1 > 0 ) < 1 ` is equivalent to ` True < 1 ` and
` ` ` py
>> > int ( True )
1
>> > True + 1 #not relevant for this example, but just for fun
2
` ` `
So , ` 1 < 1 ` evaluates to ` False `
2018-01-07 06:36:47 +01:00
2018-01-22 14:07:35 +01:00
- - -
2018-01-07 06:55:34 +01:00
2018-01-22 14:07:35 +01:00
### ▶ Name resolution ignoring class scope
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 03 f73d96 - 151 c - 4929 - b0a8 - f74430788324 - - - >
2018-01-22 14:07:35 +01:00
1 \.
2018-01-07 06:55:34 +01:00
` ` ` py
2018-01-22 14:07:35 +01:00
x = 5
class SomeClass :
x = 17
y = ( x for i in range ( 10 ) )
2018-01-07 06:55:34 +01:00
` ` `
* * Output : * *
` ` ` py
2018-01-22 14:07:35 +01:00
>> > list ( SomeClass . y ) [ 0 ]
5
2018-01-07 06:55:34 +01:00
` ` `
2018-01-22 14:07:35 +01:00
2 \.
` ` ` py
x = 5
class SomeClass :
x = 17
y = [ x for i in range ( 10 ) ]
` ` `
2018-01-07 06:55:34 +01:00
2018-01-22 14:07:35 +01:00
* * Output ( Python 2. x ) : * *
` ` ` py
>> > SomeClass . y [ 0 ]
17
` ` `
2018-01-07 06:55:34 +01:00
2018-01-22 14:07:35 +01:00
* * Output ( Python 3. x ) : * *
` ` ` py
>> > SomeClass . y [ 0 ]
5
` ` `
#### 💡 Explanation
- Scopes nested inside class definition ignore names bound at the class level .
- A generator expression has its own scope .
- Starting from Python 3. X , list comprehensions also have their own scope .
2018-01-07 06:55:34 +01:00
2019-06-10 19:52:08 +02:00
- - -
2019-07-08 18:31:41 +02:00
### ▶ Needles in a Haystack
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 52 a199b1 - 989 a - 4 b28 - 8910 - dff562cebba9 - - - >
2018-01-22 14:07:35 +01:00
1 \.
2018-01-07 07:27:50 +01:00
` ` ` py
2018-01-22 14:07:35 +01:00
x , y = ( 0 , 1 ) if True else None , None
2018-01-07 07:27:50 +01:00
` ` `
* * Output : * *
2019-07-29 23:34:58 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
>> > x , y # expected (0, 1)
( ( 0 , 1 ) , None )
2018-01-07 07:27:50 +01:00
` ` `
2018-01-27 17:02:59 +01:00
Almost every Python programmer has faced a similar situation .
2018-01-22 14:07:35 +01:00
2 \.
` ` ` py
t = ( ' one ' , ' two ' )
for i in t :
print ( i )
t = ( ' one ' )
for i in t :
print ( i )
2018-01-07 07:27:50 +01:00
2018-01-22 14:07:35 +01:00
t = ( )
print ( t )
` ` `
2018-01-07 08:14:22 +01:00
* * Output : * *
` ` ` py
2018-01-22 14:07:35 +01:00
one
two
o
n
e
tuple ( )
2018-01-07 08:14:22 +01:00
` ` `
2019-06-10 21:13:32 +02:00
3 \.
` ` `
ten_words_list = [
" some " ,
" very " ,
" big " ,
" list " ,
" that "
" consists " ,
" of " ,
" exactly " ,
" ten " ,
" words "
]
` ` `
* * Output * *
` ` ` py
>> > len ( ten_words_list )
9
` ` `
4 \. Not asserting strongly enough
2019-06-10 19:52:08 +02:00
` ` ` py
a = " python "
b = " javascript "
` ` `
* * Output : * *
` ` ` py
# An assert statement with an assertion failure message.
>> > assert ( a == b , " Both languages are different " )
# No AssertionError is raised
` ` `
2018-01-07 08:14:22 +01:00
#### 💡 Explanation:
2018-01-22 14:07:35 +01:00
* For 1 , the correct statement for expected behavior is ` x , y = ( 0 , 1 ) if True else ( None , None ) ` .
* For 2 , the correct statement for expected behavior is ` t = ( ' one ' , ) ` or ` t = ' one ' , ` ( missing comma ) otherwise the interpreter considers ` t ` to be a ` str ` and iterates over it character by character .
* ` ( ) ` is a special token and denotes empty ` tuple ` .
2019-06-10 21:13:32 +02:00
* In 3 , as you might have already figured out , there ' s a missing comma after 5th element (` " that " `) in the list. So by implicit string literal concatenation,
` ` ` py
>> > ten_words_list
[ ' some ' , ' very ' , ' big ' , ' list ' , ' thatconsists ' , ' of ' , ' exactly ' , ' ten ' , ' words ' ]
` ` `
* No ` AssertionError ` was raised in 4 th snippet because instead of asserting the individual expression ` a == b ` , we ' re asserting entire tuple. The following snippet will clear things up,
2019-06-10 19:52:08 +02:00
` ` ` py
>> > a = " python "
>> > b = " javascript "
>> > assert a == b
Traceback ( most recent call last ) :
File " <stdin> " , line 1 , in < module >
AssertionError
>> > assert ( a == b , " Values are not equal " )
< stdin > : 1 : SyntaxWarning : assertion is always true , perhaps remove parentheses ?
>> > assert a == b , " Values are not equal "
Traceback ( most recent call last ) :
File " <stdin> " , line 1 , in < module >
AssertionError : Values aren not equal
` ` `
2018-01-07 08:14:22 +01:00
2018-01-07 10:35:00 +01:00
- - -
2019-09-13 04:03:27 +02:00
### ▶ Yielding from... return!
1 \.
` ` ` py
2019-09-13 11:31:45 +02:00
def some_func ( x ) :
if x == 3 :
return [ " wtf " ]
2019-09-13 04:03:27 +02:00
else :
2019-09-13 11:31:45 +02:00
yield from range ( x )
2019-09-13 04:03:27 +02:00
` ` `
* * Output : * *
` ` ` py
2019-09-13 11:31:45 +02:00
>> > list ( some_func ( 3 ) )
2019-09-13 04:03:27 +02:00
[ ]
` ` `
2019-09-13 11:31:45 +02:00
Where did the ` " wtf " ` go ? Is it due to some special effect of ` yield from ` ? Let ' s validate that,
2019-09-13 04:03:27 +02:00
2 \.
` ` ` py
2019-09-13 11:31:45 +02:00
def some_func ( x ) :
if x == 3 :
return [ " wtf " ]
2019-09-13 04:03:27 +02:00
else :
2019-09-13 11:31:45 +02:00
for i in range ( x ) :
yield i
2019-09-13 04:03:27 +02:00
` ` `
* * Output : * *
` ` ` py
2019-09-13 11:31:45 +02:00
>> > list ( some_func ( 3 ) )
2019-09-13 04:03:27 +02:00
[ ]
` ` `
2019-09-13 11:31:45 +02:00
Same result , that didn ' t work either.
2019-09-13 04:03:27 +02:00
#### 💡 Explanation:
2019-09-13 11:31:45 +02:00
+ From Python 3.3 onwards , it became possible to use ` return ` statement with values inside generators ( See [ PEP380 ] ( https : / / www . python . org / dev / peps / pep - 0380 / ) ) . The [ official docs ] ( https : / / www . python . org / dev / peps / pep - 0380 / #enhancements-to-stopiteration) say that,
2019-09-13 04:03:27 +02:00
2019-09-13 11:31:45 +02:00
> " ... `return expr` in a generator causes `StopIteration(expr)` to be raised upon exit from the generator. "
2019-09-13 04:03:27 +02:00
2019-09-13 11:31:45 +02:00
+ In case of ` some_func ( 3 ) ` , ` StopIteration ` is raised at the beginning because of ` return ` statement . The ` StopIteration ` exception is automatically catched inside the ` list ( . . . ) ` wrapper and the ` for ` loop . Therefore , the above two snippets result in an empty list .
2019-09-13 04:03:27 +02:00
2019-09-13 11:31:45 +02:00
+ To get ` [ " wtf " ] ` from the generator ` some_func ` we need to catch the ` StopIteration ` exception ,
` ` ` py
try :
next ( some_func ( 3 ) )
except StopIteration as e :
some_string = e . value
` ` `
` ` ` py
>> > some_string
[ " wtf " ]
` ` `
2019-09-13 04:03:27 +02:00
2019-06-10 21:13:32 +02:00
- - -
### ▶ Wild imports
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 83 deb561 - bd55 - 4461 - bb5e - 77 dd7f411e1c - - - >
2019-06-10 21:13:32 +02:00
` ` ` py
# File: module.py
def some_weird_name_func_ ( ) :
print ( " works! " )
def _another_weird_name_func ( ) :
print ( " works! " )
` ` `
* * Output * *
` ` ` py
>> > from module import *
>> > some_weird_name_func_ ( )
" works! "
>> > _another_weird_name_func ( )
Traceback ( most recent call last ) :
File " <stdin> " , line 1 , in < module >
NameError : name ' _another_weird_name_func ' is not defined
` ` `
#### 💡 Explanation:
2019-07-07 18:47:33 +02:00
- It if often adivsable to not use wildcard imports . The first obvious reason for this is In wildcard imports , the names with leading underscore are be imported . This may lead to errors in runtime .
2019-06-10 21:13:32 +02:00
- Had we used ` from . . . import a , b , c ` syntax , the above ` NameError ` won ' t have occurred.
` ` ` py
>> > from module import some_weird_name_func_ , _another_weird_name_func
>> > _another_weird_name_func ( )
works !
` ` `
- If you really want to use wildcard imports , then you ' d have to define the list `__all__` in your module that will contain a list of public objects that ' ll be available when we do wildcard imports .
` ` ` py
__all__ = [ ' _another_weird_name_func ' ]
def some_weird_name_func_ ( ) :
print ( " works! " )
2019-09-13 04:03:27 +02:00
2019-06-10 21:13:32 +02:00
def _another_weird_name_func ( ) :
print ( " works! " )
` ` `
* * Output * *
` ` ` py
>> > _another_weird_name_func ( )
" works! "
>> > some_weird_name_func_ ( )
Traceback ( most recent call last ) :
File " <stdin> " , line 1 , in < module >
NameError : name ' some_weird_name_func_ ' is not defined
` ` `
2019-09-13 04:03:27 +02:00
- - -
2018-01-22 14:07:35 +01:00
- - -
2018-01-07 10:35:00 +01:00
2018-01-25 18:46:33 +01:00
## Section: The Hidden treasures!
This section contains few of the lesser - known interesting things about Python that most beginners like me are unaware of ( well , not anymore ) .
2019-07-08 18:31:41 +02:00
### ▶ Okay Python, Can you make me fly?
2019-07-07 23:30:33 +02:00
< ! - - Example ID : a92f3645 - 1899 - 4 d50 - 9721 - 0031 be4aec3f - - - >
2018-01-25 18:46:33 +01:00
Well , here you go
` ` ` py
import antigravity
` ` `
* * Output : * *
Sshh . . It ' s a super secret.
#### 💡 Explanation:
+ ` antigravity ` module is one of the few easter eggs released by Python developers .
+ ` import antigravity ` opens up a web browser pointing to the [ classic XKCD comic ] ( http : / / xkcd . com / 353 / ) about Python .
2018-10-03 10:07:17 +02:00
+ Well , there ' s more to it. There ' s * * another easter egg inside the easter egg * * . If you look at the [ code ] ( https : / / github . com / python / cpython / blob / master / Lib / antigravity . py #L7-L17), there's a function defined that purports to implement the [XKCD's geohashing algorithm](https://xkcd.com/426/).
2018-01-25 18:46:33 +01:00
- - -
2019-07-08 18:31:41 +02:00
### ▶ `goto`, but why?
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 2 aff961e - 7 fa5 - 4986 - a18a - 9e5894 bd89fe - - - >
2018-01-25 18:46:33 +01:00
` ` ` py
from goto import goto , label
for i in range ( 9 ) :
for j in range ( 9 ) :
for k in range ( 9 ) :
print ( " I ' m trapped, please rescue! " )
if k == 2 :
goto . breakout # breaking out from a deeply nested loop
label . breakout
print ( " Freedom! " )
` ` `
* * Output ( Python 2.3 ) : * *
` ` ` py
I ' m trapped, please rescue!
I ' m trapped, please rescue!
Freedom !
` ` `
#### 💡 Explanation:
- A working version of ` goto ` in Python was [ announced ] ( https : / / mail . python . org / pipermail / python - announce - list / 2004 - April / 002982. html ) as an April Fool ' s joke on 1st April 2004.
- Current versions of Python do not have this module .
- Although it works , but please don ' t use it. Here ' s the [ reason ] ( https : / / docs . python . org / 3 / faq / design . html #why-is-there-no-goto) to why `goto` is not present in Python.
- - -
2019-07-08 18:31:41 +02:00
### ▶ Brace yourself!
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 5 c0c75f2 - ddd9 - 4 da3 - ba49 - c4be7ec39acf - - - >
2018-01-25 18:46:33 +01:00
If you are one of the people who doesn ' t like using whitespace in Python to denote scopes, you can use the C-style {} by importing,
` ` ` py
from __future__ import braces
` ` `
* * Output : * *
` ` ` py
File " some_file.py " , line 1
from __future__ import braces
SyntaxError : not a chance
` ` `
2019-06-09 14:04:50 +02:00
Braces ? No way ! If you think that ' s disappointing, use Java. Okay, another surprising thing, can you find where ' s the ` SyntaxError ` raised in ` __future__ ` module [ code ] ( https : / / github . com / python / cpython / blob / master / Lib / __future__ . py ) ?
2018-01-25 18:46:33 +01:00
#### 💡 Explanation:
2019-06-09 14:04:50 +02:00
+ The ` __future__ ` module is normally used to provide features from future versions of Python . The " future " in this specific context is however ironic .
2018-01-25 18:46:33 +01:00
+ This is an easter egg concerned with the community ' s feelings on this issue.
2019-06-09 14:04:50 +02:00
+ The code is actually present [ here ] ( https : / / github . com / python / cpython / blob / 025 eb98dc0c1dc27404df6c544fc2944e0fa9f3a / Python / future . c #L49) in `future.c` file.
+ When the CPython compiler encounters a [ future statement ] ( https : / / docs . python . org / 3.3 / reference / simple_stmts . html #future-statements), it first runs the appropriate code in `future.c` before treating it as a normal import statement.
2018-01-25 18:46:33 +01:00
- - -
2019-07-08 18:31:41 +02:00
### ▶ Let's meet Friendly Language Uncle For Life
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 6427 fae6 - e959 - 462 d - 85 da - ce4c94ce41be - - - >
2018-01-25 18:46:33 +01:00
* * Output ( Python 3. x ) * *
` ` ` py
>> > from __future__ import barry_as_FLUFL
>> > " Ruby " != " Python " # there's no doubt about it
File " some_file.py " , line 1
" Ruby " != " Python "
^
SyntaxError : invalid syntax
>> > " Ruby " < > " Python "
True
` ` `
There we go .
#### 💡 Explanation:
- This is relevant to [ PEP - 401 ] ( https : / / www . python . org / dev / peps / pep - 0401 / ) released on April 1 , 2009 ( now you know , what it means ) .
- Quoting from the PEP - 401
2019-06-07 20:18:07 +02:00
2018-01-25 18:46:33 +01:00
> Recognized that the != inequality operator in Python 3.0 was a horrible , finger pain inducing mistake , the FLUFL reinstates the < > diamond operator as the sole spelling .
- There were more things that Uncle Barry had to share in the PEP ; you can read them [ here ] ( https : / / www . python . org / dev / peps / pep - 0401 / ) .
2019-06-08 11:19:59 +02:00
- It works well on interactive environment , but it will raise a ` SyntaxError ` when you run via python file ( see this [ issue ] ( https : / / github . com / satwikkansal / wtfpython / issues / 94 ) ) . However , you can wrap the statement inside an ` eval ` or ` compile ` to get it working ,
` ` ` py
from __future__ import barry_as_FLUFL
print ( eval ( ' " Ruby " <> " Python " ' ) )
` ` `
2018-01-25 18:46:33 +01:00
- - -
2019-07-08 18:31:41 +02:00
### ▶ Even Python understands that love is complicated
2019-07-07 23:30:33 +02:00
< ! - - Example ID : b93cad9e - d341 - 45 d1 - 999 c - fcdce65bed25 - - - >
2018-01-25 18:46:33 +01:00
` ` ` py
import this
` ` `
Wait , what ' s **this**? `this` is love :heart:
* * Output : * *
` ` `
The Zen of Python , by Tim Peters
Beautiful is better than ugly .
Explicit is better than implicit .
Simple is better than complex .
Complex is better than complicated .
Flat is better than nested .
Sparse is better than dense .
Readability counts .
Special cases aren ' t special enough to break the rules.
Although practicality beats purity .
Errors should never pass silently .
Unless explicitly silenced .
In the face of ambiguity , refuse the temptation to guess .
There should be one - - and preferably only one - - obvious way to do it .
Although that way may not be obvious at first unless you ' re Dutch.
Now is better than never .
Although never is often better than * right * now .
If the implementation is hard to explain , it ' s a bad idea.
If the implementation is easy to explain , it may be a good idea .
Namespaces are one honking great idea - - let ' s do more of those!
` ` `
It ' s the Zen of Python!
` ` ` py
>> > love = this
>> > this is love
True
>> > love is True
False
>> > love is False
False
>> > love is not True or False
True
>> > love is not True or False ; love is love # Love is complicated
True
` ` `
#### 💡 Explanation:
* ` this ` module in Python is an easter egg for The Zen Of Python ( [ PEP 20 ] ( https : / / www . python . org / dev / peps / pep - 0020 ) ) .
* And if you think that ' s already interesting enough, check out the implementation of [this.py](https://hg.python.org/cpython/file/c3896275c0f6/Lib/this.py). Interestingly, the code for the Zen violates itself (and that ' s probably the only place where this happens ) .
* Regarding the statement ` love is not True or False ; love is love ` , ironic but it ' s self-explanatory.
- - -
### ▶ Yes, it exists!
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 4286 db3d - 1 ea7 - 47 c9 - 8 fb6 - a9a04cac6e49 - - - >
2018-01-25 18:46:33 +01:00
* * The ` else ` clause for loops . * * One typical example might be :
` ` ` py
def does_exists_num ( l , to_find ) :
for num in l :
if num == to_find :
print ( " Exists! " )
break
else :
print ( " Does not exist " )
` ` `
* * Output : * *
` ` ` py
>> > some_list = [ 1 , 2 , 3 , 4 , 5 ]
>> > does_exists_num ( some_list , 4 )
Exists !
>> > does_exists_num ( some_list , - 1 )
Does not exist
` ` `
* * The ` else ` clause in exception handling . * * An example ,
` ` ` py
try :
pass
except :
print ( " Exception occurred!!! " )
else :
print ( " Try block executed successfully... " )
` ` `
* * Output : * *
` ` ` py
Try block executed successfully . . .
` ` `
#### 💡 Explanation:
- The ` else ` clause after a loop is executed only when there ' s no explicit `break` after all the iterations.
- ` else ` clause after try block is also called " completion clause " as reaching the ` else ` clause in a ` try ` statement means that the try block actually completed successfully .
- - -
2019-07-08 18:31:41 +02:00
### ▶ Ellipsis
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 969 b7100 - ab3d - 4 a7d - ad7d - a6be16181b2b - - - >
2019-06-08 13:00:53 +02:00
` ` ` py
def some_func ( ) :
Ellipsis
` ` `
* * Output * *
` ` ` py
>> > some_func ( )
# No output, No Error
>> > SomeRandomString
Traceback ( most recent call last ) :
File " <stdin> " , line 1 , in < module >
NameError : name ' SomeRandomString ' is not defined
>> > Ellipsis
Ellipsis
` ` `
#### 💡 Explanation
- In Python , ` Ellipsis ` is a globally available builtin object which is equivalent to ` . . . ` .
` ` ` py
>> > . . .
Ellipsis
` ` `
- Eliipsis can be used for several purposes ,
+ As a placeholder for code that hasn ' t been written yet (just like `pass` statement)
+ In slicing syntax to represent the full slices in remaining direction
` ` ` py
>> > import numpy as np
>> > three_dimensional_array = np . arange ( 8 ) . reshape ( 2 , 2 , 2 )
array ( [
[
[ 0 , 1 ] ,
[ 2 , 3 ]
] ,
[
[ 4 , 5 ] ,
[ 6 , 7 ]
]
] )
` ` `
So our ` three_dimensional_array ` is an array of array of arrays . Let ' s say we want to print the second element (index `1`) of all the innermost arrays, we can use Ellipsis to bypass all the preceding dimensions
` ` ` py
>> > three_dimensional_array [ : , : , 1 ]
array ( [ [ 1 , 3 ] ,
[ 5 , 7 ] ] )
>> > three_dimensional_array [ . . . , 1 ] # using Ellipsis.
array ( [ [ 1 , 3 ] ,
[ 5 , 7 ] ] )
` ` `
Note : this will work for any number of dimensions . You can even select slice in first and last dimension and ignore the middle ones this way ( ` n_dimensional_array [ firs_dim_slice , . . . , last_dim_slice ] ` )
+ In [ type hinting ] ( https : / / docs . python . org / 3 / library / typing . html ) to indicate only a part of the type ( like ` ( Callable [ . . . , int ] ` or ` Tuple [ str , . . . ] ` ) )
+ You may also use Ellipsis as a default function argument ( in the cases when you want to differentiate between the " no argument passed " and " None value passed " scenarios ) .
2019-06-08 19:55:37 +02:00
- - -
2018-01-25 18:46:33 +01:00
2019-07-08 18:31:41 +02:00
### ▶ Inpinity
2019-07-07 23:30:33 +02:00
< ! - - Example ID : ff473ea8 - a3b1 - 4876 - a6f0 - 4378 aff790c1 - - - >
2018-01-25 18:46:33 +01:00
The spelling is intended . Please , don ' t submit a patch for this.
* * Output ( Python 3. x ) : * *
` ` ` py
>> > infinity = float ( ' infinity ' )
>> > hash ( infinity )
314159
>> > hash ( float ( ' -inf ' ) )
- 314159
` ` `
#### 💡 Explanation:
- Hash of infinity is 10 ⁵ x π .
- Interestingly , the hash of ` float ( ' -inf ' ) ` is " -10⁵ x π " in Python 3 , whereas " -10⁵ x e " in Python 2.
- - -
2019-07-08 18:31:41 +02:00
### ▶ Let's mangle
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 37146 d2d - 9e67 - 43 a9 - 8729 - 3 c17934b910c - - - >
2019-07-07 18:47:33 +02:00
1 \.
2018-01-25 18:46:33 +01:00
` ` ` py
class Yo ( object ) :
def __init__ ( self ) :
self . __honey = True
2019-06-10 21:13:32 +02:00
self . bro = True
2018-01-25 18:46:33 +01:00
` ` `
* * Output : * *
` ` ` py
2019-06-10 21:13:32 +02:00
>> > Yo ( ) . bro
2018-01-25 18:46:33 +01:00
True
>> > Yo ( ) . __honey
AttributeError : ' Yo ' object has no attribute ' __honey '
>> > Yo ( ) . _Yo__honey
True
` ` `
2019-07-07 18:47:33 +02:00
2 \.
` ` ` py
class Yo ( object ) :
def __init__ ( self ) :
# Let's try something symmetrical this time
self . __honey__ = True
self . bro = True
` ` `
* * Output : * *
` ` ` py
>> > Yo ( ) . bro
True
>> > Yo ( ) . _Yo__honey__
Traceback ( most recent call last ) :
File " <stdin> " , line 1 , in < module >
AttributeError : ' Yo ' object has no attribute ' _Yo__honey__ '
` ` `
2019-06-10 21:13:32 +02:00
Why did ` Yo ( ) . _Yo__honey ` work ?
2018-01-25 18:46:33 +01:00
2019-07-07 18:47:33 +02:00
2 \.
` ` ` py
_A__variable = " Some value "
class A ( object ) :
def some_func ( self ) :
return __variable # not initiatlized anywhere yet
` ` `
* * Output : * *
` ` ` py
Traceback ( most recent call last ) :
File " <stdin> " , line 1 , in < module >
AttributeError : ' A ' object has no attribute ' __variable '
>> > >> > A ( ) . some_func ( )
' Some value '
` ` `
3 \.
` ` ` py
` ` `
2018-01-25 18:46:33 +01:00
#### 💡 Explanation:
* [ Name Mangling ] ( https : / / en . wikipedia . org / wiki / Name_mangling ) is used to avoid naming collisions between different namespaces .
2019-07-07 18:47:33 +02:00
* In Python , the interpreter modifies ( mangles ) the class member names starting with ` __ ` ( double underscore a . k . a " dunder " ) and not ending with more than one trailing underscore by adding ` _NameOfTheClass ` in front .
* So , to access ` __honey ` attribute in first snippet , we had to append ` _Yo ` to the front which would prevent conflicts with the same name attribute defined in any other class .
* But then why didn ' t it work in the second snippet? Because name mangling excludes the names ending with double underscores.
* The third snippet was also a consequence of name mangling . The name ` __variable ` in the statement ` return __variable ` was mangled to ` _A__variable ` which also happens to be the name of the variable we declared in outer scope .
2018-01-25 18:46:33 +01:00
- - -
2019-06-10 21:13:32 +02:00
2018-01-25 18:46:33 +01:00
- - -
2018-02-28 21:37:48 +01:00
## Section: Miscellaneous
2018-01-07 10:35:00 +01:00
2018-01-22 14:07:35 +01:00
### ▶ `+=` is faster
2019-07-07 23:30:33 +02:00
< ! - - Example ID : bfd19c60 - a807 - 4 a26 - 9598 - 4912 b86ddb36 - - - >
2018-01-07 10:35:00 +01:00
` ` ` py
2018-01-22 14:07:35 +01:00
# using "+", three strings:
>> > timeit . timeit ( " s1 = s1 + s2 + s3 " , setup = " s1 = ' ' * 100000; s2 = ' ' * 100000; s3 = ' ' * 100000 " , number = 100 )
0.25748300552368164
# using "+=", three strings:
>> > timeit . timeit ( " s1 += s2 + s3 " , setup = " s1 = ' ' * 100000; s2 = ' ' * 100000; s3 = ' ' * 100000 " , number = 100 )
0.012188911437988281
2018-01-07 10:35:00 +01:00
` ` `
#### 💡 Explanation:
2018-01-22 14:07:35 +01:00
+ ` + = ` is faster than ` + ` for concatenating more than two strings because the first string ( example , ` s1 ` for ` s1 + = s2 + s3 ` ) is not destroyed while calculating the complete string .
2018-01-07 10:35:00 +01:00
2018-01-22 14:07:35 +01:00
- - -
2018-01-07 10:35:00 +01:00
2018-01-22 14:07:35 +01:00
### ▶ Let's make a giant string!
2019-07-07 23:30:33 +02:00
< ! - - Example ID : c7a07424 - 63 fe - 4504 - 9842 - 8 f3d334f28fc - - - >
2018-01-22 14:07:35 +01:00
` ` ` py
def add_string_with_plus ( iters ) :
s = " "
for i in range ( iters ) :
s + = " xyz "
assert len ( s ) == 3 * iters
2018-01-07 10:35:00 +01:00
2018-01-22 14:07:35 +01:00
def add_bytes_with_plus ( iters ) :
s = b " "
for i in range ( iters ) :
s + = b " xyz "
assert len ( s ) == 3 * iters
2018-01-11 09:42:44 +01:00
2018-01-22 14:07:35 +01:00
def add_string_with_format ( iters ) :
fs = " {} " * iters
s = fs . format ( * ( [ " xyz " ] * iters ) )
assert len ( s ) == 3 * iters
2018-01-11 09:42:44 +01:00
2018-01-22 14:07:35 +01:00
def add_string_with_join ( iters ) :
l = [ ]
for i in range ( iters ) :
l . append ( " xyz " )
s = " " . join ( l )
assert len ( s ) == 3 * iters
def convert_list_to_string ( l , iters ) :
s = " " . join ( l )
assert len ( s ) == 3 * iters
2018-01-11 09:42:44 +01:00
` ` `
* * Output : * *
2019-06-09 00:29:09 +02:00
2018-01-11 09:42:44 +01:00
` ` ` py
2019-06-09 00:29:09 +02:00
# Executed in ipython shell using %timeit for better readablity of results.
# You can also use the timeit module in normal python shell/scriptm=, example usage below
# timeit.timeit('add_string_with_plus(10000)', number=1000, globals=globals())
>> > NUM_ITERS = 1000
>> > % timeit - n1000 add_string_with_plus ( NUM_ITERS )
124 µs ± 4.73 µs per loop ( mean ± std . dev . of 7 runs , 100 loops each )
>> > % timeit - n1000 add_bytes_with_plus ( NUM_ITERS )
211 µs ± 10.5 µs per loop ( mean ± std . dev . of 7 runs , 1000 loops each )
>> > % timeit - n1000 add_string_with_format ( NUM_ITERS )
61 µs ± 2.18 µs per loop ( mean ± std . dev . of 7 runs , 1000 loops each )
>> > % timeit - n1000 add_string_with_join ( NUM_ITERS )
117 µs ± 3.21 µs per loop ( mean ± std . dev . of 7 runs , 1000 loops each )
>> > l = [ " xyz " ] * NUM_ITERS
>> > % timeit - n1000 convert_list_to_string ( l , NUM_ITERS )
10.1 µs ± 1.06 µs per loop ( mean ± std . dev . of 7 runs , 1000 loops each )
2018-01-11 09:42:44 +01:00
` ` `
2018-01-22 14:07:35 +01:00
Let ' s increase the number of iterations by a factor of 10.
2018-01-11 09:42:44 +01:00
` ` ` py
2019-06-09 00:29:09 +02:00
>> > NUM_ITERS = 10000
>> > % timeit - n1000 add_string_with_plus ( NUM_ITERS ) # Linear increase in execution time
1.26 ms ± 76.8 µs per loop ( mean ± std . dev . of 7 runs , 1000 loops each )
>> > % timeit - n1000 add_bytes_with_plus ( NUM_ITERS ) # Quadratic increase
6.82 ms ± 134 µs per loop ( mean ± std . dev . of 7 runs , 1000 loops each )
>> > % timeit - n1000 add_string_with_format ( NUM_ITERS ) # Linear increase
645 µs ± 24.5 µs per loop ( mean ± std . dev . of 7 runs , 1000 loops each )
>> > % timeit - n1000 add_string_with_join ( NUM_ITERS ) # Linear increase
1.17 ms ± 7.25 µs per loop ( mean ± std . dev . of 7 runs , 1000 loops each )
>> > l = [ " xyz " ] * NUM_ITERS
>> > % timeit - n1000 convert_list_to_string ( l , NUM_ITERS ) # Linear increase
86.3 µs ± 2 µs per loop ( mean ± std . dev . of 7 runs , 1000 loops each )
2018-01-11 09:42:44 +01:00
` ` `
2018-01-22 14:07:35 +01:00
#### 💡 Explanation
2019-06-09 00:29:09 +02:00
- You can read more about [ timeit ] ( https : / / docs . python . org / 3 / library / timeit . html ) or [ % timeit ] ( https : / / ipython . org / ipython - doc / dev / interactive / magics . html #magic-timeit) on these links. They are used to measure the execution time of code pieces.
2018-01-22 14:07:35 +01:00
- Don ' t use `+` for generating long strings — In Python, `str` is immutable, so the left and right strings have to be copied into the new string for every pair of concatenations. If you concatenate four strings of length 10, you ' ll be copying ( 10 + 10 ) + ( ( 10 + 10 ) + 10 ) + ( ( ( 10 + 10 ) + 10 ) + 10 ) = 90 characters instead of just 40 characters . Things get quadratically worse as the number and size of the string increases ( justified with the execution times of ` add_bytes_with_plus ` function )
2019-06-09 00:29:09 +02:00
- Therefore , it ' s advised to use `.format.` or ` % ` syntax (however, they are slightly slower than `+` for very short strings).
2018-01-22 14:07:35 +01:00
- Or better , if already you ' ve contents available in the form of an iterable object, then use ` ' ' .join(iterable_object)` which is much faster.
- ` add_string_with_plus ` didn ' t show a quadratic increase in execution time unlike `add_bytes_with_plus` because of the `+=` optimizations discussed in the previous example. Had the statement been `s = s + " x " + " y " + " z " ` instead of `s += " xyz " `, the increase would have been quadratic.
` ` ` py
def add_string_with_plus ( iters ) :
s = " "
for i in range ( iters ) :
s = s + " x " + " y " + " z "
assert len ( s ) == 3 * iters
2018-01-11 09:42:44 +01:00
2019-06-09 00:29:09 +02:00
>> > % timeit - n100 add_string_with_plus ( 1000 )
388 µs ± 22.4 µs per loop ( mean ± std . dev . of 7 runs , 1000 loops each )
>> > % timeit - n100 add_string_with_plus ( 10000 ) # Quadratic increase in execution time
9 ms ± 298 µs per loop ( mean ± std . dev . of 7 runs , 100 loops each )
2018-01-22 14:07:35 +01:00
` ` `
2019-06-10 19:52:08 +02:00
- So many ways to format and create a giant string are somewhat in contrast to the [ Zen of Python ] ( https : / / www . python . org / dev / peps / pep - 0020 / ) , according to which ,
> There should be one - - and preferably only one - - obvious way to do it .
2018-01-11 09:42:44 +01:00
- - -
2018-01-22 14:07:35 +01:00
### ▶ Explicit typecast of strings
2019-07-07 23:30:33 +02:00
< ! - - Example ID : 59 bee91a - 36e0 - 47 a4 - 8 c7d - aa89bf1d3976 - - - >
2017-09-01 00:19:20 +02:00
` ` ` py
2018-01-22 14:07:35 +01:00
a = float ( ' inf ' )
b = float ( ' nan ' )
c = float ( ' -iNf ' ) #These strings are case-insensitive
d = float ( ' nan ' )
2017-09-01 00:19:20 +02:00
` ` `
2017-09-01 00:19:20 +02:00
* * Output : * *
` ` ` py
>> > a
2018-01-22 14:07:35 +01:00
inf
>> > b
nan
>> > c
- inf
>> > float ( ' some_other_string ' )
ValueError : could not convert string to float : some_other_string
>> > a == - c #inf==inf
True
>> > None == None # None==None
True
>> > b == d #but nan!=nan
False
>> > 50 / a
0.0
>> > a / a
nan
>> > 23 + b
nan
2017-09-01 00:19:20 +02:00
` ` `
#### 💡 Explanation:
2018-01-22 14:07:35 +01:00
` ' inf ' ` and ` ' nan ' ` are special strings ( case - insensitive ) , which when explicitly typecasted to ` float ` type , are used to represent mathematical " infinity " and " not a number " respectively .
2017-09-01 00:19:20 +02:00
- - -
2018-01-22 14:07:35 +01:00
### ▶ Minor Ones
2019-07-07 23:30:33 +02:00
< ! - - Example ID : f885cb82 - f1e4 - 4 daa - 9 ff3 - 972 b14cb1324 - - - >
2017-08-30 16:17:09 +02:00
* ` join ( ) ` is a string operation instead of list operation . ( sort of counter - intuitive at first usage )
2018-02-28 21:37:48 +01:00
2017-08-30 20:29:05 +02:00
* * 💡 Explanation : * *
2017-08-31 19:21:34 +02:00
If ` join ( ) ` is a method on a string then it can operate on any iterable ( list , tuple , iterators ) . If it were a method on a list , it ' d have to be implemented separately by every type. Also, it doesn ' t make much sense to put a string - specific method on a generic ` list ` object API .
2017-08-28 20:04:00 +02:00
2017-08-30 16:17:09 +02:00
* Few weird looking but semantically correct statements :
+ ` [ ] = ( ) ` is a semantically correct statement ( unpacking an empty ` tuple ` into an empty ` list ` )
2017-09-06 12:38:42 +02:00
+ ` ' a ' [ 0 ] [ 0 ] [ 0 ] [ 0 ] [ 0 ] ` is also a semantically correct statement as strings are [ sequences ] ( https : / / docs . python . org / 3 / glossary . html #term-sequence)(iterables supporting element access using integer indices) in Python.
2017-09-02 14:09:50 +02:00
+ ` 3 - - 0 - - 5 == 8 ` and ` - - 5 == 5 ` are both semantically correct statements and evaluate to ` True ` .
2017-09-01 19:17:05 +02:00
2018-01-22 10:05:42 +01:00
* Given that ` a ` is a number , ` + + a ` and ` - - a ` are both valid Python statements but don ' t behave the same way as compared with similar statements in languages like C, C++ or Java.
2017-09-06 12:34:28 +02:00
` ` ` py
>> > a = 5
>> > a
5
>> > + + a
5
>> > - - a
5
` ` `
* * 💡 Explanation : * *
+ There is no ` + + ` operator in Python grammar . It is actually two ` + ` operators .
+ ` + + a ` parses as ` + ( + a ) ` which translates to ` a ` . Similarly , the output of the statement ` - - a ` can be justified .
+ This StackOverflow [ thread ] ( https : / / stackoverflow . com / questions / 3654830 / why - are - there - no - and - operators - in - python ) discusses the rationale behind the absence of increment and decrement operators in Python .
2017-08-30 20:29:05 +02:00
* Python uses 2 bytes for local variable storage in functions . In theory , this means that only 65536 variables can be defined in a function . However , python has a handy solution built in that can be used to store more than 2 ^ 16 variable names . The following code demonstrates what happens in the stack when more than 65536 local variables are defined ( Warning : This code prints around 2 ^ 18 lines of text , so be prepared ! ) :
2017-08-30 19:27:10 +02:00
` ` ` py
import dis
exec ( """
2018-01-27 17:33:32 +01:00
def f ( ) :
""" + """
2017-08-30 19:27:10 +02:00
""" .join([ " X " +str(x)+ " = " + str(x) for x in range(65539)]))
2017-08-30 20:29:05 +02:00
2017-08-30 19:27:10 +02:00
f ( )
2017-08-30 20:29:05 +02:00
2017-08-30 19:27:10 +02:00
print ( dis . dis ( f ) )
` ` `
2017-08-30 15:00:01 +02:00
2017-09-06 12:50:54 +02:00
* Multiple Python threads won ' t run your *Python code* concurrently (yes you heard it right!). It may seem intuitive to spawn several threads and let them execute your Python code concurrently, but, because of the [Global Interpreter Lock](https://wiki.python.org/moin/GlobalInterpreterLock) in Python, all you ' re doing is making your threads execute on the same core turn by turn . Python threads are good for IO - bound tasks , but to achieve actual parallelization in Python for CPU - bound tasks , you might want to use the Python [ multiprocessing ] ( https : / / docs . python . org / 2 / library / multiprocessing . html ) module .
2017-08-30 18:12:15 +02:00
2017-08-31 15:06:48 +02:00
* List slicing with out of the bounds indices throws no errors
2017-08-30 15:00:01 +02:00
` ` ` py
>> > some_list = [ 1 , 2 , 3 , 4 , 5 ]
>> > some_list [ 111 : ]
[ ]
` ` `
2018-01-07 08:14:22 +01:00
* ` int ( ' ١٢٣٤٥٦٧٨٩ ' ) ` returns ` 123456789 ` in Python 3. In Python , Decimal characters include digit characters , and all characters that can be used to form decimal - radix numbers , e . g . U + 0660 , ARABIC - INDIC DIGIT ZERO . Here ' s an [interesting story](http://chris.improbable.org/2014/8/25/adventures-in-unicode-digits/) related to this behavior of Python.
2018-01-07 10:35:00 +01:00
* ` ' abc ' . count ( ' ' ) == 4 ` . Here ' s an approximate implementation of `count` method, which would make the things more clear
` ` ` py
def count ( s , sub ) :
result = 0
for i in range ( len ( s ) + 1 - len ( sub ) ) :
result + = ( s [ i : i + len ( sub ) ] == sub )
return result
` ` `
The behavior is due to the matching of empty substring ( ` ' ' ` ) with slices of length 0 in the original string .
2017-08-30 19:40:42 +02:00
- - -
2017-08-30 19:27:10 +02:00
2017-08-26 20:31:11 +02:00
# Contributing
2018-01-22 10:12:35 +01:00
All patches are Welcome ! Please see [ CONTRIBUTING . md ] ( / CONTRIBUTING . md ) for further details .
2017-10-11 12:25:32 +02:00
2018-01-22 10:12:35 +01:00
For discussions , you can either create a new [ issue ] ( https : / / github . com / satwikkansal / wtfpython / issues / new ) or ping on the Gitter [ channel ] ( https : / / gitter . im / wtfpython / Lobby )
2017-08-26 20:31:11 +02:00
# Acknowledgements
2018-01-19 13:00:51 +01:00
The idea and design for this collection were initially inspired by Denys Dovhan ' s awesome project [wtfjs](https://github.com/denysdovhan/wtfjs). The overwhelming support by the community gave it the shape it is in right now.
2017-08-26 20:31:11 +02:00
2017-08-30 19:27:10 +02:00
#### Some nice Links!
2017-08-27 22:11:15 +02:00
* https : / / www . youtube . com / watch ? v = sH4XF6pKKmk
2017-08-28 22:19:09 +02:00
* https : / / www . reddit . com / r / Python / comments / 3 cu6ej / what_are_some_wtf_things_about_python
* https : / / sopython . com / wiki / Common_Gotchas_In_Python
2017-08-29 21:10:21 +02:00
* https : / / stackoverflow . com / questions / 530530 / python - 2 - x - gotchas - and - landmines
2018-01-19 13:00:51 +01:00
* https : / / stackoverflow . com / questions / 1011431 / common - pitfalls - in - python
* https : / / www . python . org / doc / humor /
2018-12-04 23:46:32 +01:00
* https : / / www . codementor . io / satwikkansal / python - practices - for - efficient - code - performance - memory - and - usability - aze6oiq65
2017-08-26 20:31:11 +02:00
# 🎓 License
[ ! [ CC 4.0 ] [ license - image ] ] [ license - url ]
& copy ; [ Satwik Kansal ] ( https : / / satwikkansal . xyz )
[ license - url ] : http : / / www . wtfpl . net
[ license - image ] : https : / / img . shields . io / badge / License - WTFPL % 202.0 - lightgrey . svg ? style = flat - square
2018-01-25 18:46:33 +01:00
## Help
If you have any wtfs , ideas or suggestions , please share .
2017-12-23 20:33:36 +01:00
2018-12-04 23:46:32 +01:00
## Surprise your geeky pythonist friends?
2018-01-14 18:25:41 +01:00
2018-12-04 23:46:32 +01:00
You can use these quick links to recommend wtfpython to your friends ,
2018-01-19 15:56:44 +01:00
2018-02-28 21:37:48 +01:00
[ Twitter ] ( https : / / twitter . com / intent / tweet ? url = https : / / github . com / satwikkansal / wtfpython & hastags = python , wtfpython )
| [ Linkedin ] ( https : / / www . linkedin . com / shareArticle ? url = https : / / github . com / satwikkansal & title = What % 20 the % 20 f * ck % 20 Python ! & summary = An % 20 interesting % 20 collection % 20 of % 20 subtle % 20 and % 20 tricky % 20 Python % 20 snippets . )
2017-12-23 20:33:36 +01:00
2018-01-19 15:56:44 +01:00
## Need a pdf version?
2018-01-22 10:05:42 +01:00
I ' ve received a few requests for the pdf version of wtfpython. You can add your details [here](https://satwikkansal.xyz/wtfpython-pdf/) to get the pdf as soon as it is finished.