Introduction to Computing Using Python Namespaces and Exceptions, revisited  Encapsulation in Functions  Global versus Local Namespaces  Exceptional.

Slides:



Advertisements
Similar presentations
15. Python - Modules A module allows you to logically organize your Python code. Grouping related code into a module makes the code easier to understand.
Advertisements

Setting the PYTHONPATH PYTHONPATH is where Python looks for modules it is told to import List of paths Add new path to the end with: setenv PYTHONPATH.
Perkovic, Chapter 7 Functions revisited Python Namespaces
Modular Programming With Functions
BBS514 Structured Programming (Yapısal Programlama)1 Functions and Structured Programming.
Introduction to Computing Using Python Exceptions (Oops! When things go wrong)  Errors and Exceptions  Syntax errors  State (execution) errors.
Tcl and Otcl Tutorial Part I Internet Computing KUT Youn-Hee Han.
Concepts when Python retrieves a variable’s value Namespaces – Namespaces store information about identifiers and the values to which they are bound –
 2002 Prentice Hall. All rights reserved. 1 Chapter 4 – Control Structures Outline 4.1 Introduction 4.2 Program Components in Python 4.3 Functions 4.4Module.
 2002 Prentice Hall. All rights reserved Exception-Handling Overview Exception handling –improves program clarity and modifiability by removing.
