When Python executes a script file, the whole file is parsed first. You can notice that when you introduce a syntax error somewhere: Regardless of where it is, it will prevent any line from executing.
So since Python parses the file first, literals can be loaded effectively into the memory. Since Python knows that these are constant, all variables that represent those constant values can point to the same object in memory. So the object is shared.
This works for ints and floats, but even for strings; even when there is a constant expression that needs to be evaluated first:
a = "foo"
b = "foo"
c = "fo" + "o"
print(a is b)
print(a is c)
Now in IDLE, the behavior is very different: As an interactive interpreter, IDLE executes every line separately. So a = 1.1
and b = 1.1
are executed in separated contexts which makes it impossible (or just very hard) to figure out that they both share the same constant literal value and could share the memory. So instead, the interpreter will allocate two different objects, which causes the identity check using is
to fail.
For small integers, the situation is a bit different. Because they are often used, CPython stores a set of integers (in the range between -5 and 256) statically and makes that every value of these points to the same int
object. That’s why you get a different result for small integers than for any other object. See also the following questions:
$python
at the command line and entered the 3 lines in standard interactive (REPL) mode, I believe the answer would have been False, as it is with 3.11. If the script had been loaded in an IDLE editor window and run, I believe the answer would have been True, as it is today. – Inexpugnable