1 Python Chapter 2 © Samuel Marateck, After you install the compiler, an icon labeled IDLE (Python GUI) will appear on the screen. If you click.
Introduction to Computing Using Python Dictionary container class + modules  Container Class dict  Modules, revisited.
Python Programming, 2/e1 Python Programming: An Introduction to Computer Science Chapter 3 Computing with Numbers.
Exceptions COMPSCI 105 S Principles of Computer Science.
CSC 107 – Programming For Science. Announcements  Lectures may not cover all material from book  Material that is most difficult or challenging is focus.
CSC 110 Numeric data types [Reading: chapter 3] CSC 110 D 1.
CS 100: Roadmap to Computing Fall 2014 Lecture 01.
Builtins, namespaces, functions. There are objects that are predefined in Python Python built-ins When you use something without defining it, it means.
17. Python Exceptions Handling Python provides two very important features to handle any unexpected error in your Python programs and to add debugging.
General Programming Introduction to Computing Science and Programming I.
1 TAC2000/ Protocol Engineering and Application Research Laboratory (PEARL) MATH Functions in C Language.
1 CSC 221: Introduction to Programming Fall 2012 Functions & Modules  standard modules: math, random  Python documentation, help  user-defined functions,
Computer Science 111 Fundamentals of Programming I Basic Program Elements.
H3D API Training  Part 3.1: Python – Quick overview.
Modules and Decomposition UW CSE 190p Summer 2012 download examples from the calendar.
Procedures and Functions Computing Module 1. What is modular programming? Most programs written for companies will have thousands of lines of code. Most.
Input, Output, and Processing
Oct 15, 2007Sprenkle - CS1111 Objectives Creating your own functions.
Functions, Procedures, and Abstraction Dr. José M. Reyes Álamo.
By James Braunsberg. What are Modules? Modules are files containing Python definitions and statements (ex. name.py) A module’s definitions can be imported.
Functions Chapter 4 Python for Informatics: Exploring Information
Overview Intro to functions What are functions? Why use functions? Defining functions Calling functions Documenting functions Top-down design Variable.
Cem Sahin CS  There are two distinguishable kinds of errors: Python's Errors Syntax ErrorsExceptions.
MA/CS 375 Fall 2002 Lecture 3. Example 2 A is a matrix with 3 rows and 2 columns.
Department of Electrical and Computer Engineering Introduction to C++: Primitive Data Types, Libraries and Operations By Hector M Lugo-Cordero August 27,
Introduction to Computing Using Python Namespaces – Local and Global  The Purpose of Functions  Global versus Local Namespaces  The Program Stack 
Lecture 10: Modular Programming (functions) B Burlingame 13 April 2015.
Introduction to Computing Using Python Namespaces – Local and Global  The Purpose of Functions  Global versus Local Namespaces  The Program Stack 
MA/CS 375 Fall 2002 Lecture 2. Motivation for Suffering All This Math and Stuff Try the Actor demo from
Introduction to Computing Using Python Exceptions (Oops! When things go wrong)  Errors and Exceptions  Syntax errors  State (execution) errors.
12. MODULES Rocky K. C. Chang November 6, 2015 (Based on from Charles Dierbach. Introduction to Computer Science Using Python and William F. Punch and.
CSx 4091 – Python Programming Spring 2013 Lecture L2 – Introduction to Python Page 1 Help: To get help, type in the following in the interpreter: Welcome.
Introduction to Programming Lecture 6. Functions – Call by value – Call by reference Today's Lecture Includes.
6. FUNCTIONS Rocky K. C. Chang October 4, 2015 (Adapted from John Zelle’s slides)
CS 115 Lecture 5 Math library; building a project Taken from notes by Dr. Neil Moore.
FILES AND EXCEPTIONS Topics Introduction to File Input and Output Using Loops to Process Files Processing Records Exceptions.
Introduction to Computing Science and Programming I
Chapter 6 - Functions modular programming general function format
G. Pullaiah College of Engineering and Technology
CS 100: Roadmap to Computing
Exceptions and File Input
User-Defined Functions
Exceptions and files Taken from notes by Dr. Neil Moore
Object-Oriented Programming
CHAPTER FOUR Functions.
Topics Introduction to File Input and Output
Python’s Errors and Exceptions
Exceptions and files Taken from notes by Dr. Neil Moore
ERRORS AND EXCEPTIONS Errors:
(Oops! When things go wrong)
5. Functions Rocky K. C. Chang 30 October 2018
Namespaces – Local and Global
Rocky K. C. Chang 15 November 2018 (Based on Dierbach)
CHAPTER 3: String And Numeric Data In Python
Terminal-Based Programs
By Ryan Christen Errors and Exceptions.
Topics Introduction to File Input and Output
Namespaces – Local and Global
 A function is a named sequence of statement(s) that performs a computation. It contains  line of code(s) that are executed sequentially from top.
Presentation transcript:

Introduction to Computing Using Python Namespaces and Exceptions, revisited  Encapsulation in Functions  Global versus Local Namespaces  Exceptional Control Flow  Modules as Namespaces  Classes as Namespaces

Introduction to Computing Using Python The purpose of functions Wrapping code into functions has several desirable goals: Modularity: The complexity of developing a large program can be dealt with by breaking down the program into smaller, simpler, self-contained pieces. Each smaller piece (e.g., function) can be designed, implemented, tested, and debugged independently. Wrapping code into functions has several desirable goals: Modularity: The complexity of developing a large program can be dealt with by breaking down the program into smaller, simpler, self-contained pieces. Each smaller piece (e.g., function) can be designed, implemented, tested, and debugged independently. Code reuse: A fragment of code that is used multiple times in a program—or by multiple programs— should be packaged in a function. The program ends up being shorter, with a single function call replacing a code fragment, and clearer, because the name of the function can be more descriptive of the action being performed by the code fragment. Debugging also becomes easier because a bug in the code fragment will need to be fixed only once. Wrapping code into functions has several desirable goals: Modularity: The complexity of developing a large program can be dealt with by breaking down the program into smaller, simpler, self-contained pieces. Each smaller piece (e.g., function) can be designed, implemented, tested, and debugged independently. Code reuse: A fragment of code that is used multiple times in a program—or by multiple programs— should be packaged in a function. The program ends up being shorter, with a single function call replacing a code fragment, and clearer, because the name of the function can be more descriptive of the action being performed by the code fragment. Debugging also becomes easier because a bug in the code fragment will need to be fixed only once. Encapsulation: A function hides its implementation details from the user of the function; removing the implementation details from the developer’s radar makes her job easier.

Introduction to Computing Using Python Encapsulation through local variables >>> x Traceback (most recent call last): File " ", line 1, in x NameError: name 'x' is not defined >>> y Traceback (most recent call last): File " ", line 1, in y NameError: name 'y' is not defined >>> >>> x Traceback (most recent call last): File " ", line 1, in x NameError: name 'x' is not defined >>> y Traceback (most recent call last): File " ", line 1, in y NameError: name 'y' is not defined >>> def double(y): x=2 print('x = {}, y = {}'.format(x,y)) return x*y def double(y): x=2 print('x = {}, y = {}'.format(x,y)) return x*y Before executing function double(), variables x and y do not exist x and y exist only during the execution of function call double(5) ; they are said to be local variables of function double() >>> x Traceback (most recent call last): File " ", line 1, in x NameError: name 'x' is not defined >>> y Traceback (most recent call last): File " ", line 1, in y NameError: name 'y' is not defined >>> res = double(5) x = 2, y = 5 >>> >>> x Traceback (most recent call last): File " ", line 1, in x NameError: name 'x' is not defined >>> y Traceback (most recent call last): File " ", line 1, in y NameError: name 'y' is not defined >>> res = double(5) x = 2, y = 5 >>> >>> x Traceback (most recent call last): File " ", line 1, in x NameError: name 'x' is not defined >>> y Traceback (most recent call last): File " ", line 1, in y NameError: name 'y' is not defined >>> res = double(5) x = 2, y = 5 >>> x Traceback (most recent call last): File " ", line 1, in x NameError: name 'x' is not defined >>> y Traceback (most recent call last): File " ", line 1, in y NameError: name 'y' is not defined >>> x Traceback (most recent call last): File " ", line 1, in x NameError: name 'x' is not defined >>> y Traceback (most recent call last): File " ", line 1, in y NameError: name 'y' is not defined >>> res = double(5) x = 2, y = 5 >>> x Traceback (most recent call last): File " ", line 1, in x NameError: name 'x' is not defined >>> y Traceback (most recent call last): File " ", line 1, in y NameError: name 'y' is not defined After executing function double(), variables x and y still do not exist Encapsulation makes modularity and code reuse possible

>>> x, y = 20, 50 >>> >>> x, y = 20, 50 >>> Introduction to Computing Using Python Function call namespace def double(y): x=2 print('x = {}, y = {}'.format(x,y)) return x*y def double(y): x=2 print('x = {}, y = {}'.format(x,y)) return x*y Even during the execution of double(), local variables x and y are invisible outside of the function! >>> x Traceback (most recent call last): File " ", line 1, in x NameError: name 'x' is not defined >>> y Traceback (most recent call last): File " ", line 1, in y NameError: name 'y' is not defined >>> res = double(5) x = 2, y = 5 >>> x Traceback (most recent call last): File " ", line 1, in x NameError: name 'x' is not defined >>> y Traceback (most recent call last): File " ", line 1, in y NameError: name 'y' is not defined >>> x Traceback (most recent call last): File " ", line 1, in x NameError: name 'x' is not defined >>> y Traceback (most recent call last): File " ", line 1, in y NameError: name 'y' is not defined >>> res = double(5) x = 2, y = 5 >>> x Traceback (most recent call last): File " ", line 1, in x NameError: name 'x' is not defined >>> y Traceback (most recent call last): File " ", line 1, in y NameError: name 'y' is not defined >>> x, y = 20, 50 >>> res = double(5) x = 2, y = 5 >>> x, y (20, 50) >>> >>> x, y = 20, 50 >>> res = double(5) x = 2, y = 5 >>> x, y (20, 50) >>> y 5 5 Function call double(5) 50 y shell 20 x x 2 2 >>> x, y = 20, 50 >>> res = double(5) >>> x, y = 20, 50 >>> res = double(5) Every function call has a namespace in which local variables are stored How is it possible that the values of x and y do not interfere with each other?

Introduction to Computing Using Python Function call namespace def h(n): print('Start h') print(1/n) print(n) def g(n): print('Start g') h(n-1) print(n) def f(n): print('Start f') g(n-1) print(n) def h(n): print('Start h') print(1/n) print(n) def g(n): print('Start g') h(n-1) print(n) def f(n): print('Start f') g(n-1) print(n) >>> f(4) Start f >>> f(4) Start f >>> f(4) Start f Start g >>> f(4) Start f Start g >>> f(4) Start f Start g Start h >>> f(4) Start f Start g Start h >>> f(4) Start f Start g Start h 0.5 >>> f(4) Start f Start g Start h 0.5 >>> f(4) Start f Start g Start h >>> f(4) Start f Start g Start h >>> f(4) Start f Start g Start h >>> f(4) Start f Start g Start h >>> f(4) Start f Start g Start h >>> f(4) Start f Start g Start h n = 4 f(4) n = 4 f(4) n = 3 g(3) n = 3 g(3) n = 2 h(2) n = 2 h(2) n = 4 print('Start f') f(4) n = 4 print('Start f') f(4) n = 4 print('Start f') g(n-1) f(4) n = 4 print('Start f') g(n-1) f(4) n = 4 print('Start f') g(n-1) print(n) f(4) n = 4 print('Start f') g(n-1) print(n) f(4) n = 3 print('Start g') g(3) n = 3 print('Start g') g(3) n = 3 print('Start g') h(n-1) g(3) n = 3 print('Start g') h(n-1) g(3) n = 3 print('Start g') h(n-1) print(n) g(3) n = 3 print('Start g') h(n-1) print(n) g(3) n = 2 print('Start h') print(1/n) h(2) n = 2 print('Start h') print(1/n) h(2) n = 2 print('Start h') h(2) n = 2 print('Start h') h(2) n = 2 print('Start h') print(1/n) print(n) h(2) n = 2 print('Start h') print(1/n) print(n) h(2) Every function call has a namespace in which local variables are stored Note that there are several active values of n, one in each namespace; how are all the namespaces managed by Python? How does Python know which line to return to?

Introduction to Computing Using Python Program stack 1. def h(n): 2. print('Start h') 3. print(1/n) 4. print(n) def g(n): 7. print('Start g') 8. h(n-1) 9. print(n ) def f(n): 12. print('Start f') 13. g(n-1) 14. print(n) 1. def h(n): 2. print('Start h') 3. print(1/n) 4. print(n) def g(n): 7. print('Start g') 8. h(n-1) 9. print(n ) def f(n): 12. print('Start f') 13. g(n-1) 14. print(n) >>> f(4) Start f >>> f(4) Start f >>> f(4) Start f Start g >>> f(4) Start f Start g >>> f(4) Start f Start g Start h >>> f(4) Start f Start g Start h >>> f(4) Start f Start g Start h 0.5 >>> f(4) Start f Start g Start h 0.5 >>> f(4) Start f Start g Start h >>> f(4) Start f Start g Start h >>> f(4) Start f Start g Start h >>> f(4) Start f Start g Start h >>> f(4) Start f Start g Start h >>> f(4) Start f Start g Start h n = 4 f(4) n = 4 f(4) n = 3 g(3) n = 3 g(3) n = 2 h(2) n = 2 h(2) n = 4 print('Start f') f(4) n = 4 print('Start f') f(4) n = 4 print('Start f') g(n-1) f(4) n = 4 print('Start f') g(n-1) f(4) n = 4 print('Start f') g(n-1) print(n) f(4) n = 4 print('Start f') g(n-1) print(n) f(4) n = 3 print('Start g') g(3) n = 3 print('Start g') g(3) n = 3 print('Start g') h(n-1) g(3) n = 3 print('Start g') h(n-1) g(3) n = 3 print('Start g') h(n-1) print(n) g(3) n = 3 print('Start g') h(n-1) print(n) g(3) n = 2 print('Start h') print(1/n) h(2) n = 2 print('Start h') print(1/n) h(2) n = 2 print('Start h') h(2) n = 2 print('Start h') h(2) n = 2 print('Start h') print(1/n) print(n) h(2) n = 2 print('Start h') print(1/n) print(n) h(2) line = 14 n = 4 line = 9 n = 3 The system dedicates a chunk of memory to the program stack; its job is to remember the values defined in a function call Program stack … the statement to be executed after g(n-1) returns The system dedicates a chunk of memory to the program stack; its job is to remember the values defined in a function call and …

Introduction to Computing Using Python Scope and global vs. local namespace Every function call has a namespace associated with it. This namespace is where names defined during the execution of the function (e.g., local variables) live. The scope of these names (i.e., the space where they live) is the namespace of the function. In fact, every name in a Python program has a scope Whether the name is of a variable, function, class, … Outside of its scope, the name does not exist, and any reference to it will result in an error. Names assigned/defined in the interpreter shell or in a module and outside of any function are said to have global scope.

Introduction to Computing Using Python Scope and global vs. local namespace In fact, every name in a Python program has a scope Whether the name is of a variable, function, class, … Outside of its scope, the name does not exist, and any reference to it will result in an error. Names assigned/defined in the interpreter shell or in a module and outside of any function are said to have global scope. Their scope is the namespace associated with the shell or the whole module. Variables with global scope are referred to as global variables. x = 5 >>> x = 5 >>> x 5 >>> >>> x = 5 >>> x 5 >>> scope.py 5 5 shell x In fact, every name in a Python program has a scope Whether the name is of a variable, function, class, … Outside of its scope, the name does not exist, and any reference to it will result in an error. Names assigned/defined in the interpreter shell or in a module and outside of any function are said to have global scope. Their scope is the namespace associated with the shell module scope

Introduction to Computing Using Python Example: variable with local scope b Function call f(3) >>> === RESTART ==== >>> >>> === RESTART ==== >>> def f(b): # f has global scope, b has local scope a = 6 # this a has scope local to function call f() return a*b # this a is the local a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope a = 6 # this a has scope local to function call f() return a*b # this a is the local a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 a shell 0 0 def f(b): # f has global scope, b has local scope a = 6 # this a has scope local to function call f() return a*b # this a is the local a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope a = 6 # this a has scope local to function call f() return a*b # this a is the local a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope a = 6 # this a has scope local to function call f() return a*b # this a is the local a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope a = 6 # this a has scope local to function call f() return a*b # this a is the local a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope a = 6 # this a has scope local to function call f() return a*b # this a is the local a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope a = 6 # this a has scope local to function call f() return a*b # this a is the local a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope a = 6 # this a has scope local to function call f() return a*b # this a is the local a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope a = 6 # this a has scope local to function call f() return a*b # this a is the local a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still a 6 6 >>> === RESTART ==== >>> f(3) = 18 >>> === RESTART ==== >>> f(3) = 18 >>> === RESTART ==== >>> f(3) = 18 a is 0 >>> >>> === RESTART ==== >>> f(3) = 18 a is 0 >>>

def f(b): # f has global scope, b has local scope return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 Introduction to Computing Using Python Example: variable with global scope b Function call f(3) >>> === RESTART ==== >>> >>> === RESTART ==== >>> def f(b): # f has global scope, b has local scope return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 a shell >>> === RESTART ==== >>> f(3) = 0 >>> === RESTART ==== >>> f(3) = 0 >>> === RESTART ==== >>> f(3) = 0 a is 0 >>> >>> === RESTART ==== >>> f(3) = 0 a is 0 >>>

Introduction to Computing Using Python How Python evaluates names How does the Python interpreter decide whether to evaluate a name (of a variable, function, etc.) as a local or as a global name? Whenever the Python interpreter needs to evaluate a name, it searches for the name definition in this order: 1.First the enclosing function call namespace 2.Then the global (module) namespace 3.Finally the namespace of module builtins b Function call f(3) 3 3 a global namespace printf module builtins def f(b): # f has global scope, b has local scope return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still 0 def f(b): # f has global scope, b has local scope return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a is still f f() printf()

def f(b): global a # all references to a in f() are to the global a a = 6 # global a is changed return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a has been changed to 6 def f(b): global a # all references to a in f() are to the global a a = 6 # global a is changed return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a has been changed to 6 def f(b): global a # all references to a in f() are to the global a a = 6 # global a is changed return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a has been changed to 6 def f(b): global a # all references to a in f() are to the global a a = 6 # global a is changed return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a has been changed to 6 def f(b): global a # all references to a in f() are to the global a a = 6 # global a is changed return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a has been changed to 6 def f(b): global a # all references to a in f() are to the global a a = 6 # global a is changed return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a has been changed to 6 def f(b): global a # all references to a in f() are to the global a a = 6 # global a is changed return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a has been changed to 6 def f(b): global a # all references to a in f() are to the global a a = 6 # global a is changed return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a has been changed to 6 Introduction to Computing Using Python Modifying a global variable inside a function b Function call f(3) >>> === RESTART ==== >>> >>> === RESTART ==== >>> def f(b): global a # all references to a in f() are to the global a a = 6 # global a is changed return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a has been changed to 6 def f(b): global a # all references to a in f() are to the global a a = 6 # global a is changed return a*b # this a is the global a a = 0 # this a has global scope print('f(3) = {}'.format(f(3))) print('a is {}'.format(a)) # global a has been changed to 6 a shell >>> === RESTART ==== >>> f(3) = 18 >>> === RESTART ==== >>> f(3) = 18 >>> === RESTART ==== >>> f(3) = 18 a is 6 >>> >>> === RESTART ==== >>> f(3) = 18 a is 6 >>> 6 6

Introduction to Computing Using Python Exceptions, revisited The reason behind the term “exception” is that when an error occurs and an exception object is created, the normal execution flow of the program is interrupted and execution switches to the exceptional control flow Recall that when the program execution gets into an erroneous state, an exception object is created This object has a type that is related to the type of error The object contains information about the error The default behavior is to print this information and interrupt the execution of the statement that “caused” the error

Introduction to Computing Using Python Exceptional control flow >>> f(2) Start f >>> f(2) Start f >>> f(2) Start f Start g >>> f(2) Start f Start g >>> f(2) Start f Start g Start h >>> f(2) Start f Start g Start h n = 2 f(2) n = 2 f(2) n = 1 g(1) n = 1 g(1) n = 0 h(0) n = 0 h(0) n = 2 print('Start f') f(2) n = 2 print('Start f') f(2) n = 2 print('Start f') g(n-1) f(2) n = 2 print('Start f') g(n-1) f(2) n = 2 print('Start f') g(n-1) print(n) f(2) n = 2 print('Start f') g(n-1) print(n) f(2) n = 1 print('Start g') g(1) n = 1 print('Start g') g(1) n = 1 print('Start g') h(n-1) g(1) n = 1 print('Start g') h(n-1) g(1) n = 1 print('Start g') h(n-1) print(n) g(1) n = 1 print('Start g') h(n-1) print(n) g(1) n = 0 print('Start h') print(1/n) h(0) n = 0 print('Start h') print(1/n) h(0) n = 0 print('Start h') h(0) n = 0 print('Start h') h(0) n = 0 print('Start h') print(1/n) print(n) h(0) n = 0 print('Start h') print(1/n) print(n) h(0) >>> f(2) Start f Start g Start h Traceback (most recent call last): File " ", line 1, in f(2) File "/Users/me/ch7/stack.py", line 13, in f g(n-1) File "/Users/me/ch7/stack.py", line 8, in g h(n-1) File "/Users/me/ch7/stack.py", line 3, in h print(1/n) ZeroDivisionError: division by zero >>> >>> f(2) Start f Start g Start h Traceback (most recent call last): File " ", line 1, in f(2) File "/Users/me/ch7/stack.py", line 13, in f g(n-1) File "/Users/me/ch7/stack.py", line 8, in g h(n-1) File "/Users/me/ch7/stack.py", line 3, in h print(1/n) ZeroDivisionError: division by zero >>> Normal control flowExceptional control flow The default behavior is to interrupt the execution of each “active” statement and print the error information contained in the exception object. 1. def h(n): 2. print('Start h') 3. print(1/n) 4. print(n) def g(n): 7. print('Start g') 8. h(n-1) 9. print(n ) def f(n): 12. print('Start f') 13. g(n-1) 14. print(n) 1. def h(n): 2. print('Start h') 3. print(1/n) 4. print(n) def g(n): 7. print('Start g') 8. h(n-1) 9. print(n ) def f(n): 12. print('Start f') 13. g(n-1) 14. print(n)

Introduction to Computing Using Python Catching and handling exceptions It is possible to override the default behavior (print error information and “crash”) when an exception is raised, using try / except statements strAge = input('Enter your age: ') intAge = int(strAge) print('You are {} years old.'.format(intAge)) strAge = input('Enter your age: ') intAge = int(strAge) print('You are {} years old.'.format(intAge)) try: strAge = input('Enter your age: ') intAge = int(strAge) print('You are {} years old.'.format(intAge)) except: print('Enter your age using digits 0-9!') try: strAge = input('Enter your age: ') intAge = int(strAge) print('You are {} years old.'.format(intAge)) except: print('Enter your age using digits 0-9!') >>> ======================== RESTART ======================== >>> Enter your age: fifteen Traceback (most recent call last): File "/Users/me/age1.py", line 2, in intAge = int(strAge) ValueError: invalid literal for int() with base 10: 'fifteen' >>> >>> ======================== RESTART ======================== >>> Enter your age: fifteen Traceback (most recent call last): File "/Users/me/age1.py", line 2, in intAge = int(strAge) ValueError: invalid literal for int() with base 10: 'fifteen' >>> >>> ========== RESTART ========== >>> Enter your age: fifteen Enter your age using digits 0-9! >>> >>> ========== RESTART ========== >>> Enter your age: fifteen Enter your age using digits 0-9! >>> Default behavior:Custom behavior: If an exception is raised while executing the try block, then the block of the associated except statement is executed The except code block is the exception handler

Introduction to Computing Using Python Format of a try / except statement pair try: except: try: except: The format of a try / except pair of statements is: The exception handler handles any exception raised in the try block The except statement is said to catch the (raised) exception It is possible to restrict the except statement to catch exceptions of a specific type only try: except : try: except :

>>> readAge('age.txt') Value cannot be converted to integer. >>> >>> readAge('age.txt') Value cannot be converted to integer. >>> Introduction to Computing Using Python Format of a try / except statement pair It is possible to restrict the except statement to catch exceptions of a specific type only def readAge(filename): 'converts first line of file filename to an integer and prints it' try: infile = open(filename) strAge = infile.readline() age = int(strAge) print('age is', age) except ValueError: print('Value cannot be converted to integer.') def readAge(filename): 'converts first line of file filename to an integer and prints it' try: infile = open(filename) strAge = infile.readline() age = int(strAge) print('age is', age) except ValueError: print('Value cannot be converted to integer.') 1 fifteen age.txt >>> readAge('age.txt') Value cannot be converted to integer. >>> readAge('age.text') Traceback (most recent call last): File " ", line 1, in readAge('age.text') File "/Users/me/ch7.py", line 12, in readAge infile = open(filename) IOError: [Errno 2] No such file or directory: 'age.text' >>> >>> readAge('age.txt') Value cannot be converted to integer. >>> readAge('age.text') Traceback (most recent call last): File " ", line 1, in readAge('age.text') File "/Users/me/ch7.py", line 12, in readAge infile = open(filename) IOError: [Errno 2] No such file or directory: 'age.text' >>> default exception handler prints this

Introduction to Computing Using Python Multiple exception handlers It is possible to restrict the except statement to catch exceptions of a specific type only def readAge(filename): 'converts first line of file filename to an integer and prints it' try: infile = open(filename) strAge = infile.readline() age = int(strAge) print('age is',age) except IOError: # executed only if an IOError exception is raised print('Input/Output error.') except ValueError: # executed only if a ValueError exception is raised print('Value cannot be converted to integer.') except: # executed if an exception other than IOError or ValueError is raised print('Other error.') def readAge(filename): 'converts first line of file filename to an integer and prints it' try: infile = open(filename) strAge = infile.readline() age = int(strAge) print('age is',age) except IOError: # executed only if an IOError exception is raised print('Input/Output error.') except ValueError: # executed only if a ValueError exception is raised print('Value cannot be converted to integer.') except: # executed if an exception other than IOError or ValueError is raised print('Other error.')

Introduction to Computing Using Python Controlling the exceptional control flow >>> try: f(2) except: print('!!') >>> try: f(2) except: print('!!') >>> try: f(2) except: print('!!') Start f >>> try: f(2) except: print('!!') Start f >>> try: f(2) except: print('!!') Start f Start g >>> try: f(2) except: print('!!') Start f Start g >>> try: f(2) except: print('!!') Start f Start g Start h >>> try: f(2) except: print('!!') Start f Start g Start h n = 2 f(2) n = 2 f(2) n = 1 g(1) n = 1 g(1) n = 0 h(0) n = 0 h(0) n = 2 print('Start f') f(2) n = 2 print('Start f') f(2) n = 2 print('Start f') g(n-1) f(2) n = 2 print('Start f') g(n-1) f(2) n = 2 print('Start f') g(n-1) print(n) f(2) n = 2 print('Start f') g(n-1) print(n) f(2) n = 1 print('Start g') g(1) n = 1 print('Start g') g(1) n = 1 print('Start g') h(n-1) g(1) n = 1 print('Start g') h(n-1) g(1) n = 1 print('Start g') h(n-1) print(n) g(1) n = 1 print('Start g') h(n-1) print(n) g(1) n = 0 print('Start h') print(1/n) h(0) n = 0 print('Start h') print(1/n) h(0) n = 0 print('Start h') h(0) n = 0 print('Start h') h(0) n = 0 print('Start h') print(1/n) print(n) h(0) n = 0 print('Start h') print(1/n) print(n) h(0) 1. def h(n): 2. print('Start h') 3. print(1/n) 4. print(n) def g(n): 7. print('Start g') 8. h(n-1) 9. print(n ) def f(n): 12. print('Start f') 13. g(n-1) 14. print(n) 1. def h(n): 2. print('Start h') 3. print(1/n) 4. print(n) def g(n): 7. print('Start g') 8. h(n-1) 9. print(n ) def f(n): 12. print('Start f') 13. g(n-1) 14. print(n) >>> try: f(2) except: print('!!') Start f Start g Start h !! >>> try: f(2) except: print('!!') Start f Start g Start h !!

>>> import math >>> >>> import math >>> >>> import math >>> dir(math) ['__doc__', '__file__', '__name__', '__package__', 'acos', 'acosh', 'asin', 'asinh', 'atan', 'atan2', 'atanh', 'ceil', 'copysign', 'cos', 'cosh', 'degrees', 'e', 'erf', 'erfc', 'exp', 'expm1', 'fabs', 'factorial', 'floor', 'fmod', 'frexp', 'fsum', 'gamma', 'hypot', 'isfinite', 'isinf', 'isnan', 'ldexp', 'lgamma', 'log', 'log10', 'log1p', 'modf', 'pi', 'pow', 'radians', 'sin', 'sinh', 'sqrt', 'tan', 'tanh', 'trunc’] >>> >>> import math >>> dir(math) ['__doc__', '__file__', '__name__', '__package__', 'acos', 'acosh', 'asin', 'asinh', 'atan', 'atan2', 'atanh', 'ceil', 'copysign', 'cos', 'cosh', 'degrees', 'e', 'erf', 'erfc', 'exp', 'expm1', 'fabs', 'factorial', 'floor', 'fmod', 'frexp', 'fsum', 'gamma', 'hypot', 'isfinite', 'isinf', 'isnan', 'ldexp', 'lgamma', 'log', 'log10', 'log1p', 'modf', 'pi', 'pow', 'radians', 'sin', 'sinh', 'sqrt', 'tan', 'tanh', 'trunc’] >>> >>> import math >>> dir(math) ['__doc__', '__file__', '__name__', '__package__', 'acos', 'acosh', 'asin', 'asinh', 'atan', 'atan2', 'atanh', 'ceil', 'copysign', 'cos', 'cosh', 'degrees', 'e', 'erf', 'erfc', 'exp', 'expm1', 'fabs', 'factorial', 'floor', 'fmod', 'frexp', 'fsum', 'gamma', 'hypot', 'isfinite', 'isinf', 'isnan', 'ldexp', 'lgamma', 'log', 'log10', 'log1p', 'modf', 'pi', 'pow', 'radians', 'sin', 'sinh', 'sqrt', 'tan', 'tanh', 'trunc’] >>> math.sqrt >>> math.pi >>> import math >>> dir(math) ['__doc__', '__file__', '__name__', '__package__', 'acos', 'acosh', 'asin', 'asinh', 'atan', 'atan2', 'atanh', 'ceil', 'copysign', 'cos', 'cosh', 'degrees', 'e', 'erf', 'erfc', 'exp', 'expm1', 'fabs', 'factorial', 'floor', 'fmod', 'frexp', 'fsum', 'gamma', 'hypot', 'isfinite', 'isinf', 'isnan', 'ldexp', 'lgamma', 'log', 'log10', 'log1p', 'modf', 'pi', 'pow', 'radians', 'sin', 'sinh', 'sqrt', 'tan', 'tanh', 'trunc’] >>> math.sqrt >>> math.pi Introduction to Computing Using Python Modules, revisited When the module is executed (imported), then the module is (also) a namespace. A module is a file containing Python code. When the module is executed (imported), then the module is (also) a namespace. This namespace has a name, typically the name of the module. When the module is executed (imported), then the module is (also) a namespace. This namespace has a name, typically the name of the module. In this namespace live the names that are defined in the global scope of the module: the names of functions, values, and classes defined in the module. These names are the module’s attributes. When the module is executed (imported), then the module is (also) a namespace. This namespace has a name, typically the name of the module. In this namespace live the names that are defined in the global scope of the module: the names of functions, values, and classes defined in the module. Built-in function dir() returns the names defined in a namespace To access the imported module’s attributes, the name of the namespace must be specified

Introduction to Computing Using Python Importing a module When the Python interpreter executes an import statement, it: 1.Looks for the file corresponding to the module to be imported. 2.Runs the module’s code to create the objects defined in the module. 3.Creates a namespace where the names of these objects will live. An import statement only lists a name, the name of the module without any directory information or.py suffix. Python uses the Python search path to locate the module. The search path is a list of directories where Python looks for modules. The variable name path defined in the Standard Library module sys refers to this list. When the Python interpreter executes an import statement, it: 1.Looks for the file corresponding to the module to be imported. >>> import sys >>> sys.path ['/Users/me', '/Library/Frameworks/Python.framework/Versions/3.2/lib/python32.zip',... '/Library/Frameworks/Python.framework/Versions/3.2/lib/python3.2/site-packages'] >>> >>> import sys >>> sys.path ['/Users/me', '/Library/Frameworks/Python.framework/Versions/3.2/lib/python32.zip',... '/Library/Frameworks/Python.framework/Versions/3.2/lib/python3.2/site-packages'] >>> current working directoryStandard Library folders

>>> ================ RESTART ================ >>> dir() ['__builtins__', '__doc__', '__name__', '__package__'] >>> import example Traceback (most recent call last): File " ", line 1, in import example ImportError: No module named example >>> >>> ================ RESTART ================ >>> dir() ['__builtins__', '__doc__', '__name__', '__package__'] >>> import example Traceback (most recent call last): File " ", line 1, in import example ImportError: No module named example >>> >>> ================ RESTART ================ >>> dir() ['__builtins__', '__doc__', '__name__', '__package__'] >>> >>> ================ RESTART ================ >>> dir() ['__builtins__', '__doc__', '__name__', '__package__'] >>> Introduction to Computing Using Python The Python search path Suppose we want to import module example stored in folder /Users/me that is not in list sys.path >>> ================ RESTART ================ >>> >>> ================ RESTART ================ >>> >>> ================ RESTART ================ >>> dir() ['__builtins__', '__doc__', '__name__', '__package__'] >>> import example Traceback (most recent call last): File " ", line 1, in import example ImportError: No module named example >>> import sys >>> sys.path.append('/Users/me') >>> import example >>> example.f >>> example.x 0 >>> >>> ================ RESTART ================ >>> dir() ['__builtins__', '__doc__', '__name__', '__package__'] >>> import example Traceback (most recent call last): File " ", line 1, in import example ImportError: No module named example >>> import sys >>> sys.path.append('/Users/me') >>> import example >>> example.f >>> example.x 0 >>> >>> ================ RESTART ================ >>> dir() ['__builtins__', '__doc__', '__name__', '__package__'] >>> import example Traceback (most recent call last): File " ", line 1, in import example ImportError: No module named example >>> import sys >>> sys.path.append('/Users/me') >>> import example >>> example.f >>> example.x 0 >>> dir() ['__builtins__', '__doc__', '__name__', '__package__', 'example', 'sys’] >>> ================ RESTART ================ >>> dir() ['__builtins__', '__doc__', '__name__', '__package__'] >>> import example Traceback (most recent call last): File " ", line 1, in import example ImportError: No module named example >>> import sys >>> sys.path.append('/Users/me') >>> import example >>> example.f >>> example.x 0 >>> dir() ['__builtins__', '__doc__', '__name__', '__package__', 'example', 'sys’] By just adding folder /Users/me to the search path, module example can be imported When called without an argument, function dir() returns the names in the top-level module the shell, in this case. 'an example module' def f(): 'function f' print('Executing f()') def g(): 'function g' print('Executing g()') x = 0 # global var 'an example module' def f(): 'function f' print('Executing f()') def g(): 'function g' print('Executing g()') x = 0 # global var names in the shell namespace; note that example is not in no folder in the Python search path contains module example

Introduction to Computing Using Python Top-level module A computer application is a program typically split across multiple modules. One of the modules is special: It contains the “main program”. This module is referred to as the top-level module. The remaining modules are “library” modules that are imported by other modules and that contain functions and classes used by it >>> === RESTART === >>> My name is __main__ >>> >>> === RESTART === >>> My name is __main__ >>> print('My name is {}'.format(__name__)) name.txt A module is a top-level module if: it is run from the shell it is run at the command line > python name.py My name is __main__ > python name.py My name is __main__ A computer application is a program typically split across multiple modules. One of the modules is special: It contains the “main program”. This module is referred to as the top-level module. The remaining modules are “library” modules that are imported by other modules and that contain functions and classes used by it When a module is imported, Python creates a few “bookkeeping” variables in the module namespace, including variable __name__ : set to '__main__', if the module is being run as a top-level module A module is a top-level module if: it is run from the shell it is run at the command line A module is a top-level module if: it is run from the shell it is run at the command line

Introduction to Computing Using Python Top-level module >>> import name My name is name >>> import name My name is name print('My name is {}'.format(__name__)) name.txt A computer application is a program typically split across multiple modules. One of the modules is special: It contains the “main program”. This module is referred to as the top-level module. The remaining modules are “library” modules that are imported by the top-level module and that contain functions and classes used by it When a module is imported, Python creates a few “bookkeeping” variables in the module namespace, including variable __name__ : set to '__main__', if the module is being run as a top-level module set to the module’s name, if the file is being imported by another module import name import.txt >>> === RESTART === >>> My name is name >>> === RESTART === >>> My name is name

Introduction to Computing Using Python Three ways to import module attributes 'an example module' def f(): 'function f' print('Executing f()') def g(): 'function g' print('Executing g()') x = 0 # global var 'an example module' def f(): 'function f' print('Executing f()') def g(): 'function g' print('Executing g()') x = 0 # global var example.txt namespace __main__ >>> f module example example f() 0 0 gx g() 1. Import the (name of the) module >>> import example >>> >>> import example >>> >>> import example >>> example.x 0 >>> example.f >>> example.f() Executing f() >>> >>> import example >>> example.x 0 >>> example.f >>> example.f() Executing f() >>>

Introduction to Computing Using Python Three ways to import module attributes 'an example module' def f(): 'function f' print('Executing f()') def g(): 'function g' print('Executing g()') x = 0 # global var 'an example module' def f(): 'function f' print('Executing f()') def g(): 'function g' print('Executing g()') x = 0 # global var example.txt namespace __main__ >>> f module example f f() 0 0 gx g() 2. Import specific module attributes >>> from example import f >>> >>> from example import f >>> >>> from example import f >>> f() Executing f() >>> x Traceback (most recent call last): File " ", line 1, in x NameError: name 'x' is not defined >>> >>> from example import f >>> f() Executing f() >>> x Traceback (most recent call last): File " ", line 1, in x NameError: name 'x' is not defined >>>

Introduction to Computing Using Python Three ways to import module attributes 'an example module' def f(): 'function f' print('Executing f()') def g(): 'function g' print('Executing g()') x = 0 # global var 'an example module' def f(): 'function f' print('Executing f()') def g(): 'function g' print('Executing g()') x = 0 # global var example.txt namespace __main__ >>> 3. Import all module attributes >>> from example import * >>> >>> from example import * >>> >>> from example import * >>> f() Executing f() >>> g() Executing g() >>> x 0 >>> >>> from example import * >>> f() Executing f() >>> g() Executing g() >>> x 0 >>> f module example f f() 0 0 gx g() g x

Introduction to Computing Using Python A class is a namespace >>> list.pop >>> list.sort >>> >>> list.pop >>> list.sort >>> __add__ namespace list __add__() sort() count x count() pop pop() A class is really a namespace The name of this namespace is the name of the class The names defined in this namespace are the class attributes (e.g., class methods)... A class is really a namespace The name of this namespace is the name of the class The names defined in this namespace are the class attributes (e.g., class methods) The class attributes can be accessed using the standard namespace notation Function dir() can be used to list the class attributes >>> list.pop >>> list.sort >>> dir(list) ['__add__', '__class__',... 'index', 'insert', 'pop', 'remove', 'reverse', 'sort'] >>> list.pop >>> list.sort >>> dir(list) ['__add__', '__class__',... 'index', 'insert', 'pop', 'remove', 'reverse', 'sort']

A class method is really a function defined in the class namespace; when Python executes it first translates it to and actually executes this last statement Introduction to Computing Using Python Class methods >>> lst = [9, 1, 8, 2, 7, 3] >>> lst [9, 1, 8, 2, 7, 3] >>> lst.sort() >>> lst [1, 2, 3, 7, 8, 9] >>> >>> lst = [9, 1, 8, 2, 7, 3] >>> lst [9, 1, 8, 2, 7, 3] >>> lst.sort() >>> lst [1, 2, 3, 7, 8, 9] >>> __add__ namespace list __add__() sort() count x count() pop pop()... >>> lst = [9, 1, 8, 2, 7, 3] >>> lst [9, 1, 8, 2, 7, 3] >>> lst.sort() >>> lst [1, 2, 3, 7, 8, 9] >>> lst = [9, 1, 8, 2, 7, 3] >>> lst [9, 1, 8, 2, 7, 3] >>> list.sort(lst) >>> lst [1, 2, 3, 7, 8, 9] >>> >>> lst = [9, 1, 8, 2, 7, 3] >>> lst [9, 1, 8, 2, 7, 3] >>> lst.sort() >>> lst [1, 2, 3, 7, 8, 9] >>> lst = [9, 1, 8, 2, 7, 3] >>> lst [9, 1, 8, 2, 7, 3] >>> list.sort(lst) >>> lst [1, 2, 3, 7, 8, 9] >>> >>> lst = [9, 1, 8, 2, 7, 3] >>> lst [9, 1, 8, 2, 7, 3] >>> lst.sort() >>> lst [1, 2, 3, 7, 8, 9] >>> lst = [9, 1, 8, 2, 7, 3] >>> lst [9, 1, 8, 2, 7, 3] >>> list.sort(lst) >>> lst [1, 2, 3, 7, 8, 9] >>> lst.append(6) >>> lst [1, 2, 3, 7, 8, 9, 6] >>> >>> lst = [9, 1, 8, 2, 7, 3] >>> lst [9, 1, 8, 2, 7, 3] >>> lst.sort() >>> lst [1, 2, 3, 7, 8, 9] >>> lst = [9, 1, 8, 2, 7, 3] >>> lst [9, 1, 8, 2, 7, 3] >>> list.sort(lst) >>> lst [1, 2, 3, 7, 8, 9] >>> lst.append(6) >>> lst [1, 2, 3, 7, 8, 9, 6] >>> >>> lst = [9, 1, 8, 2, 7, 3] >>> lst [9, 1, 8, 2, 7, 3] >>> lst.sort() >>> lst [1, 2, 3, 7, 8, 9] >>> lst = [9, 1, 8, 2, 7, 3] >>> lst [9, 1, 8, 2, 7, 3] >>> list.sort(lst) >>> lst [1, 2, 3, 7, 8, 9] >>> lst.append(6) >>> lst [1, 2, 3, 7, 8, 9, 6] >>> list.append(lst, 5) >>> lst [1, 2, 3, 7, 8, 9, 6, 5] >>> lst = [9, 1, 8, 2, 7, 3] >>> lst [9, 1, 8, 2, 7, 3] >>> lst.sort() >>> lst [1, 2, 3, 7, 8, 9] >>> lst = [9, 1, 8, 2, 7, 3] >>> lst [9, 1, 8, 2, 7, 3] >>> list.sort(lst) >>> lst [1, 2, 3, 7, 8, 9] >>> lst.append(6) >>> lst [1, 2, 3, 7, 8, 9, 6] >>> list.append(lst, 5) >>> lst [1, 2, 3, 7, 8, 9, 6, 5] lst.sort() list.sort(lst) lst.append(6) instance.method(arg1, arg2, …) list.append(lst, 6) class.method(instance, arg1, arg2, …) The function has an extra argument, which is the object invoking the method

Rewrite the below Python statement so that instead of making the usual method invocations you use the notation Introduction to Computing Using Python Exercise >>> s = 'hello' >>> s = 'ACM' >>> s.lower() 'acm' >>> s.find('C') 1 >>> s.replace('AC', 'IB') 'IBM' >>> s = 'hello' >>> s = 'ACM' >>> s.lower() 'acm' >>> s.find('C') 1 >>> s.replace('AC', 'IB') 'IBM' lst.sort() list.sort(lst) lst.append(6) instance.method(arg1, arg2, …) list.append(lst, 6) class.method(instance, arg1, arg2, …) >>> s = 'ACM' >>> str.lower(s) 'acm' >>> str.find(s, 'C') 1 >>> str.replace(s, 'AC', 'IB') 'IBM' >>> >>> s = 'ACM' >>> str.lower(s) 'acm' >>> str.find(s, 'C') 1 >>> str.replace(s, 'AC', 'IB') 'IBM' >>>