Python For Advance 3 Days With Python
Python For Advance 3 Days With Python
Tony Snake was received Bachelor of Computer Science from the American
University, and Bachelor of Business Administration from the American
University, USA.
He is becoming Ph. Candidate of Department of Data Informatics, (National)
Korea Maritime and Ocean University, Busan 49112, Republic of Korea
(South Korea).
His research interests are social network analysis, big data, AI and robotics.
He received Best Paper Award the 15th International Conference on
Multimedia Information Technology and Applications (MITA 2019)
Table of Contents
Contents
About the Authors 1
He received Best Paper Award the 15th International Conference on Multimedia Information
Technology and Applications (MITA 2019) 1
Table of Contents 2
Chapter 1: Python File Handling 2
1. Opening and Closing Files in Python 2
What is File Handling in Python? 2
Why file handling? 2
Opening and Closing a File in Python 2
open() Function: 2
Example of Opening and Closing Files in Python 3
close() function: 3
Example with with statement: 3
Explanation 4
2. How to Read a File in Python 5
Introduction 5
File Access Modes 5
Reading a File in Python 6
Opening a File in Python 6
Usage: 6
Closing a File in Python 7
Usage: 7
1. The read() method in python: 7
2. The readline() method in python: 8
Usage: 8
3. The readlines() method in python: 9
Usage: 9
Reading a Binary File in Python 9
Example: 9
Reading Using the with open() Method in Python 10
Syntax: 10
3. How to Write a File in Python 10
Writing to a text file in Python 10
1. The write() function in Python: 10
Output 11
2. The writelines() function in Python: 12
Writing to a binary file in Python 12
Example 12
4. How to Delete File in Python? 12
Introduction 12
How to Delete Files Using Python? 13
1. Using the os module in python 13
2. Using the shutil module in python: 13
3. Using the Pathlib Module in Python 14
5. With Statement in Python 15
What is “with” Statement in Python 15
Wait… What is a buffer? 15
Try…Finally in Python 15
With in Python 15
But how does “with” know to close a file? 16
Using “with” in user-defined objects 16
Wait… Why would I use “with” in my custom function? 16
Output: 17
Contextlib Library 18
Output: 18
Chapter 2: Python Exception Handling 18
1. Exception Handling in Python 18
Introduction 18
What is Exception Handling in Python? 19
Common Exceptions in Python 20
Catching Specific Exceptions in Python 21
Raising Custom Exceptions 22
Syntax to Raise an Exception 22
try except and ELSE! 23
Syntax With Else Clause 23
Try Clause with Finally 24
Why Use Finally or Else in try..except? 26
What Happens if Errors are Raised in Except or Finally Block? 26
2. Assert Keyword in Python 27
Introduction to Assert in Python 27
What is Assert? 28
Why Use Assert in Python? 28
Python Assert Statement 28
What Does the Assertion Error do to our Code? 30
Where is Assertion used in Python? 31
1. Checking parameter types / classes / values 31
2. Checking “can’t happen” situations 31
3. Documentation of Understanding 32
Another Example of Assertion in Python 32
Chapter 3: Python Object & Class 33
1. Python Object Oriented Programming 33
Object Oriented Programming 33
Class 34
Object 34
Example 1: Creating Class and Object in Python 35
Methods 36
Example 2 : Creating Methods in Python 36
Inheritance 37
Example 3: Use of Inheritance in Python 37
Encapsulation 38
Example 4: Data Encapsulation in Python 38
Polymorphism 39
Example 5: Using Polymorphism in Python 40
2. Python Objects and Classes 41
Python Objects and Classes 41
Defining a Class in Python 41
Creating an Object in Python 43
Constructors in Python 44
Deleting Attributes and Objects 46
3. Python Inheritance 47
Inheritance in Python 47
Python Inheritance Syntax 47
Example of Inheritance in Python 48
Method Overriding in Python 50
4. Python Multiple Inheritance 51
Python Multiple Inheritance 51
Example 51
Python Multilevel Inheritance 52
Method Resolution Order in Python 52
5. Python Operator Overloading 54
Python Operator Overloading 54
Python Special Functions 55
Overloading the + Operator 57
Overloading Comparison Operators 60
Chapter 4: Python Advance Constructs 62
1. Python Modules 62
What are Modules in Python? 62
Formal Definition of Python Module 62
Features of Python Module 62
Types of Python Modules 63
InBuilt Module: 63
How to call a Built-In module?** 63
Variables in Module 64
The Import Statement 65
What is an Import Statement? 65
How to Use Import Modules 65
Using from <module_name> import statement 66
Python Module Search Path 67
Naming a Module 67
Re-Naming a Module 68
Reloading a Module 68
The dir() built-in Function 68
2. Packages in Python and Import Statement 69
Introduction 69
Importing Module From a Package in Python 69
Syntax: 70
Installing a Python Package Globally 70
Import Statement in Python 71
Output: 72
Example: 72
Output: 72
Importing Attributes Using the from Import Statement in Python 72
Syntax: 72
Output: 72
Output: 73
Importing Modules Using the From Import * Statement in Python 73
Syntax: 73
Example: 73
Output: 73
Importing Modules in Python Using the Import as Statement 73
Syntax: 73
Example: 73
Output: 74
Importing Class/Functions from Module in Python 74
Output: 74
Import User-defined Module in Python 74
Output: 74
Importing from Another Directory in Python 75
Output: 76
Importing Class from Another File in Python 76
Output: 77
3. Python Collection Module 77
Introduction to Modules 77
Collection Module 78
namedtuple() 78
OrderedDict() 80
defaultdict() 82
Counter() 84
deque() 85
Chainmap 87
4. Regular Expression in Python 89
RegEx Module 90
Python RegEx Expressions Functions 90
Meta Characters 90
Special Sequences in Python RegEx 91
Examples for each sequence are given below 91
Sets 92
findall(pattern, string) 93
search(pattern, string) 93
split(pattern, string) 94
sub(pattern, repl, string) 94
Match Object 94
5. Python Datetime 96
Introduction to Python Datetime 96
How to Use Date and Datetime Class? 97
Classes of DateTime Python Module 98
Date 98
Time 99
Points to Remember About Time Class 99
Datetime 100
Timedelta 101
Tzinfo 102
Naive & Aware Datetime Objects 102
Timezone 103
Basics of pytz Library 103
Date Class 104
.date() 104
.today() 105
.min 105
.max 105
.day 106
.month 106
.year 106
strftime() 107
Current Date 108
.today() 108
.now() 108
Use of datetime.strptime() 109
How to Get Current timeStamp? 109
How to Know the Day of the Given Date? 110
Generate a List of Dates from a Given Date 111
Algorithm 111
Applications Of Python DateTime Module 112
Chapter 5: Python Advanced Topics 113
1. Python Iterators 113
Iterators in Python 113
Iterating Through an Iterator 113
Working of for loop for Iterators 115
Building Custom Iterators 116
Python Infinite Iterators 117
2. Python Generators 119
Generators in Python 119
Create Generators in Python 120
Differences between Generator function and Normal function 120
Python Generators with a Loop 123
Python Generator Expression 124
Use of Python Generators 126
1. Easy to Implement 126
2. Memory Efficient 127
3. Represent Infinite Stream 127
4. Pipelining Generators 127
3. Python Closures 128
Nonlocal variable in a nested function 128
Defining a Closure Function 129
When do we have closures? 131
When to use closures? 131
3. Python Decorators 133
Decorators in Python 133
Prerequisites for learning decorators 133
Getting back to Decorators 136
Decorating Functions with Parameters 137
Chaining Decorators in Python 139
4. Python @property decorator 141
Class Without Getters and Setters 142
Using Getters and Setters 143
The property Class 146
The @property Decorator 149
5. Python Regular Expressions 151
Python RegEx 151
re.findall() 152
Example 1: re.findall() 152
re.split() 152
Example 2: re.split() 152
re.sub() 153
Example 3: re.sub() 153
re.subn() 154
Example 4: re.subn() 155
re.search() 155
Example 5: re.search() 155
Match object 156
match.group() 156
Example 6: Match object 156
match.start(), match.end() and match.span() 157
match.re and match.string 158
Using r prefix before RegEx 158
Example 7: Raw string using r prefix 158
Chapter 6: Error Handling 159
1. Python Error and In-built Exception in Python 159
Python: Syntax Error 159
Python: What is an Exception? 160
Decoding the Exception Message in Python 161
2. Python Exception Handling 161
Handling Exceptions using try and except 162
The try block 163
The except block 163
Code Execution continues after except block 163
Catching Multiple Exceptions in Python 164
Multiple except blocks 165
Handling Multiple Exceptions with on except block 165
Generic except block to Handle unknown Exceptions 166
3. Exeption Handling: Finally 167
finally block with/without Exception Handling 167
Exception in except block 168
4. Python Exception Handling: raise Keyword 169
raise Without Specifying Exception Class 171
raise With an Argument 172
Chapter 7: Multithreading 172
1. Introduction to Multithreading In Python 172
Threads 172
Types Of Thread 173
What is Multithreading? 173
Time for an Example 173
Multithreading in Python 174
2. Threading Module In Python 175
threading Module Functions 175
threading.active_count() Function 175
threading.current_thread() 176
threading.get_ident() 178
threading.enumerate() 178
threading.main_thread() 178
threading.settrace(fun) 179
threading.setprofile(fun) 179
threading.stack_size([size]) 179
threading.TIMEOUT_MAX 180
threading Module Objects 180
3. Thread class and its Object - Python Multithreading 180
How Thread works? 181
Functions and Constructor in the Thread class 181
Thread class Constructor 182
start() method 182
run() method 182
join([timeout]) method 183
getName() method 183
setName(name) method 183
isAlive() method 183
isDaemon() method 183
setDaemon(daemonic) method 183
4. Thread Synchronization using Event Object 184
Python Multithreading: Event Object 184
Initialize Event object 184
isSet() method 185
set() method 185
clear() method 185
wait([timeout]) method 185
Time for an Example 186
5. Timer Object - Python Multithreading 186
Syntax for creating Timer object 187
Methods of Timer class 187
start() method 187
cancel() method 187
Time for an Example 187
6. Condition Object - Thread Synchronization in Python 188
Condition object: wait(), notify() and notifyAll() 189
Condition class methods 189
acquire(*args) method 189
release() method 189
wait([timeout]) method 189
notify() method 190
notifyAll() method 190
Time for an Example! 190
7. Barrier Object - Python Multithreading 190
Functions provided by Barrier class 191
wait(timeout=None) method 191
reset() method 192
abort() method 192
parties 192
n_waiting 192
broken 193
Time for an Example! 193
Chapter 8: Python Logging 193
1. Python Logging Basic Configurations 193
Let's take an Example 194
Points to remember: 194
Store Logs in File 194
Set Format of Logs 195
Add process ID to logs with loglevel and message 195
Add Timestamp to logs with log message 195
Use the datefmt attribute 196
2. Python - Print Logs in a File 197
Python Logging - Store Logs in a File 197
3. Python Logging Variable Data 199
Example 199
There is Another way 200
4. Python Logging Classes and Functions 200
1. Logger class 201
2. LogRecord 201
3. Handler 201
4. Formatter 201
Several Logger objects 201
Chapter 9: Python With MySQL 203
1. MySQL with Python 203
Python MySQL - Prerequisites 203
What is MySQL? 204
MySQL Connector 204
Test the MySQL Connector 205
Creating the Connection 205
2. Python MySQL - Create Database 206
Python MySQL - CREATE DATABASE 206
Python MySQL - Create Database Example 206
Python MySQL - List all Database 207
3. Python MySQL - Create and List Table 209
Python MySQL - Create Table 209
SQL Query to Create Table 210
List existing Tables in a Database 211
Python MySQL - Table with Primary Key 212
What is Primary Key? 212
Add Primary Key during Table creation 213
Python MySQL - Describe the Table 214
Add Primary Key to Existing Table 215
4. Python MySQL - Insert data in Table 216
Python MySQL - INSERT Data 216
Inserting Single Row in MySQL Table 217
Inserting Multiple Rows in MySQL Table 218
5. Python MySQL - Select data from Table 220
Python MySQL - SELECT Data 220
Retrieve All records from MySQL Table 220
Retrieve data from specific Column(s) of a Table 222
Selecting Multiple columns from a Table 223
To fetch the first record - fetchone() 225
6. Python MySQL - Update Table data 226
Python MySQL UPDATE: Syntax 226
Python MySQL UPDATE Table Data: Example 226
7. Python MySQL - Delete Table Data 229
Python MySQL DELETE: Syntax 229
Python MySQL DELETE Table Data: Example 229
8. Python MySQL - Drop Table 232
Python MySQL DROP TABLE: Example 232
Python MySQL - Drop Table if it exists 233
9. Python MySQL - WHERE Clause 235
Python MySQL WHERE Clause 235
Using WHERE Clause 235
10. Python MySQL - Orderby Clause 237
Python MySQL ORDER BY Example 237
Python MySQL ORDER BY DESC 239
11. Python MySQL - Limit Clause 241
Python MySQL LIMIT: Example 242
Using OFFSET Keyword with LIMIT clause 243
12. Python MySQL - Table Joins 244
Python MySQL - Joining two tables 245
Python MySQL - Left Join 247
Python MySQL - Right Join 249
Conclusion 251
Python for Advance: 3 Days with
Python
Chapter 1: Python File Handling
Sr.
Modes Description
No
Opens a file in read only mode. The pointer of the file is at the beginning of the file. This is also the defa
1. r
mode.
2. rb Same as r mode except this opens the file in binary mode.
3. r+ Opens the file for both reading and writing. The pointer is at the beginning of the file.
4. rb+ Same as r+ mode except this opens the file in binary mode.
5. w Opens the file for writing. Overwrites the existing file and in file is not present then creates a new one
6. wb Same as w mode except this opens the file in binary format.
7. w+ Opens the file for both reading and writing, Rest is the same as w mode.
8. wb+ Same as w+ except this opens the file in binary format.
Opens the file for appending. If the file is present then the pointer is at the end of the file else it creates a
9. a
file for writing.
10. ab Same as a mode except this opens the file in binary format.
Opens the file for appending and reading. The file pointer is at the end of the file if the file exists else cre
11. a+
a new file for reading and writing.
Same as a+ mode except this opens the file in binary format.
12. ab+
# When the file is not in the same folder where the python script is present.In this case whole path of
file should be written.
file = open('D:/data/test.txt',mode='r')
It is general practice to close an opened file as closed file reduces the risk of
being unwarrantedly updated or read. We can close files in python using
close function. Let's discuss about it.
close() function:
This function don't take any argument and you can directly call close function
using file object. It can be call multiple times but if any operation is
performed on closed file, "ValueError" exception is raised.
Syntax:
file.**close()**
PS: You can use 'with' statement with open also as it provides
better exception handling and simplifies it with by providing some cleanup
tasks. Also it will automatically close the file and you don't have to do it
manually.
Example with with statement:
with open("test.txt", mode='r') as f:
# perform file operations
The method shown in the above section is not entirely safe. If some exception
occurs while opening the file then the code will exit without closing the file.
A more safer way is to use try-finally block while opening files.
try:
file = open('test.txt',mode='r')
# Perform file handling operations
finally:
file.close()
Now this guarantees that the file will close even if an exception occurs while
opening the file. So, you can also use the ‘with’ statement method instead of
this. Any of the two methods is good.
Now we will see some examples on how to open and close files in python in
various modes. Below is an example of a few important modes for rest you
can try yourself.
The file that I will be using contains the following content.
Now we will perform some operations on the file and print the content of the
file after each operation to get a better understanding how this works. The
example is given below. Both the file and the python script should be in the
same folder.
# Opening file in read mode and printing the contents of the file.
with open("test.txt", mode='r') as f:
data = f.readlines() #This reads all the lines from the file in a list.
print(data) #This will print content of file that is Hello World!
Explanation
In above example firstly test.txt file was opened in read(r) mode to read its
content and that file data will be printed and after that file was opened
with write(w) mode, so it will overwrite all the content of that file and new
data will be written to that file. After that file was opened in append(a) mode,
so new data will be appended to existing data of file, will not be overwritten.
I have covered just three most important modes of file handling. You can try
some other modes.on your own.
@contextmanager
def myContextManager():
print("Entering...")
yield "Hello There"
print("Exiting...")
Introduction
Writing code in order to solve or automate some huge problems is a pretty
powerful thing to do, isn’t it? But as Peter Parker says, ‘with great power
comes great responsibility’. This in fact holds true when writing your code as
well! While our code is being executed, there might be some events that
disrupt the normal flow of your code. These events are errors, and once these
occur python interpreter is in a situation that it cannot deal with and hence
raises an exception.
In python, an exception is a class that represents error. If these exceptions are
not handled, our application or programs go into a crash state. As a
developer, we definitely have the power to write code and solve problems but
it becomes our responsibility to handle these exceptions that might occur and
disrupt the code flow.
Let’s see how these exceptions can be handled in this article but first of all,
let’s take a look at an example that will introduce you to an exception:
a = 10
b=0
c = b/a
print(c)
Output:
Traceback (most recent call last):
File "main.py", line 3, in <module>
c = a/b
ZeroDivisionError: division by zero
Above is an example of what we call an unhandled exception. On line 3, the
code went into a halt state as it's an unhandled exception. To avoid such halt
states, let’s take a look at how to handle exceptions/errors.
What is Exception Handling in Python?
Exceptions can be unexpected or in some cases, a developer might expect
some disruption in the code flow due to an exception that might come up in a
specific scenario. Either way, it needs to be handled.
Python just like any other programming language provides us with a
provision to handle exceptions. And that is by try & except block. Try block
allows us to write the code that is prone to exceptions or errors. If any
exception occurs inside the try block, the except block is triggered, in which
you can handle the exception as now the code will not go into a halt state but
the control will flow into the except block where it can be manually handled.
Any critical code block can be written inside a try clause. Once the expected
or unexpected exception is raised inside this try, it can be handled inside
the except clause (This ‘except’ clause is triggered when an exception is
raised inside ‘try’), as this except block detects any exception raised in the try
block. By default, except detects all types of exceptions, as all the built-in
exceptions in python are inherited from common class Exception.
The basic syntax is as follows:
try:
# Some Code
except:
# Executed if we get an error in the try block
# Handle exception here
Try and except go hand in hand i.e. the syntax is to write and use both of
them. Writing just try or except will give an error.
Let’s consider an example where we’re dividing two numbers. Python
interpreter will raise an exception when we try to divide a number with 0.
When it does, we can take custom action on it in the except clause.
def divideNos(a, b):
return a/b
try:
divideNos(10, 0)
except:
print('some exception occured')
Output:
some exception occured
But how do we know what exactly error was read by the python interpreter?
Well, when a python interpreter raises an exception, it is in the form of an
object that holds information about the exception type and message. Also,
every exception type in python inherits from the base class Exception.
def divideNos(a, b):
return a/b
try:
divideNos(10, 0)
# Any exception raised by the python interpreter, is inherited by the base class ‘Exception’, hence
any exception raised in the try block will be detected and collected further in the except block for
handling.
except Exception as e:
print(e) # as e is an object of type Exception, Printing here to see what message it holds.
print(e.__class__)
division by zero
<class 'ZeroDivisionError'>
In the above code, we used the Exception class with the except statement. It
is used by using the as keyword. This object will contain the cause of the
exception and we’re printing it in order to look what the reason is inside this
Exception object.
For example:
def isStringEmpty(a):
if(type(a)!=str):
raise TypeError('a has to be string')
if(not a):
raise ValueError('a cannot be null')
a.strip()
if(a == ''):
return False
return True
try:
a = 123
print('isStringEmpty:', isStringEmpty(a))
except ValueError as e:
print('ValueError raised:', e)
except TypeError as e:
print('TypeError raised:', e)
Output:
TypeError raised: a has to be string
In the above code, the variable ‘a’ can hold whatever value that is
assigned to it. Here we assign it a number and we’re passing to a
custom method isStringEmpty that checks if a string is an empty string.
But we orchestrated it to throw a TypeError, by assigning ‘a’ variable a
number.
In the method, we’re checking if the variable is a string or not and if it
holds a value or not. In this case, it is supposed to be a string, but
assigned it as a number as we’re raising an exception by using
itemsInCart = ['mobile', 'earphones', 'charger'] # assume user has these items in their cart
Checkout(itemsInCart) # checking out items the user has in their cart.
Consider above as a pseudo-code to a program where it checks out
items added by the user in their cart.
The Checkout method always creates an order id before booking all the
items in the cart.
Assuming itemsInCart contains all the items the user has added to a
cart.
In the checkout experience of an e-commerce website, an orderId is
created irrespective of whether a booking of all items has been a
success, failed, or partially failed.
Hence in the above pseudo-code, we’re creating
an orderId using GetOrderIdForBooking() and assume it returns a
random string for each order. BookItems() books all the items for us in
the itemsInCart array.
If some error occurs while booking items, we have to log the reason for
the exception LogErrorOccuredWhileBooking() does that job for us
and if it's successful we have to log that the booking was successful.
Now whatever the status of the order is, either way we have to email
the status of the order to the user and that’s what
the EmailOrderStatus() does for us.
The above example perfectly fits for an application of try, except or else and
finally block as each block is playing a role that might’ve been difficult to
achieve if we didn’t use these blocks.
# do something
However, if say, a program involves calculations with radius and diameter of
a circle and the radius of the circle isn’t half of the diameter at any point in
the program, then this comes under the “this should not happen” case, as it is
a fact that cannot be changed or violated.
For example,
# some complex operations involving the radius of a circle
Class
A class is a blueprint for the object.
We can think of class as a sketch of a parrot with labels. It contains all the
details about the name, colors, size etc. Based on these descriptions, we can
study about the parrot. Here, a parrot is an object.
The example for class of parrot can be :
class Parrot:
pass
Here, we use the class keyword to define an empty class Parrot. From class,
we construct instances. An instance is a specific object created from a
particular class.
Object
An object (instance) is an instantiation of a class. When class is defined, only
the description for the object is defined. Therefore, no memory or storage is
allocated.
The example for object of parrot class can be:
obj = Parrot()
# class attribute
species = "bird"
# instance attribute
def __init__(self, name, age):
self.name = name
self.age = age
Output
Blu is a bird
Woo is also a bird
Blu is 10 years old
Woo is 15 years old
In the above program, we created a class with the name Parrot. Then, we
define attributes. The attributes are a characteristic of an object.
These attributes are defined inside the __init__ method of the class. It is the
initializer method that is first run as soon as the object is created.
Then, we create instances of the Parrot class. Here, blu and woo are
references (value) to our new objects.
We can access the class attribute using __class__.species. Class attributes are
the same for all instances of a class. Similarly, we access the instance
attributes using blu.name and blu.age. However, instance attributes are
different for every instance of a class.
Methods
Methods are functions defined inside the body of a class. They are used to
define the behaviors of an object.
# instance attributes
def __init__(self, name, age):
self.name = name
self.age = age
# instance method
def sing(self, song):
return "{} sings {}".format(self.name, song)
def dance(self):
return "{} is now dancing".format(self.name)
Output
Blu sings 'Happy'
Blu is now dancing
In the above program, we define two methods i.e sing() and dance(). These
are called instance methods because they are called on an instance object
i.e blu.
Inheritance
Inheritance is a way of creating a new class for using details of an existing
class without modifying it. The newly formed class is a derived class (or
child class). Similarly, the existing class is a base class (or parent class).
def __init__(self):
print("Bird is ready")
def whoisThis(self):
print("Bird")
def swim(self):
print("Swim faster")
# child class
class Penguin(Bird):
def __init__(self):
# call super() function
super().__init__()
print("Penguin is ready")
def whoisThis(self):
print("Penguin")
def run(self):
print("Run faster")
peggy = Penguin()
peggy.whoisThis()
peggy.swim()
peggy.run()
Output
Bird is ready
Penguin is ready
Penguin
Swim faster
Run faster
In the above program, we created two classes i.e. Bird (parent class)
and Penguin (child class). The child class inherits the functions of parent
class. We can see this from the swim() method.
Again, the child class modified the behavior of the parent class. We can see
this from the whoisThis() method. Furthermore, we extend the functions of
the parent class, by creating a new run() method.
Additionally, we use the super() function inside the __init__() method. This
allows us to run the __init__() method of the parent class inside the child
class.
Encapsulation
Using OOP in Python, we can restrict access to methods and variables. This
prevents data from direct modification which is called encapsulation. In
Python, we denote private attributes using underscore as the prefix i.e
single _ or double __.
Example 4: Data Encapsulation in Python
class Computer:
def __init__(self):
self.__maxprice = 900
def sell(self):
print("Selling Price: {}".format(self.__maxprice))
c = Computer()
c.sell()
Output
Selling Price: 900
Selling Price: 900
Selling Price: 1000
c.__maxprice = 1000
Here, we have tried to modify the value of __maxprice outside of the class.
However, since __maxprice is a private variable, this modification is not seen
on the output.
As shown, to change the value, we have to use a setter function
i.e setMaxPrice() which takes price as a parameter.
Polymorphism
Polymorphism is an ability (in OOP) to use a common interface for multiple
forms (data types).
Suppose, we need to color a shape, there are multiple shape options
(rectangle, square, circle). However we could use the same method to color
any shape. This concept is called Polymorphism.
def fly(self):
print("Parrot can fly")
def swim(self):
print("Parrot can't swim")
class Penguin:
def fly(self):
print("Penguin can't fly")
def swim(self):
print("Penguin can swim")
# common interface
def flying_test(bird):
bird.fly()
#instantiate objects
blu = Parrot()
peggy = Penguin()
Output
Parrot can fly
Penguin can't fly
In the above program, we defined two classes Parrot and Penguin. Each of
them have a common fly() method. However, their functions are different.
To use polymorphism, we created a common interface
i.e flying_test() function that takes any object and calls the
object's fly() method. Thus, when we passed the blu and peggy objects in
the flying_test() function, it ran effectively.
class MyNewClass:
'''This is a docstring. I have created a new class'''
pass
A class creates a new local namespace where all its attributes are defined.
Attributes may be data or functions.
There are also special attributes in it that begins with double underscores __.
For example, __doc__ gives us the docstring of that class.
As soon as we define a class, a new class object is created with the same
name. This class object allows us to access the different attributes as well as
to instantiate new objects of that class.
class Person:
"This is a person class"
age = 10
def greet(self):
print('Hello')
# Output: 10
print(Person.age)
Output
10
<function Person.greet at 0x7fc78c6e8160>
This is a person class
This will create a new object instance named harry. We can access the
attributes of objects using the object name prefix.
Attributes may be data or method. Methods of an object are corresponding
functions of that class.
This means to say, since Person.greet is a function object (attribute of
class), Person.greet will be a method object.
class Person:
"This is a person class"
age = 10
def greet(self):
print('Hello')
Output
<function Person.greet at 0x7fd288e4e160>
<bound method Person.greet of <__main__.Person object at 0x7fd288e9fa30>>
Hello
You may have noticed the self parameter in function definition inside the
class but we called the method simply as harry.greet() without
any arguments. It still worked.
This is because, whenever an object calls its method, the object itself is
passed as the first argument. So, harry.greet() translates
into Person.greet(harry).
In general, calling a method with a list of n arguments is equivalent to calling
the corresponding function with an argument list that is created by inserting
the method's object before the first argument.
For these reasons, the first argument of the function in class must be the
object itself. This is conventionally called self. It can be named otherwise but
we highly recommend to follow the convention.
Now you must be familiar with class object, instance object, function object,
method object and their differences.
Constructors in Python
Class functions that begin with double underscore __ are called special
functions as they have special meaning.
Of one particular interest is the __init__() function. This special function gets
called whenever a new object of that class is instantiated.
This type of function is also called constructors in Object Oriented
Programming (OOP). We normally use it to initialize all the variables.
class ComplexNumber:
def __init__(self, r=0, i=0):
self.real = r
self.imag = i
def get_data(self):
print(f'{self.real}+{self.imag}j')
Output
2+3j
(5, 0, 10)
Traceback (most recent call last):
File "<string>", line 27, in <module>
print(num1.attr)
AttributeError: 'ComplexNumber' object has no attribute 'attr'
We can even delete the object itself, using the del statement.
>>> c1 = ComplexNumber(1,3)
>>> del c1
>>> c1
Traceback (most recent call last):
...
NameError: name 'c1' is not defined
2. Python Inheritance
Inheritance enables us to define a class that takes all the functionality from a
parent class and allows us to add more. In this tutorial, you will learn to use
inheritance in Python.
Inheritance in Python
Inheritance is a powerful feature in object oriented programming.
It refers to defining a new class with little or no modification to an existing
class. The new class is called derived (or child) class and the one from
which it inherits is called the base (or parent) class.
class DerivedClass(BaseClass):
Derived class inherits features from the base class where new features can be
added to it. This results in re-usability of code.
class Polygon:
def __init__(self, no_of_sides):
self.n = no_of_sides
self.sides = [0 for i in range(no_of_sides)]
def inputSides(self):
self.sides = [float(input("Enter side "+str(i+1)+" : ")) for i in range(self.n)]
def dispSides(self):
for i in range(self.n):
print("Side",i+1,"is",self.sides[i])
This class has data attributes to store the number of sides n and magnitude of
each side as a list called sides.
The inputSides() method takes in the magnitude of each side
and dispSides() displays these side lengths.
A triangle is a polygon with 3 sides. So, we can create a class
called Triangle which inherits from Polygon. This makes all the attributes
of Polygon class available to the Triangle class.
We don't need to define them again (code reusability). Triangle can be
defined as follows.
class Triangle(Polygon):
def __init__(self):
Polygon.__init__(self,3)
def findArea(self):
a, b, c = self.sides
# calculate the semi-perimeter
s = (a + b + c) / 2
area = (s*(s-a)*(s-b)*(s-c)) ** 0.5
print('The area of the triangle is %0.2f' %area)
However, class Triangle has a new method findArea() to find and print the
area of the triangle. Here is a sample run.
>>> t = Triangle()
>>> t.inputSides()
Enter side 1 : 3
Enter side 2 : 5
Enter side 3 : 4
>>> t.dispSides()
Side 1 is 3.0
Side 2 is 5.0
Side 3 is 4.0
>>> t.findArea()
The area of the triangle is 6.00
We can see that even though we did not define methods
like inputSides() or dispSides() for class Triangle separately, we were able to
use them.
If an attribute is not found in the class itself, the search continues to the base
class. This repeats recursively, if the base class is itself derived from other
classes.
>>> isinstance(t,Triangle)
True
>>> isinstance(t,Polygon)
True
>>> isinstance(t,int)
False
>>> isinstance(t,object)
True
>>> issubclass(Polygon,Triangle)
False
>>> issubclass(Triangle,Polygon)
True
>>> issubclass(bool,int)
True
Example
class Base1:
pass
class Base2:
pass
class Base:
pass
class Derived1(Base):
pass
class Derived2(Derived1):
pass
# Output: True
print(issubclass(list,object))
# Output: True
print(isinstance(5.5,object))
# Output: True
print(isinstance("Hello",object))
In the multiple inheritance scenario, any specified attribute is searched first in
the current class. If not found, the search continues into parent classes in
depth-first, left-right fashion without searching the same class twice.
So, in the above example of MultiDerived class the search order is
[MultiDerived, Base1, Base2, object]. This order is also called linearization
of MultiDerived class and the set of rules used to find this order is
called Method Resolution Order (MRO).
MRO must prevent local precedence ordering and also provide monotonicity.
It ensures that a class always appears before its parents. In case of multiple
parents, the order is the same as tuples of base classes.
MRO of a class can be viewed as the __mro__ attribute or the mro() method.
The former returns a tuple while the latter returns a list.
>>> MultiDerived.__mro__
(<class '__main__.MultiDerived'>,
<class '__main__.Base1'>,
<class '__main__.Base2'>,
<class 'object'>)
>>> MultiDerived.mro()
[<class '__main__.MultiDerived'>,
<class '__main__.Base1'>,
<class '__main__.Base2'>,
<class 'object'>]
class X:
pass
class Y:
pass
class Z:
pass
# Output:
# [<class '__main__.M'>, <class '__main__.B'>,
# <class '__main__.A'>, <class '__main__.X'>,
# <class '__main__.Y'>, <class '__main__.Z'>,
# <class 'object'>]
print(M.mro())
Output
[<class '__main__.M'>, <class '__main__.B'>, <class '__main__.A'>, <class '__main__.X'>,
<class '__main__.Y'>, <class '__main__.Z'>, <class 'object'>]
p1 = Point(1, 2)
p2 = Point(2, 3)
print(p1+p2)
Output
Traceback (most recent call last):
File "<string>", line 9, in <module>
print(p1+p2)
TypeError: unsupported operand type(s) for +: 'Point' and 'Point'
Here, we can see that a TypeError was raised, since Python didn't know how
to add two Point objects together.
However, we can achieve this task in Python through operator overloading.
But first, let's get a notion about special functions.
>>> p1 = Point(2,3)
>>> print(p1)
<__main__.Point object at 0x00000000031F8CC0>
class Point:
def __init__(self, x = 0, y = 0):
self.x = x
self.y = y
def __str__(self):
return "({0},{1})".format(self.x,self.y)
class Point:
def __init__(self, x=0, y=0):
self.x = x
self.y = y
def __str__(self):
return "({0}, {1})".format(self.x, self.y)
p1 = Point(2, 3)
print(p1)
Output
(2, 3)
That's better. Turns out, that this same method is invoked when we use the
built-in function str() or format().
>>> str(p1)
'(2,3)'
>>> format(p1)
'(2,3)'
class Point:
def __init__(self, x=0, y=0):
self.x = x
self.y = y
def __str__(self):
return "({0},{1})".format(self.x, self.y)
def __str__(self):
return "({0},{1})".format(self.x, self.y)
p1 = Point(1, 2)
p2 = Point(2, 3)
print(p1+p2)
Output
(3,5)
Addition p1 + p2 p1.__add__(p2)
Subtraction p1 - p2 p1.__sub__(p2)
Multiplication p1 * p2 p1.__mul__(p2)
Power p1 ** p2 p1.__pow__(p2)
Division p1 / p2 p1.__truediv__(p2)
Bitwise OR p1 | p2 p1.__or__(p2)
def __str__(self):
return "({0},{1})".format(self.x, self.y)
p1 = Point(1,1)
p2 = Point(-2,-3)
p3 = Point(1,-1)
# use less than
print(p1<p2)
print(p2<p3)
print(p1<p3)
Output
True
False
False
Equal to p1 == p2 p1.__eq__(p2)
1. Python Modules
What are Modules in Python?
Consider a book store, in the book store there, is a lot of section like fiction
books, physics books, financial books, language books and a lot more, and in
each section, there are 100s of books present in just one section.
From the above example, consider the book store a folder, and section are
considered to python files, and last the books in each section are called
python functions, classes or python variables.
Formal Definition of Python Module
Python module can be defined as a python file that contains python
definitions and statements. It contains python code along with python
functions, classes, or python variables. From the above example, we can
consider that each section is a python module.
In order words, we can say that the python module is a file with
the .py extension.
Features of Python Module
Modules provide us the flexibility to organize the code logically.
Modules help to break down large programs into small manageable and organized files.
It provides reusability of code.
Example:
Let's create a small module.
Create a file named example.py. The name of the file is example.py but the
name of the module is example.
Now, in the example.py file, we will create a function called print_name().
# We are in the example module
def print_name(name):
"""
This function does not return anything.
It will just print your name.
"""
print("Hello! {}, You are in example module.".format(name))
Now, we have created a python module example.
Now, let's try whether it works for us or not.
Now, create a second file at the same location where we have created
our example module. Let's name that file as test.py.
Input:
import example
# now we use some of the math and random module function to check whether the modules are
working or not?
cos30 = math.cos(30)
tan10 = math.tan(10)
pie = math.pi
Variables in Module
Already discussed that modules contain functions, classes. But apart from
function and classes, modules in python also contain variables in them.
Variables like tuples, lists, dictionaries, objects, etc.
Example:
Let's create one more module variables, and saved the .py as variables.py
# This is variables module
## this is a function in module
def factorial(n):
if n == 1 || n == 0:
return 1
else:
return n * factorial(n-1)
fact_of_6 = variables.factorial(6)
power_of_6 = variables.power[6]
alphabet_2 = variables.alphabets[1]
Example 1
Input:
from math import *
print(pi)
print(sin(155))
print(tan(0))
Output:
3.141592653589793
-0.8733119827746476
0.0
Example 2
Input:
from math import sqrt, factorial, pi
print(sqrt(100))
print(factorial(10))
print(pi)
Output:
10.0
3628800
3.141592653589793
Naming a Module
In the last 2 examples, we have created 2 python
modules variables and example, and the modules files
as variables.py and example.py. So, in this case, we get our modules names
as their file names.
So, overall we can say that we will get the name of a module by saving our
python file.
Re-Naming a Module
Python provides an easy and flexible way to rename our module with a
specific name, and then use that name to call our module resources.
For re-naming a module, we use the as keyword. The syntax is as follow:
import math as m
import numpy as np
import random as r
print(m.pi)
print(r.randint(0,10))
print(np.__version__)
Output
3.141592653589793
10
1.22.0
Reloading a Module
We can import a module only once in a session.
Suppose you using 2 modules variables and example at the same in a python
file, and the variables module is updated while you were using these
modules, and you want the updated code of that module which is updated.
Now, we know that only one time we can import a module in our session.
So, we will use the reload function which is available in the imp module to
get the updated version of our module.
import variables
import imp
imp.reload(variables)
<module 'variables' from '.\\variables.py'>
print(dir(math))
Output:
['__doc__', '__loader__', '__name__', '__package__', '__spec__', 'acos', 'acosh', 'asin'
Introduction
The meaning of Packages lies in the word itself. Packages do the work of
organizing files in Python. Physically a package is a folder containing sub-
packages or one or more modules (or files). They provide a well-defined
organizational hierarchy of modules in Python. Packages usually are named
such that their usage is apparent to the user. Python packages ensure
modularity by dividing the packages into sub-packages, making the project
easier to manage and conceptually clear.
To better understand what Packages in Python mean, let’s take an example
from real life. Your family is shifting to a new apartment. Your mom has
asked you to pack your books, CDs and toys so that it’s easy to unpack and
organize them in your new house. The simplest way to do this would be to
pack these items separately into three unique packages ( boxes for CDs,
books and toys, respectively) and name them based on their utility.
Furthermore, you can also add sections to your book package based on the
genre of books. We can do the same thing for CDs and toys as well.
Have you come across this way of organization somewhere? The most
common example would be that of files on your phone. All your files would
be saved in a folder that could further be distinguished based on the source
(WhatsApp Documents, Downloads, etc.).
Every package in Python must contain an __init__.py file since it identifies
the folder as a package. It generally contains the initialization code but may
also be left empty. A possible hierarchical representation of the shifting items
package in the first example can be given below. Here, shifting items is our
main package. We further divide this into three sub packages: books, CDs
and toys. Each package and subpackage contains the
mandatory __init__.py file. Further, the sub packages contain files based on
their utility. Books contain two files fiction.py and non_fiction.py, the
package CDs contain music.py and dvd.py, and the package toys contain
soft_toys.py and games.py.
To use the edit module from the writing package in a new file test.py that you
have created, you write the following code in the test.py:
import Writing.Book.edit
To access a function called plagiarism_check() of the edit module, you use
the following code:
Writing.Book.edit.plagiarism_check()
The calling method seems lengthy and confusing, right? Another way to
import a module would be to simply import the package prefix instead of the
whole package and call the function required directly.
from Writing.Book import edit
plagiarism_check()
However, the above method may cause problems when 2 or more packages
have similarly named functions, and the call may be ambiguous. Thus, this
method is avoided in most cases.
Installing a Python Package Globally
To ensure system-wide use of the Python package just created, you need to
run the setup script. This script uses the setup() function from
the setuptools module. As a prerequisite, you need to have the latest versions
of pip and setuptools installed on your system. Pip and setuptools are usually
installed along with the Python binary installers. To upgrade the version of
pip, use the following command:
python -m pip install --upgrade pip
Once you confirm that you have the latest versions of pip and setuptools
installed, you can create a setup.py file in the main package (Writing) folder.
The setup() function that will be imported here takes various arguments like
the package's name, version, description, author, license, etc. The zip_safe
argument is used to know the mode of storage of the package (compressed or
uncompressed).
from setuptools import setup
setup(name='Writing',
version='1.0',
description='A Sample Package for all Writing Modules',
url='#',
author='username',
author_email='username@gmail.com',
license='MIT',
packages=['Writing'],
zip_safe=False)
Now to install the Writing package, open a terminal in your parent package
folder (Writing) and type in the following command:
pip install Writing
The Writing package, which contains functions to assist writing (like
count_words which is used to count the number of words in a string) will
now be available for use anywhere on the system and can be imported using
any script or interpreter by using the following commands:
import Writing
Writing.count_words(hello.txt)
This is how you create a system-wide package. If you want to create a
package that can be used by users globally, follow the following steps:
1. Organize your package well with a readme.md, license and other files.
Give an easy-to-understand and memorable name for your package.
2. Register an account on https://pypi.org.
3. Use twine upload dist/* to upload your package on PyPi and enter the
credentials you used to create the account. The package will be
uploaded directly to PyPi.
4. You can now install your Python package from PyPi using the
command python -m pip install [package_name] in your terminal.
Import Statement in Python
The files in packages that contain python code are called modules. Modules
are used to break down large code into smaller and more understandable
parts. Commonly used code (functions) can be written in the form of modules
and imported as and when needed, thus ensuring the reusability of code.
Let’s create a function prod(x,y) to multiply two numbers x and y, which are
passed as arguments to a function and store it in a module named product.py.
def prod(x, y):
res = x*y
return res
To import the prod function in another module or use it in the interactive
interpreter shell of Python, type:
import product
This statement does not import the module's functions into the symbol table.
To access functions of the given module, we use the dot (.) operator as
follows:
product.prod(3,5)
Output:
15
Python has a lot of in-built modules which can be accessed in the same way.
For example, to print the value of pi, we can import the math module.
Example:
import math
print(math.pi)
Output:
3.141592653589793
Example:
import math as m
print(m.pi)
Output:
3.141592653589793
Output:
SAFA
def greet():
print("Hello World!")
final.py - This is the file where the class calc needs to be imported. Since
calc lies in another directory (as shown in the directory structure above), we
import the importlib and importlib.util packages. A function called
module_directory contains two parameters: name_module and path that
represent the module's name and the module's path to be imported
respectively. A variable P is used to save the ModuleSpec, i.e. all the import-
related information required to load the module. This is done by calling the
spec_from_file method from importlib.util after passing the name_module
and path as parameters. Another variable import_module saves the module
from the ModuleSpec saved in the variable P. The exec_module is used to
execute the import_module module in its own namespace. The
import_module is then returned by the function. A result variable is used to
call the module_directory function. An object named object is then created by
instantiating the class calc from the imported module. Different functions of
the calc class can be then called by using the dot (.) operator as shown in the
code below.
import importlib, importlib.util
def module_directory(name_module, path):
P = importlib.util.spec_from_file_location(name_module, path)
import_module = importlib.util.module_from_spec(P)
P.loader.exec_module(import_module)
return import_module
result = module_directory("result", "../calc.py")
# Created a class object
object = result.calc()
Output:
25
15
100
Hello World!
# Defining the name of the object and positions for the object
nameTup = namedtuple('nameTup', 'value1, value2, value3')
myTuple = nameTup('Scaler', 'Topics', 'Python')
print('\n',myTuple)
using a list for creating the tuple hepls us to access the values of the tuple by
both using indexes and the names.
Output: The program will print the values from the tuple object by using the
indexes of the list passed and also by the names as their positions.
Using the indexes to get the values
Name is: ABC
Address is: Street No.1 ABC complex
Profession is: Teacher
Using the names to get the values
Name is: ABC
Address is: Street No.1 ABC complex
Profession is: Teacher
Simply printing the tuple
tuple1(name='ABC', address='Street No.1 ABC complex', profession='Teacher')
OrderedDict()
OrderedDict is a dictionary in which the keys are always inserted in the same
order. The order of the keys in a dictionary is preserved if they are inserted in
a specific order. Even if the value of the key is changed later, the position
will not change.
So if we will iterate over the Dictionary it will always print out the values in
the order they are inserted.
for importing the OrderedDict() we use the below syntax:
Syntax:
from collections import OrderedDict
myDict = OrderedDict()
After importing, you can simply just initialize the object and can use it
according to the code.
First define the structure using the OrderedDict() function and initialize the
dictionary object by passing the values according to the keys.
Further you can use inbuilt function for dictionary like:
d.clear()
d.get([, ])
d.items()
d.keys()
d.values()
d.pop([, ])
d.popitem()
d.update()
Lets see some examples related to OrderedDict() function.
Example 1:
First we will define the structure using the OrderedDict() function and we
will initialize the dictionary object by passing the values according to the
keys.
Now you can use the dictionary to get the key-values.
# Importing the dictionary
from collections import OrderedDict
mydict = defaultdict()
After importing, you can simply just initialize the object and can use it
according to the code.
First define the structure using the defaultdict() function and initialize the
dictionary object by passing the values according to the keys.
Further you can use inbuilt function for dictionary like:
d.clear()
d.get([, ])
d.items()
d.keys()
d.values()
d.pop([, ])
d.popitem()
d.update()
Lets see some examples related to defaultdict() function.
Example 1:
First we will define the structure using the defaultdict() function and we will
initialize the dictionary object by passing the values according to the keys.
Now you can use the dictionary to get the key-values.
# Importing the dictionary
from collections import defaultdict
# Making a list
l = [1,1,1,1,1,2,2,3,3,4]
# Making a list
stringList = ['a', 'b', 'c', 'c', 'c', 'b', 'b', 'b', 'a', 'a', 'a', 'a']
deque()
Deque is a more efficient variant of a list for adding and removing items
easily. It can add or remove items from the beginning or the end of the list.
It is a double ended queue which has insertion and deletion operations
available at both the ends.
For importing deque() we use the below syntax:
Syntax:
from collections import deque
deq = deque()
After importing, you can simply just initialize the deque() object and can use
it according to the code.
After importing, you can simply just initialize the object and pass the list in
the deque object.
Further you can use inbuilt function for deque like:
append(): append to right of deque()
appendleft(): append to left of deque()
pop(): pop an element from the right of the deque()
popleft(): pop for the left of deque()
index(ele, beg, end): get the index of the element from deque()
insert(i, a): insert at a particular index of the deque()
remove(): remove an element from deque
count(): count occurences of an element in deque()
reverse(): reverse the deque()
rotate(): rotate the deque() by specified number. Lets see some examples related to deque().
Example:
First we will initialize the deque object and we will pass the list in the deque
object.
Now we will be applying the removal and insertion on both the ends of the
deque.
We added the items at both ends using append() and appendleft() function
and then printed it.
Now finally we have removed the items from both the ends using pop() and
popleft() function.
# Importing the deque
from collections import deque
# Initialization
l = ['Hi', 'This', 'is', 'Scaler']
myDeq = deque(l)
myDeq.popleft()
print("Deque after removal from beginning is :",myDeq)
myDeq.insert(1, "New")
print("Deque after insertion is :",myDeq)
myDeq.remove("New")
print("Deque after removal of value New is :",myDeq)
myDeq.reverse()
print("Deque after reversing is :",myDeq)
myDeq.rotate(1)
print("Deque after rotation by 1 element is :",myDeq)
We have performed every above mentioned functions on our deque.
Output:
First simply a list is passed in the deque() function, Then we added the items
at both ends using append() and appendleft() function and then printed it.
Now finally we have removed the items from both the ends using pop() and
popleft() function.
Chainmap
ChainMap returns a list of dictionaries after combining them and chaining
them together. ChainMaps can encapsulate a multiple dictionaries into a
single object or unit and has no limitations for the number of dictionaries to
contain.
For importing Chainmap() we use the below syntax:
Syntax:
from collections import Chainmap
cm = chainmap(dic1, dic2)
After importing, you can simply just initialize the Chainmap() object by
passing the dictionaries you want to encapsulate.
Further you can use inbuilt function for chainmap like:
keys(): it is used to get all the keys of chainmap
values(): it is used to get all the values of chainmap
maps(): it is used to get all the keys and their particular values in the object
newchild(): It adds a new dictionary at the beginning of the chainmap
reversed(): It is used to reverse the chainmap object.
Lets see some examples related to Chainmap().
Example:
First we will initialize the Chainmap object and we will pass all the
dictionaries that we wanna encapsulate in the chainmap object.
Finally we will print the final chainmap object.
# Importing the deque
from collections import ChainMap
# Dictionary 1
dict1 = { 'k1' : 1, 'k2' : 2, 'k3': 3 }
# Dictionary 2
dict2 = { 'k4' : 4 }
dict3 = { 'k5' : 5 }
chain1 = finalChainMap.new_child(dict3)
print("chainmap after addition at beginning is :", chain1)
Output:
We Simply Just printed out the chainMap object after encapsulating the
dictionaries to the Object.
ChainMap({'k1': 1, 'k2': 2, 'k3': 3}, {'k4': 4})
All the keys of the chainmap is: ['k2', 'k3', 'k1', 'k4']
All the values the chainmap is: [2, 3, 1, 4]
All the keys-value pairs of the chainmap is: [{'k1': 1, 'k2': 2, 'k3': 3}, {'k4': 4}]
All the keys of the chainmap is: ['k2', 'k3', 'k1', 'k4']
chainmap after addition at beginning is: ChainMap({'k5': 5}, {'k1': 1, 'k2': 2, 'k3': 3}, {'k4': 4})
Reversed chainmap is: [{'k4': 4}, {'k1': 1, 'k2': 2, 'k3': 3}, {'k5': 5}]
Output:
We have a match!
This will give “We have a match!” as output as the given string starts with
Turing and ends with London. We will see how this works later in the article.
Python RegEx Expressions Functions
The ‘re’ package in python provides various functions to work with regular
expressions. We will discuss some commonly used ones.
S Function Description
No
1 findall(pattern,string) This matches all the occurrences of the pattern present in the string.
This matches the pattern which is present at any position in the string. This will
2 search(pattern,string)
match the first occurrence of the pattern.
3 split(pattern,string) This splits the string on the given pattern.
4 sub(pattern,rep_substring,string) This replaces one or more matching pattern in the string with the given substring.
Meta Characters
These are the characters which have special meaning. The following are some
of the meta characters with their uses.
S Meta
Description
No character
[ ](Square
1 This matches any single character in this bracket with the given string.
brackets)
This matches all the characters except the newline. If we pass this as a pattern in the findall() function i
2 . (Period)
will match with all the characters present in the string except newline characters.
This matches the given pattern at the start of the string.This is used to check if the string starts with a
3 ^ (Carret)
particular pattern or not.
This matches the given pattern at the end of string. This is used to check if the string ends with a pattern
4 $ (Dollar)
or not.
5 * (Star) This matches 0 or more occurrences of the pattern to its left.
6 + (Plus) This matches 1 or more occurrences of the pattern to its left.
? (Question
7 This matches 0 or 1 occurrence of the pattern to its left.
mark)
8 { } (Braces) This matches the specified number of occurrences of pattern present in the braces.
This works like ‘or’ condition. In this we can give two or more patterns. If the string contains at least
9 (Alternation)
one of the given patterns this will give a match.
10 ( ) (Group) This is used to group various regular expressions together and then find a match in the string.
11 \ (Backslash) This is used to match special sequences or can be used as escape characters also.
Output:
Result for \A = ['Alan']
-------------------------------------------------------------------------------
Result for \b = ['Lon']
-------------------------------------------------------------------------------
Result for \b = ['ring']
-------------------------------------------------------------------------------
Result for \B = ['on', 'on']
-------------------------------------------------------------------------------
Result for \d = ['2', '3', '1', '9', '1', '2']
-------------------------------------------------------------------------------
Result for \D = ['A', 'l', 'a', 'n', ' ', 'T', 'u', 'r', 'i', 'n', 'g', ' ', 'w', 'a', 's', ' ', 'b', 'o', 'r', 'n', ' ', 'o', 'n', ' ', ' ', 'J',
'u', 'n', 'e', ' ', ' ', 'i', 'n', ' ', 'L', 'o', 'n', 'd', 'o', 'n', '.']
-------------------------------------------------------------------------------
Result for \s = [' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ']
-------------------------------------------------------------------------------
Result for \S = ['A', 'l', 'a', 'n', 'T', 'u', 'r', 'i', 'n', 'g', 'w', 'a', 's', 'b', 'o', 'r', 'n', 'o', 'n', '2', '3', 'J', 'u', 'n', 'e',
'1', '9', '1', '2', 'i', 'n', 'L', 'o', 'n', 'd', 'o', 'n', '.']
-------------------------------------------------------------------------------
Result for \w = ['A', 'l', 'a', 'n', 'T', 'u', 'r', 'i', 'n', 'g', 'w', 'a', 's', 'b', 'o', 'r', 'n', 'o', 'n', '2', '3', 'J', 'u', 'n', 'e',
'1', '9', '1', '2', 'i', 'n', 'L', 'o', 'n', 'd', 'o', 'n']
-------------------------------------------------------------------------------
Result for \W = [' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', '.']
-------------------------------------------------------------------------------
Result for \Z = ['London.']
Sets
A set is a set of characters inside the square bracket which is treated as a
pattern. Given below are some examples of set:
No Set Description
1 [abcd] Gives a match if the string contains a,b,c or d.
2 [a-z] Gives a match if the string contains any character from a to z.
3 [A-Z] Gives a match if the string contains any character from A to Z.
4 [0-9] Gives a match if string contains digits from 0 to 9
[a-zA-Z0-
5 Gives a match if any of the above conditions holds true.
9]
6 [^a-zA-Z]Gives a match if the string doesn’t contain any alphabet.
Gives a match if the string contains any of these characters. When these characters are in square brackets
7 [%&$#@*]
they are treated as normal characters.
findall(pattern, string)
This function is the same as search but it matches all the occurrences of the
pattern in the given string and returns a list. The list contains the number of
times it is present in the string.
Ex: The following example will make it clear.
import re
text = '''Alan Turing was a pioneer of theoretical computer science and artificial intelligence. He was
born on 23 June 1912 in Maida Vale, London'''
res = re.findall('Turing',text)
print("Result = {}".format(res))
Output:
Result = ['Turing']
In the output you can clearly see that the function finds a match for the
pattern ‘Turing’’. It is advisable to use findall while searching for a pattern in
a string as it covers both match and search functions.
search(pattern, string)
This is the same as match function but this function can search patterns
irrespective of the position at which the pattern is present. The pattern can be
present anywhere in the string. This function matches the first occurrence of
the pattern.
Ex: The following example shows how to use the function
import re
text = '''Alan Turing was a pioneer of theoretical computer science and artificial intelligence. He was
born on 23 June 1912 in Maida Vale, London'''
res = re.search('Turing',text)
print("Result = {} and start,end position = {}".format(res,res.span()))
Output:
Result = <re.Match object; span=(5, 11), match='Turing'> and start,end position = (5, 11)
The function returns re.Match object if pattern if present in the string else
returns None.
We can also get the start and end positions of matching pattern by calling
span method on the re.Match object.
split(pattern, string)
This function splits a string on the given pattern. This returns the result as a
list after splitting. The example given below will make it clear.
import re
text = '''Alan Turing was a pioneer of theoretical computer science and artificial intelligence. He was
born on 23 June 1912 in Maida Vale, London'''
res = re.sub('theoretical','practical',text)
print("Result = {}".format(res))
Output:
Result = Alan Turing was a pioneer of practical computer science and artificial intelligence. He was
born on 23 June 1912 in Maida Vale, London
In the output we can see that the function replaced the pattern ‘theoretical’
with the given substring ‘practical’’. This function will replace all the
patterns present in the string with the given substring.
Match Object
Whenever we call any regex method/function it searches the pattern in the
string. If it finds a match then it returns a match object else return None. We
will see how the match object looks like and how to access methods and
properties of that object. Let’s search a pattern in a string and print the match
object.
import re
text = '''Alan Turing was a pioneer of theoretical computer science and artificial intelligence. He was
born on 23 June 1912 in Maida Vale, London'''
# Searches the pattern in the string.
res = re.search('computer',text)
print("Match object = {}".format(res))
Output:
Match object = <re.Match object; span=(41, 49), match='computer'>
In the example above we can see that if a match happens then the re.Match
object is returned. If there is no match then None will be returned.
Now we will see the attributes and properties of re.Match objects one by one.
They are as follows:
match.group(): This returns the part of the string where the match was
there.
match.start(): This returns the start position of the matching pattern in
the string.
match.end(): This returns the end position of the matching pattern in
the string.
match.span(): This returns a tuple which has start and end positions of
matching pattern.
match.re: This returns the pattern object used for matching.
match.string: This returns the string given for matching.
Using r prefix before regex: This is used to convert the pattern to raw
string.This means any special character will be treated as normal
character. Ex: \ character will not be treated as an escape character if
we use r before the pattern.
Now we will see examples for each of the functions mentioned above.
import re
text = '''Alan Turing was a pioneer of theoretical computer science and artificial intelligence. He was
born on 23 June 1912 in Maida Vale, London'''
# Searches the pattern in the string.
res = re.search('computer',text)
print("Match object = {}".format(res))
print("--"*30)
print("group method output = ",res.group())
print("--"*30)
print("start method output = ",res.start())
print("--"*30)
print("end method output = ",res.end())
print("--"*30)
print("span method output = ",res.span())
print("--"*30)
print("re attribute output = ",res.re)
print("--"*30)
print("string attribute output = ",res.string)
print("--"*30)
5. Python Datetime
Introduction to Python Datetime
Ever wondered how does the banking system manages date and time, like
how do they calculate the remaining tenure of a loan. Well, the answer lies in
the functions of python datetime module of python. Read along to know
more.
DateTime module is provided in Python to work with dates and times. In
python DateTime, is an inbuilt module rather than being a primitive data
type, We just have to import the module mentioned above to work with dates
as date object.
There are several classes in the datetime python module of python which help
to deal with the date & time. Moreover, there are so many functions of these
classes from which we can extract the date and time.
How to Use Date and Datetime Class?
Firstly, you need to import the python datetime module of python using the
following line of code, so that you can use its inbuilt classes to get current
date & time etc.
import datetime
1. Now, we can use the date class of the datetime module. This class helps
to convert the numeric date attributes to date format in the form
of YYYY-MM-DD. This class accepts 3 attributes Year, Month & Day
in the following order (Year, Month, Day).
Syntax
import datetime
# This will convert the numeral date to the datetime object in the form of
YYYY-MM-DD hr:min:s:ms.
Example
import datetime
# returns the time with the value which are specified by the user in the attributes
var2 = datetime.time(hour=?, minute=?, second=?, microsecond=?)
Example
import datetime
time1 = datetime.time()
print('without passing any attribute: ', time1)
time2 = time2.replace(minute=00)
print('by replacing the minute attribute in time2: ', time2)
Output
without passing any attribute: 00:00:00
by passing hour, minute and second attributes: 12:55:50
by replacing the hour attribute in time1: 17:00:00
by replacing the minute attribute in time2: 12:00:50
Explanation of Code
In the above code we imported the python datetime library.
We then saved two different type of time object into the variable (one
having no attributes and other with some attributes).
Then we used replace() to change the value of above generated time.
Then we print the variable to get different times according to the
attributes we passed in the time class.
Datetime
Now, we can use the datetime python class of the datetime module. This class
helps to convert the numeric date & time attributes to date-time format in the
form of YYYY-MM-DD hr:min:s:ms.
This class accepts 7 attributes Year, Month, Day, Hour, Minutes, Seconds,
MilliSeconds in the following order (Year, Month, Day, Hour, Minutes,
Seconds, MilliSeconds).
Syntax
import datetime
# This will convert the numeral date to the datetime object in the form of
YYYY-MM-DD hr:min:s:ms.
Example
import datetime
# This will return the exact date & time object by using all the attributes
used by the user.
Example
import datetime
Example
import datetime
import pytz
localtime= datetime.datetime.now()
print('local time: ', localtime)
utctime = datetime.datetime.now(pytz.utc)
print('UTC time zone: ', utctime)
ustime = datetime.datetime.now(pytz.timezone('US/CENTRAL'))
print('US time zone: ', ustime)
Output
local time: 2021-10-15 01:37:39.603581
UTC time zone: 2021-10-14 20:07:39.603716+00:00
US time zone: 2021-10-14 15:07:39.626474-05:00
Explanation of Example
In the above code we have imported the datetime and pytz libraries.
We then saved different date time object on the basis of time zones into
the variable.
Then we print the variables whose output is down below.
Date Class
.date()
This class helps to convert the numeric date attributes to date format in the
form of YYYY-MM-DD.
This class accepts 3 attributes Year, Month & Day in the following order
(Year, Month, Day).
Syntax
import datetime
Output
dateByUser: 2018-12-31
type of dateByUser: <class 'datetime.date'>
Explanation of Example
In the above code we have imported the datetime library.
We then saved the date time object into the variable.
Then we print the variable and also the type of variable whose output is
down below.
As we can see that we have changed the numeral datatime to the datetime
object and the same can be verified by the type of object printed below.
.today()
This function of Date class returns the today's date on the output screen in the
format of 'YYYY-MM-DD'.
Example
import datetime
.min
The .min function returns the earliest representable date. This uses 3
attributes 'MinYear', 'Month' & 'Day' in the following order
(datetime.MINYEAR, Month, Day).
Example
import datetime
.max
The .max function returns the latest representable date. This uses 3 attributes
'MaxYear', 'Month' & 'Day' in the following order (datetime.MAXYEAR,
Month, Day).
Example
import datetime
.day
This function returns the days count of the date which is entered by the user.
It gives count from 1 to 30/31 (or precisely number of days in the specified
month) both inclusive.
Example
import datetime
dateByUser = datetime.date(2020, 5, 27)
print('day count of the date entered by the user: ', dateByUser.day)
Output
day count of the date entered by the user: 27
.month
This function returns the month count of the date which is entered by the
user. It gives count from 1 to 12 both inclusive.
Example
import datetime
.year
This function returns the year count of the date which is entered by the user.
It give count from MINYEAR to MAXYEAR both inclusive.
Example
import datetime
strftime()
This function helps us to represent date in different types of formats such as
Short/Long days(Mon, Monday), Short/Long years(21, 2021),
Numeric/Alphabetic month(03, March).
The image below represents the notations for the representation of different
types of formats of the dates.
Example
import datetime
todaydatetime = datetime.datetime.now()
print('current date and time: ', todaydatetime)
print('fetched out year from current date: ', todaydatetime.strftime('%Y'))
print('fetched out month from current date: ', todaydatetime.strftime('%B'))
print('fetched out day from current date: ', todaydatetime.strftime('%A'))
print('formatted the time from above generated date time: ', todaydatetime.strftime('%H:%M:%S'))
print(todaydatetime.strftime('formatted both date and time: ', "%d/%m/%Y, %H:%M:%S"))
Output
current date and time: 2021-10-17 12:14:52.394962
fetched out year from current date: 2021
fetched out month from current date: October
fetched out day from current date: Sunday
formatted the time from above generated date time: 12:14:52
formatted both date and time: 17/10/2021, 12:14:52
Explaination of Example
1. Firstly, we stored the current date and time into our variable
'todaydatetime'.
2. After storing the date we print the actual format of date & time
generated.
3. We have used '%Y' for extracting year from the above mentioned date.
4. We have used '%B' for extracting month from the date.
5. We have used '%A' for extracting day from the date.
6. We have use different time formatting formatters '%H, %M & %S' for
formatting the time.
7. Lastly, we format both time and date.
Current Date
We have two ways to find out the current date which are mentioned below:
1. .today()
2. .now()
.today()
.today() is used to return the current local date. We just can use it from the
date class of datetime module to get the current date.
Code
import datetime
currentdate = datetime.date.today()
print('current date: ', currentdate)
Output
current date: 2021-10-17
.now()
.now() is used to return the current local date and time as datetime object. So
what we can do is, we can format the datetime object to get only current date.
Code
import datetime
currentdate = datetime.datetime.now()
Use of datetime.strptime()
There is a function strptime() of datetime python class of python datetime
module which is used for conversion of timestamp string to the datetime
python objects.
.strptime() accepts 2 attributes timestamp and format in which we can convert
it to datetime object.
Syntax
import datetime
datetimeObj = datetime.datetime.strptime('2002-01-02T10::52::59.456777'
,'%Y-%m-%dT%H::%M::%S.%f')
var1ToTimestamp = var1.timestamp()
*Example
import datetime
timestamp = datetime.datetime.now()
timestamp = datetime.datetime.now()
timestamp = datetime.datetime.now()
Code
import datetime
Iterators in Python
Iterators are everywhere in Python. They are elegantly implemented
within for loops, comprehensions, generators etc. but are hidden in plain
sight.
Iterator in Python is simply an object that can be iterated upon. An object
which will return data, one element at a time.
Technically speaking, a Python iterator object must implement two special
methods, __iter__() and __next__(), collectively called the iterator
protocol.
An object is called iterable if we can get an iterator from it. Most built-in
containers in Python like: list, tuple, string etc. are iterables.
The iter() function (which in turn calls the __iter__() method) returns an
iterator from them.
Iterating Through an Iterator
We use the next() function to manually iterate through all the items of an
iterator. When we reach the end and there is no more data to be returned, it
will raise the StopIteration Exception. Following is an example.
# define a list
my_list = [4, 7, 0, 3]
# Output: 4
print(next(my_iter))
# Output: 7
print(next(my_iter))
# Output: 0
print(my_iter.__next__())
# Output: 3
print(my_iter.__next__())
4
7
0
3
Traceback (most recent call last):
File "<string>", line 24, in <module>
next(my_iter)
StopIteration
A more elegant way of automatically iterating is by using the for loop. Using
this, we can iterate over any object that can return an iterator, for example
list, string, file etc.
# infinite loop
while True:
try:
# get the next item
element = next(iter_obj)
# do something with element
except StopIteration:
# if StopIteration is raised, break from loop
break
class PowTwo:
"""Class to implement an iterator
of powers of two"""
def __iter__(self):
self.n = 0
return self
def __next__(self):
if self.n <= self.max:
result = 2 ** self.n
self.n += 1
return result
else:
raise StopIteration
# create an object
numbers = PowTwo(3)
1
2
4
8
Traceback (most recent call last):
File "/home/bsoyuj/Desktop/Untitled-1.py", line 32, in <module>
print(next(i))
File "<string>", line 18, in __next__
raise StopIteration
StopIteration
We can also use a for loop to iterate over our iterator class.
>>> for i in PowTwo(5):
... print(i)
...
1
2
4
8
16
32
def __iter__(self):
self.num = 1
return self
def __next__(self):
num = self.num
self.num += 2
return num
>>> a = iter(InfIter())
>>> next(a)
1
>>> next(a)
3
>>> next(a)
5
>>> next(a)
7
And so on...
Be careful to include a terminating condition, when iterating over these types
of infinite iterators.
The advantage of using iterators is that they save resources. Like shown
above, we could get all the odd numbers without storing the entire number
system in memory. We can have infinite items (theoretically) in finite
memory.
There's an easier way to create iterators in Python. To learn more
visit: Python generators using yield.
2. Python Generators
Generators in Python
There is a lot of work in building an iterator in Python. We have to
implement a class with __iter__() and __next__() method, keep track of
internal states, and raise StopIteration when there are no values to be
returned.
This is both lengthy and counterintuitive. Generator comes to the rescue in
such situations.
Python generators are a simple way of creating iterators. All the work we
mentioned above are automatically handled by generators in Python.
Simply speaking, a generator is a function that returns an object (iterator)
which we can iterate over (one value at a time).
n += 1
print('This is printed second')
yield n
n += 1
print('This is printed at last')
yield n
An interactive run in the interpreter is given below. Run these in the Python
shell to see the output.
>>> # Local variables and theirs states are remembered between successive calls.
>>> next(a)
This is printed second
2
>>> next(a)
This is printed at last
3
>>> # Finally, when the function terminates, StopIteration is raised automatically on further calls.
>>> next(a)
Traceback (most recent call last):
...
StopIteration
>>> next(a)
Traceback (most recent call last):
...
StopIteration
One interesting thing to note in the above example is that the value of
variable n is remembered between each call.
Unlike normal functions, the local variables are not destroyed when the
function yields. Furthermore, the generator object can be iterated only once.
To restart the process we need to create another generator object using
something like a = my_gen().
One final thing to note is that we can use generators with for loops directly.
This is because a for loop takes an iterator and iterates over it
using next() function. It automatically ends when StopIteration is raised.
n += 1
print('This is printed second')
yield n
n += 1
print('This is printed at last')
yield n
Output
o
l
l
e
h
print(list_)
print(generator)
Output
[1, 9, 36, 100]
<generator object <genexpr> at 0x7f5d4eb4bf50>
We can see above that the generator expression did not produce the required
result immediately. Instead, it returned a generator object, which produces
items only on demand.
Here is how we can start getting items from the generator:
# Initialize the list
my_list = [1, 3, 6, 10]
print(next(a))
print(next(a))
print(next(a))
next(a)
1
9
36
100
Traceback (most recent call last):
File "<string>", line 15, in <module>
StopIteration
1. Easy to Implement
Generators can be implemented in a clear and concise way as compared to
their iterator class counterpart. Following is an example to implement a
sequence of power of 2 using an iterator class.
class PowTwo:
def __init__(self, max=0):
self.n = 0
self.max = max
def __iter__(self):
return self
def __next__(self):
if self.n > self.max:
raise StopIteration
result = 2 ** self.n
self.n += 1
return result
The above program was lengthy and confusing. Now, let's do the same using
a generator function.
def PowTwoGen(max=0):
n=0
while n < max:
yield 2 ** n
n += 1
2. Memory Efficient
A normal function to return a sequence will create the entire sequence in
memory before returning the result. This is an overkill, if the number of items
in the sequence is very large.
Generator implementation of such sequences is memory friendly and is
preferred since it only produces one item at a time.
def all_even():
n=0
while True:
yield n
n += 2
4. Pipelining Generators
Multiple generators can be used to pipeline a series of operations. This is best
illustrated using an example.
Suppose we have a generator that produces the numbers in the Fibonacci
series. And we have another generator for squaring numbers.
If we want to find out the sum of squares of numbers in the Fibonacci series,
we can do it in the following way by pipelining the output of generator
functions together.
def fibonacci_numbers(nums):
x, y = 0, 1
for _ in range(nums):
x, y = y, x+y
yield x
def square(nums):
for num in nums:
yield num**2
print(sum(square(fibonacci_numbers(10))))
Output
4895
3. Python Closures
Nonlocal variable in a nested function
Before getting into what a closure is, we have to first understand what a
nested function and nonlocal variable is.
A function defined inside another function is called a nested function. Nested
functions can access variables of the enclosing scope.
In Python, these non-local variables are read-only by default and we must
declare them explicitly as non-local (using nonlocal keyword) in order to
modify them.
Following is an example of a nested function accessing a non-local variable.
def print_msg(msg):
# This is the outer enclosing function
def printer():
# This is the nested function
print(msg)
printer()
Output
Hello
We can see that the nested printer() function was able to access the non-
local msg variable of the enclosing function.
def print_msg(msg):
# This is the outer enclosing function
def printer():
# This is the nested function
print(msg)
Output
Hello
That's unusual.
The print_msg() function was called with the string "Hello" and the returned
function was bound to the name another. On calling another(), the message
was still remembered although we had already finished executing
the print_msg() function.
This technique by which some data ("Hello in this case) gets attached to the
code is called closure in Python.
This value in the enclosing scope is remembered even when the variable goes
out of scope or the function itself is removed from the current namespace.
Try running the following in the Python shell to see the output.
Here, the returned function still works even when the original function was
deleted.
# Multiplier of 3
times3 = make_multiplier_of(3)
# Multiplier of 5
times5 = make_multiplier_of(5)
# Output: 27
print(times3(9))
# Output: 15
print(times5(3))
# Output: 30
print(times5(times3(2)))
Output
27
15
30
>>> make_multiplier_of.__closure__
>>> times3.__closure__
(<cell at 0x0000000002D155B8: int object at 0x000000001E39B6E0>,)
The cell object has the attribute cell_contents which stores the closed value.
>>> times3.__closure__[0].cell_contents
3
>>> times5.__closure__[0].cell_contents
5
4. Python Decorators
A decorator takes in a function, adds some functionality and returns it. In this
tutorial, you will learn how you can create a decorator and why you should
use it.
Decorators in Python
Python has an interesting feature called decorators to add functionality to an
existing code.
This is also called metaprogramming because a part of the program tries to
modify another part of the program at compile time.
second = first
second("Hello")
Output
Hello
Hello
When you run the code, both functions first and second give the same output.
Here, the names first and second refer to the same function object.
Now things start getting weirder.
Functions can be passed as arguments to another function.
If you have used functions like map, filter and reduce in Python, then you
already know about this.
Such functions that take other functions as arguments are also called higher
order functions. Here is an example of such a function.
def inc(x):
return x + 1
def dec(x):
return x - 1
>>> operate(inc,3)
4
>>> operate(dec,3)
2
Furthermore, a function can return another function.
def is_called():
def is_returned():
print("Hello")
return is_returned
new = is_called()
# Outputs "Hello"
new()
Output
Hello
def ordinary():
print("I am ordinary")
When you run the following codes in shell,
>>> ordinary()
I am ordinary
pretty = make_pretty(ordinary)
The function ordinary() got decorated and the returned function was given
the name pretty.
We can see that the decorator function added some new functionality to the
original function. This is similar to packing a gift. The decorator acts as a
wrapper. The nature of the object that got decorated (actual gift inside) does
not alter. But now, it looks pretty (since it got decorated).
Generally, we decorate a function and reassign it as,
ordinary = make_pretty(ordinary).
This is a common construct and for this reason, Python has a syntax to
simplify this.
We can use the @ symbol along with the name of the decorator function and
place it above the definition of the function to be decorated. For example,
@make_pretty
def ordinary():
print("I am ordinary")
is equivalent to
def ordinary():
print("I am ordinary")
ordinary = make_pretty(ordinary)
This function has two parameters, a and b. We know it will give an error if
we pass in b as 0.
>>> divide(2,5)
0.4
>>> divide(2,0)
Traceback (most recent call last):
...
ZeroDivisionError: division by zero
Now let's make a decorator to check for this case that will cause the error.
def smart_divide(func):
def inner(a, b):
print("I am going to divide", a, "and", b)
if b == 0:
print("Whoops! cannot divide")
return
return func(a, b)
return inner
@smart_divide
def divide(a, b):
print(a/b)
This new implementation will return None if the error condition arises.
>>> divide(2,5)
I am going to divide 2 and 5
0.4
>>> divide(2,0)
I am going to divide 2 and 0
Whoops! cannot divide
def works_for_all(func):
def inner(*args, **kwargs):
print("I can decorate any function")
return func(*args, **kwargs)
return inner
def star(func):
def inner(*args, **kwargs):
print("*" * 30)
func(*args, **kwargs)
print("*" * 30)
return inner
def percent(func):
def inner(*args, **kwargs):
print("%" * 30)
func(*args, **kwargs)
print("%" * 30)
return inner
@star
@percent
def printer(msg):
print(msg)
printer("Hello")
Output
******************************
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
Hello
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
******************************
@star
@percent
def printer(msg):
print(msg)
is equivalent to
def printer(msg):
print(msg)
printer = star(percent(printer))
The order in which we chain decorators matter. If we had reversed the order
as,
@percent
@star
def printer(msg):
print(msg)
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
******************************
Hello
******************************
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
class Celsius:
def __init__(self, temperature = 0):
self.temperature = temperature
def to_fahrenheit(self):
return (self.temperature * 1.8) + 32
def to_fahrenheit(self):
return (self.temperature * 1.8) + 32
Output
37
98.60000000000001
The extra decimal places when converting into Fahrenheit is due to the
floating point arithmetic error. To learn more, visit Python Floating Point
Arithmetic Error.
Whenever we assign or retrieve any object attribute like temperature as
shown above, Python searches it in the object's built-in __dict__ dictionary
attribute.
>>> human.__dict__
{'temperature': 37}
def to_fahrenheit(self):
return (self.get_temperature() * 1.8) + 32
# getter method
def get_temperature(self):
return self._temperature
# setter method
def set_temperature(self, value):
if value < -273.15:
raise ValueError("Temperature below -273.15 is not possible.")
self._temperature = value
As we can see, the above method introduces two
new get_temperature() and set_temperature() methods.
Furthermore, temperature was replaced with _temperature. An
underscore _ at the beginning is used to denote private variables in Python.
def to_fahrenheit(self):
return (self.get_temperature() * 1.8) + 32
# getter method
def get_temperature(self):
return self._temperature
# setter method
def set_temperature(self, value):
if value < -273.15:
raise ValueError("Temperature below -273.15 is not possible.")
self._temperature = value
Output
37
98.60000000000001
Traceback (most recent call last):
File "<string>", line 30, in <module>
File "<string>", line 16, in set_temperature
ValueError: Temperature below -273.15 is not possible.
However, the bigger problem with the above update is that all the programs
that implemented our previous class have to modify their code
from obj.temperature to obj.get_temperature() and all expressions
like obj.temperature = val to obj.set_temperature(val).
This refactoring can cause problems while dealing with hundreds of
thousands of lines of codes.
All in all, our new update was not backwards compatible. This is
where @property comes to rescue.
def to_fahrenheit(self):
return (self.temperature * 1.8) + 32
# getter
def get_temperature(self):
print("Getting value...")
return self._temperature
# setter
def set_temperature(self, value):
print("Setting value...")
if value < -273.15:
raise ValueError("Temperature below -273.15 is not possible")
self._temperature = value
def to_fahrenheit(self):
return (self.temperature * 1.8) + 32
# getter
def get_temperature(self):
print("Getting value...")
return self._temperature
# setter
def set_temperature(self, value):
print("Setting value...")
if value < -273.15:
raise ValueError("Temperature below -273.15 is not possible")
self._temperature = value
print(human.temperature)
print(human.to_fahrenheit())
human.temperature = -300
Output
Setting value...
Getting value...
37
Getting value...
98.60000000000001
Setting value...
Traceback (most recent call last):
File "<string>", line 31, in <module>
File "<string>", line 18, in set_temperature
ValueError: Temperature below -273 is not possible
As we can see, any code that retrieves the value of temperature will
automatically call get_temperature() instead of a dictionary (__dict__) look-
up. Similarly, any code that assigns a value to temperature will automatically
call set_temperature().
We can even see above that set_temperature() was called even when we
created an object.
>>> c.to_fahrenheit()
Getting value
98.60000000000001
>>> property()
<property object at 0x0000000003239B38>
def to_fahrenheit(self):
return (self.temperature * 1.8) + 32
@property
def temperature(self):
print("Getting value...")
return self._temperature
@temperature.setter
def temperature(self, value):
print("Setting value...")
if value < -273.15:
raise ValueError("Temperature below -273 is not possible")
self._temperature = value
# create an object
human = Celsius(37)
print(human.temperature)
print(human.to_fahrenheit())
coldest_thing = Celsius(-300)
Output
Setting value...
Getting value...
37
Getting value...
98.60000000000001
Setting value...
Traceback (most recent call last):
File "", line 29, in
File "", line 4, in __init__
File "", line 18, in temperature
ValueError: Temperature below -273 is not possible
import re
The module defines several functions and constants to work with RegEx.
re.findall()
The re.findall() method returns a list of strings containing all matches.
Example 1: re.findall()
# Program to extract numbers from a string
import re
Example 2: re.split()
import re
If the pattern is not found, re.split() returns a list containing the original
string.
You can pass maxsplit argument to the re.split() method. It's the maximum
number of splits that will occur.
import re
# maxsplit = 1
# split only at the first occurrence
result = re.split(pattern, string, 1)
print(result)
By the way, the default value of maxsplit is 0; meaning all possible splits.
re.sub()
The syntax of re.sub() is:
The method returns a string where matched occurrences are replaced with the
content of replace variable.
Example 3: re.sub()
# Program to remove all whitespaces
import re
# multiline string
string = 'abc 12\
de 23 \n f45 6'
# empty string
replace = ''
# Output: abc12de23f456
If the pattern is not found, re.sub() returns the original string.
You can pass count as a fourth parameter to the re.sub() method. If omited, it
results to 0. This will replace all occurrences.
import re
# multiline string
string = 'abc 12\
de 23 \n f45 6'
# Output:
# abc12de 23
# f45 6
re.subn()
The re.subn() is similar to re.sub() except it returns a tuple of 2 items
containing the new string and the number of substitutions made.
Example 4: re.subn()
# Program to remove all whitespaces
import re
# multiline string
string = 'abc 12\
de 23 \n f45 6'
# empty string
replace = ''
# Output: ('abc12de23f456', 4)
re.search()
The re.search() method takes two arguments: a pattern and a string. The
method looks for the first location where the RegEx pattern produces a match
with the string.
If the search is successful, re.search() returns a match object; if not, it
returns None.
Example 5: re.search()
import re
if match:
print("pattern found inside the string")
else:
print("pattern not found")
Match object
You can get methods and attributes of a match object using dir() function.
Some of the commonly used methods and attributes of match objects are:
match.group()
The group() method returns the part of the string where there is a match.
if match:
print(match.group())
else:
print("pattern not found")
# Output: 801 35
>>> match.group(1)
'801'
>>> match.group(2)
'35'
>>> match.group(1, 2)
('801', '35')
>>> match.groups()
('801', '35')
>>> match.start()
2
>>> match.end()
8
The span() function returns a tuple containing start and end index of the
matched part.
>>> match.span()
(2, 8)
>>> match.re
re.compile('(\\d{3}) (\\d{2})')
>>> match.string
'39801 356, 2102 1111'
Because, Python 3 onwards the syntax for using the print statement has
changed. Similarly if you forget to add colon(:) at the end of the if condition,
you will get a SyntaxError:
if 7 > 5
print("Yo Yo!")
SyntaxError: invalid syntax
Hence syntax errors are the most basic type of errors that you will encounter
while coding in python and these can easily be fixed by seeing the error
message and correcting the code as per python syntax.
b=0
b=0
The above code leads to exception and the exception message is printed as
output on the console.
If we use the try and except block, we can handle this exception gracefully.
# try block
try:
a = 10
b=0
except:
try:
a = 10
b=0
except:
try:
except(ValueError, ZeroDivisionError):
print("Please check the input value: It should be an integer greater than 0")
here we have handled both the exceptions using a single except block while
showing a meaningful message to the user.
try:
except(ZeroDivisionError):
except(ValueError):
except:
In the code above the first except block will handle the ZeroDivisionError,
second except block will handle the ValueError and for any other exception
that might occur we have the third except block.
In the coming tutorials we will learn about finally block and how to raise an
exception using the raise keyword.
try:
a = 10
b=0
except(ZeroDivisionError):
finally:
print("Code execution Wrap up!")
Clearly the finally block was the first to be printed on the console follwed by
the first exception message and then the second exception message.
4. Python Exception
Handling: raise Keyword
While the try and except block are for handling exceptions, the raise keyword
on the contrary is to raise an exception.
Following is the syntax:
raise EXCEPTION_CLASS_NAME
Taking a simple usage example:
raise ZeroDivisionError
If you have a piece of code where along with exception handling you have
put in place some conditional statements to validate input etc, then in case of
the conditions failing we can either just print a message or simple raise an
exception which can then get handled by the common exception handling
mechanism.
See the code below,
a = 10
b=9
try:
print(a/b)
except ZeroDivisionError:
b=9
try:
if a < 0 or b < 0:
raise ZeroDivisionError
print(a/b)
except ZeroDivisionError:
When we use the raise keyword, it's not necessary to provide an exception
class along with it. When we don't provide any exception class name with
the raise keyword, it reraises the exception that last occured.
This is used generally inside an except code block to reraise an exception
which is catched.
For example
a = 10
b=0
try:
print(a/b)
except ZeroDivisionError:
raise
1. Introduction to Multithreading In
Python
Threads
Threads are lightweight processes (subparts of a large process) that can run
concurrently in parallel to each other, where each thread can perform some
task. Threads are usually contained in processes. More than one thread can
exist within the same process. Within the same process, threads share
memory and the state of the process.
Types Of Thread
What is Multithreading?
Now that we have a basic idea about what threads are, let's try to understand
the concept of Multithreading.
Modern computers have a CPU that has multiple processing cores and each
of these cores can run many threads simultaneously which gives us the ability
to perform several tasks concurrently. This process of running multiple
Threads concurrently to perform tasks parallely is called Multithreading.
Multithreading provides the following benefits:
Multiple threads within a process share the same data space and can,
therefore, share information or communicate with each other more
easily than if they were separate processes.
Threads do not require much memory overhead; they are cheaper than
processes in terms of memory requirements.
Multithreaded programs can run faster on computer systems with
multiple CPUs because these threads can be executed concurrently.
Multithreading in Python
For performing multithreading in Python threading module is
used.The threadingmodule provides several functions/methods to implement
multithreading easily in python.
Before we start using the threading module, we would like to first introduce
you to a module named time, which provides a time(), ctime() etc functions
which we will be using very often to get the current system time and another
crucial function sleep() which is used to suspend the execution of the current
thread for a given number of seconds.
For example,
Now let's see how we can use the threading module to start multiple threads.
Code Example:
threading.active_count() Function
import threading
def thread1(i):
time.sleep(3)
def thread2(i):
time.sleep(3)
if __name__ == '__main__':
t1 = threading.Thread(target=thread1, args=(10,))
t2 = threading.Thread(target=thread2, args=(12,))
t1.start()
t2.start()
t1.join()
t2.join()
No. of active threads: 3
Try running the this code in the terminal above. You will see the number of
thread count to be 3, because we have created 2 threads and there in the main
thread in which the complete execution is taking place.
threading.current_thread()
This function will return the current Thread object, corresponding to the
caller's thread of control(which is in the control of caller currently). If the
caller's thread of control was not created through the threading module(for
example the main thread), then a dummy thread object with limited
functionality is returned.
import time
import threading
def thread1(i):
time.sleep(3)
def thread2(i):
time.sleep(3)
if __name__ == '__main__':
t1 = threading.Thread(target=thread1, args=(10,))
t2 = threading.Thread(target=thread2, args=(12,))
t1.start()
t2.start()
t1.join()
t2.join()
threading.get_ident()
This function returns the thread identifier of the current thread. This is a
nonzero integer value. If we started the thread, then this method will return its
identifier, otherwise, it will return None.
We can use this method to index a dictionary of thread-specific data. Thread
identifiers may be recycled when a thread exits(stops) and another thread is
created.
threading.get_ident()
140578859194112
threading.enumerate()
This method returns a list of all Thread objects currently alive. The list
includes daemonic threads(when the program quits, all the daemon threads
associated with it are killed automatically), dummy thread objects created by
the current thread, and the main thread.
Terminated threads and threads that have not yet been started are not present
in this list.
threading.enumerate()
threading.main_thread()
This method returns the main Thread object. In normal conditions,
the main thread is the thread from which the python interpreter was started.
threading.main_thread()
threading.settrace(fun)
This method is used to set a trace function/hook for all the threads started
using the threadingmodule. The trace funtion is passed
to sys.settrace() method for each thread, which is attache to the thread before
the run() method is called.
The callback function which we want to act as the trace function will receive
three arguments, frame (the stack frame from the code being run), event (a
string naming the type of notification), and arg (an event-specific value)
def hello(frame, event, arg):
threading.settrace(hello)
Try updating the code in the terminal at the top. Put the hello function outside
the main method and the statement threading.settrace(hello) just above the
statement where we call the start method for the threads.
threading.setprofile(fun)
This method is used to set a profile function for all threads started from the
threading module. Just like the trace funtion, profile function is also passed
to sys.setprofile() method for each thread, which is attache to the thread
before the run() method is called.
threading.setprofile(hello)
threading.stack_size([size])
This method returns the thread stack size utilised when creating new threads.
The size argument is optional and can be used to set the stack size to be used
for creating subsequent threads, and must be 0 or a positive integer
(D=default value is 0).
If changing the thread stack size is unsupported, a RuntimeError is raised.
If the specified stack size is invalid, a ValueError is raised.
Currently 32 KiB is the minimum stack size which is supported to provide
enough stack space for the interpreter.
threading.TIMEOUT_MAX
Apart from the above specified functions, the threading module also defines a
constant. If you specify a timeout value which is greater than the value of
the TIMEOUT_MAX constant, you will get an OverflowError.
We will explain the arguments of the Thread class constructor in the section
below.
How Thread works?
Once we initialize a thread using the Thread class constructor, we must call
its start() method to start the thread.
When the thread starts, the thread is considered alive and active. When
its run() method terminates, either normally, or due to an unhandled
exception then the thread stops being alive or active. The isAlive() method
tests whether the thread is alive or not at any given point of time.
Other threads can call a thread's join() method to join any thread. This blocks
the calling thread until the thread whose join() method is called is terminated.
For example, in the code above, from the main thread, we
call t1.join() and t2.join(), hence the main thread will wait for the
threads t1 and t2 to terminate and then end.
Every thread has a name associated with it. The name can be passed to the
constructor, or we can set or retrieve name by
using setname() and getname() methods respectively.
A flag daemon thread can be associated with any thread. The significance of
this flag is that the entire python program exits when only daemon threads are
left. The flag can be set or retrieved by using setDaemon() method
and getDaemon() method respectively.
The main thread object corresponds to the initial thread of control in the
python program. It is not a daemon thread.
Now that we have seen a basic threading program with threads running, it's
time to understand the code along with exploring all the important methods
provided by the Thread class.
start() method
This method is used to start the thread's activity. When we call this method,
internally the run() method is invoked which executes the target function or
the callable object.
run() method
Method representing the thread's activity.
You may override this method in a subclass extending the Thread class of the
threading module. The standard run() method invokes the callable object
passed to the object's constructor as the target argument with sequential and
keyword arguments taken from the args and kwargs arguments, respectively.
Here we have a simple example, where we have created a subclass in which
we will override the run() method.
join([timeout]) method
When we call this method for any thread, it blocks the calling thread until the
thread whose join() method is called terminates, either normally or through
an unhandled exception.
If you want to provide the timeout argument, it should be a floating point
number.
getName() method
This method is used to return the thread's name.
setName(name) method
Used for setting the thread's name. The name is a string used for
identification purposes only.
isAlive() method
This method returns whether the thread is alive or not. A thread is alive from
the moment the start() method returns until its run() method terminates.
isDaemon() method
This method is used to get the value of the thread's daemon flag.
setDaemon(daemonic) method
This method is used to set the thread's daemon flag to the Boolean
value daemonic. This must be called before start() method is called.
The entire Python program exits when no active non-daemon threads are left.
are_you_coming = threading.Event()
When we initialize an event object like this the internal flag is set to false by
default.
isSet() method
This method returns true if and only if the internal flag is true.
import threading
are_you_coming = threading.Event()
print(are_you_coming.isSet())
set() method
When this method is called for any event object then
the internal flag is set to true. And as soon
as set() methos is called for any event all threads
waiting for it are awakened.
clear() method
This method resets the internal flag to false.
Subsequently, threads calling wait() on the event for
which clear() is called, it will block until the internal
flag is not true again.
wait([timeout]) method
When we have to make any thread wait for an event, we can do so by calling
this method on that event which has the internal flag set to false, doing so
blocks the thread until the internal flag is true for the event.
If the internal flag is true on entry, then the thread will never get blocked.
Otherwise, it is blocked until another thread calls set() to set the flag to true,
or until the optional timeout occurs. The timeout argument specifies a timeout
for the operation in seconds.
This way we can create a timer object that will run the function with
arguments args and keyword arguments kwargs, after interval seconds have
passed.
start() method
This method is used to start the execution of the timer object. When we call
this method, then the timer object starts its timer.
cancel() method
This method is used to stop the timer and cancel the execution of the timer
object's action. This will only work if the timer has not performed its action
yet.
def task():
if __name__=='__main__':
t = threading.Timer(10, task)
The above program is a simple program, now let's use the cancel method to
cancel the timer object task's execution.
In the program above, first comment the code on line number 13 and 14 and
run the program, then uncomment those lines and see the cancel() method in
action.
acquire(*args) method
This method is used to acquire the lock. This method calls the
corresponding acquire() method on the underlying lock present in the
condition object; The return value is whatever that method returns.
release() method
this method is used to release the lock. This method calls the
corresponding release() method on the underlying lock present in the
condition object.
wait([timeout]) method
This method is used to block the thread and make it wait until some other
thread notifies it by calling the notify() or notifyAll() method on the same
condition object or until the timeout occurs.
This must only be called when the calling thread has acquired the lock.
When called, this method releases the lock and then blocks the thread until it
is awakened by a notify() or notifyAll() call for the same condition variable
from some other thread, or until the timeout occurs.
This method returns True if it is released because
of notify() or notifyAll() method else if timeout occurs this method will
return False boolean value.
notify() method
It wakes up any thread waiting on the corresponding condition. This must
only be called when the calling thread has acquired the lock. Also, calling
this method will wake only one waiting thread.
notifyAll() method
It wakes up all the threads waiting on this condition. This method acts
like notify() method, but wakes up all the waiting threads instead of one.
wait(timeout=None) method
When we want a set of threads to wait for each other, we initialise the barrier
object with the number of threads specified as parties parameter while barrier
object creation.
Then, the barrier will be released only when the same number of threads call
the wait() method of the barrier object.
If in a thread we provide the timeout value while calling the wait() method,
then that thread will get released from the barrier once the timeout time is
passed.
This method returns an integer value from 0 to parties-1. We can use this
value to identify which all threads have reached the waiting point of the
barrier and which all are still not there, for example:
i = barrier.wait()
if i == 0:
The return value of the wait() method can be used for carrying out some clean
up task for example if we have 3 threads doing some work and using a
temporary file to store data, once the threads are done, we can put a check on
the wait() method that when the last thread reaches its waiting point and the
barrier is to be released, before that delete the file.
i = barrier.wait()
if i == 2:
If the wait call times out, the barrier is put into the broken state.
The wait() method may raise a BrokenBarrierError if the barrier breaks or
resets as a thread waits.
reset() method
This function resets the barrier to its default, empty state. If there are threads
waiting for the barrier to get released will receive BrokenBarrierError.
abort() method
This method when called on a barrier puts it into the broken state. Once this
method is called by any of the waiting thread, the rest of the threads waiting
for the barrier to be released will receive BrokenBarrierError.
We may want to use this method in case of some deadlock situation to release
the waiting threads.
parties
This returns the number of threads we need to pass the barrier.
n_waiting
This returns the number of threads that currently are waiting for the barrier to
be released.
broken
This is a Boolean value that is True if the barrier is in the broken state.
Let us take a look at the example for the clear understanding of this method:
import logging
logging.basicConfig(level=logging.INFO)
Points to remember:
With log level, logger name and then the log message.
We can configure the logging module to print logs in a any format we want.
There are some standard formats which are universally used, and we can
configure the python logger to print logs in those formats.
There are some basic components of logs that are already a part of
the LogRecord and we can easily add them or remove them from the output
format.
Let's take a few examples to understand this.
If you want to include the process ID for the running python script along
with the log level and message then the code snippet for the same is given
below:
import logging
logging.basicConfig(format='%(process)d-%(levelname)s-%(message)s')
You can also add date and time info(timestamp) to your logs along with the
log message. The code snippet for the same is given below:
import logging
logging.basicConfig(format='%(asctime)s - %(message)s', level=logging.INFO)
2020-06-19 11:43:59,887 - This message is to indicate that Admin has just logged in
You can also change the format using the datefmt attribute, which uses
the same formatting language as the formatting functions in the datetime
module, such as time.strftime():
import logging
As you can see the log printed as the output of the above code has a date
format DD-MMM-YY along with time.
So, using the basicConfig(**kwargs) method, we can
configure the logs format as we want, like adding
timestamp to logs, adding process id to logs, printing
log level along with logs and log messages.
2. Python - Print Logs in a File
If you want to print python logs in a file rather than on the console then we
can do so using the basicConfig() method by
providing filename and filemode as parameter.
The format of the message can be specified by using format parameter
in basicConfig() method.
Let us take a basic example to print logs to a file rather than on the console.
The code snippet is given below:
import logging # first of all import the module
The above output shows how the message will look like but keep in mind it
will be written to a file named std.log instead of the console.
In the above code, the filemode is set to w, which means the log file is
opened in “write mode” each time basicConfig() is called, and after each run
of the program, it will rewrite the file.
The default configuration for filemode is a, that is append, which means that
logs will be appended to the log file and adding logs to the existing logs.
import logging
logger=logging.getLogger()
logger.setLevel(logging.DEBUG)
logger.warning("OOPS!!!Its a Warning")
You can change the format of logs, log level or any other attribute of
the LogRecord along with setting the filename to store logs in a file along
with the mode.
Example
import logging
In the above code, there is merging of variable data into the event
description message using the old, %s style of string formatting.
Also, the arguments that are passed to the method would be included
as variable data in the message.
As you can use any formatting style, the f-strings introduced in Python
3.6 are also an amazing way to format strings as with their help the
formatting becomes short and easy to read:
Here is a code example:
import logging
name = 'Thomas'
Object Description
Logger.info(msg) This function mainly helps to log a message with level INFO on this
logger.
This function mainly helps to log a message with level ERROR on this
Logger.error(msg)
logger.
This function is mainly used to set the threshold of the logger to lvl and
Logger.setLevel(lvl) that indicates all the messages below this level
will be ignored.
This function mainly helps to log a message with level ERROR on this
Logger.exception(msg)
logger.
Logger.log(lvl, msg) This function will Logs a message with integer level lvl on this logger.
This method is mainly used to apply the logger’s filter to the provided
Logger.filter(record) record and it will return True
if the record is to be processed. otherwise, it will return False.
This is mainly used to check if the logger has any handler configured
Logger.hasHandlers()
or not.
1. Logger class
The Logger is the class whose objects will be used in the application code
directly to call the functions.
2. LogRecord
The Logger class automatically create LogRecord objects which contain the
information related to the event being logged, or in simple words the log
message. It contains information like the name of the logger, the function,
the line number, the message, the process id, etc.
3. Handler
The Handlers are mainly used to send the LogRecord to the required output
destination, it can be console or a file
The class Handler is the base for
subclasses like StreamHandler, FileHandler, SMTPHandler, HTTPHandler,
and more. These subclasses also send the logging outputs to corresponding
destinations, like sys.stdout or a disk file.
4. Formatter
The Formatter is a class where you specify the format of the
output by specifying a string format that mainly lists out the attributes the
output should contain.
Several Logger objects
In Logging Module items with all caps are constant, the capitalize items
indicate classes, and the items which start with lowercase letters are methods.
Below we have a table consisting of several logger objects offered by the
Logging module:
Chapter 9: Python With MySQL
1. MySQL with Python
In this tutorial, we will learn What is MySQL, What are the prerequisites to
use MySQL in Python and from where you can download and install
MySQL and MySQL Python Connector.
First of all, let us see what are the Prerequisites needed to learn MySQL in
Python:
What is MySQL?
Let us first learn what MySQL is:
MySQL is an open-source, relational database management
system(RDBMS) that is based on Structured Query Language(SQL). One
important thing to note here is that MySQL is used to store data and it is not
used to create programs; Thus it is not a Programming Language.
Thus SQL can be used to program a MySQL Database.
The main advantage of MySQL is that it can run on any of the
Operating System.
It is one of the Popular Database.
If you want to practice the code examples in our upcoming tutorials then it is
mandatory for you to install MySQL on your Computer. It is freely available
you can download and install the MySQL database from its official website :
https://www.mysql.com/downloads/
After installing MySQL the next step is to install mysql connector for the
python. Now the question arises what is MySQL connector?
MySQL Connector
Basically Python needs a MySQL Driver which is used to access the
MySQL Database an in this case "MySQL Connector" is the Driver
MySQL Connector for Python is a connector that enables the Python
programs to access the MySQL database.
There are two ways to download the MySQL Connector:
1. The first way is you can download and install it from the link given:
https://dev.mysql.com/downloads/connector/python/
2. Another way is By Using PIP to install the "MySQL Connector" as
PIP is already installed in your Python environment. So you can
download and install just by navigating your command line to the
location of PIP and By writing:
python -m pip install mysql-connector-python
We Recommend you to use the second way that is by PIP; also it is your
choice.
Now, After this MySQL Driver that is "MySQL Connector" is
downloaded and installed successfully in your Computer. But in order to
confirm, we can Test the MySQL Connector.
import mysql.connector
mydb = mysql.connector.connect(
host="localhost",
user="yourusername",
password="yourpassword"
print(mydb)
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword"
)
## Now Let us create an instance of 'cursor' class which is used to execute the 'SQL' statements in
'Python'
cursor = db.cursor()
In the case, if the database named "studytonight" already exists then the
above code will raise an error.
If there is no error it means the database is created successfully.
Now let us see how we can check all the databases in our system.
We can list down all the databases in MySQL using Python and in that list we
can also check if any particular Database is available or not.
To check if any database already exists SHOW DATABASES SQL statement
is used.
Let us see how:
To list out all the databases in the system.Below is the code snippet for the
same:
#for our convenience we will import mysql.connector as mysql
host = "localhost",
user = "yourusername",
passwd = "yourpassword"
cursor = db.cursor()
cursor.execute("SHOW DATABASES")
for x in cursor:
print(x)
The output of the above code will list out all the databases in the system:
('admin_panel',)
('information_schema',)
('mysql',)
('performance_schema',)
('sakila',)
('studytonight',)
('sys',)
('world',)
('xyz',)
As you can see in the code above, we have used Python for loop to iterate
over the cursor object and print the names of all the databases found.
Similarly, we can compare the name of databases to check if any particular
database already exists, before creating it using Python if condition.
So in this tutorial we learned how to create a new database in MySQL using
Python and how to list down all the MySQL databases in Python and print
them in console.
db = mysql.connector.connect(
host = "localhost",
user = "yourusername",
password = "yourpassword",
database = "studytonight"
If the above code is executed without any errors then it means you have
successfully connected to the database named studytonight.
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database="studytonight"
cursor = db.cursor()
cursor.execute("CREATE TABLE students (name VARCHAR(255), rollno INTEGER(100), branch
VARCHAR(255), address VARCHAR(255))")
If this code executes without any error then it means the table has been
created successfully.
Now, If you want to check the existing tables in the database then you can
use the SHOW TABLES SQL statement.
Now as we have created a table in our database. Let us check the tables that
exist in our database. Use the code given below:
#for our convenience we will import mysql.connector as mysql
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database="studytonight"
cursor = db.cursor()
cursor.execute("SHOW TABLES")
print(table)
Using the above code with any column, we can make its value as auto
increment, which means the database will automatically add an incremented
value even if you do not insert any value for that column while inserting a
new row of data to your table.
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
If the above code runs without an error then it means you have successfully
created a table named "students" with the column rollno as primary key.
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
cursor.execute("DESC students")
print(cursor.fetchall())
[('name', 'varchar(255)', 'YES', '', None, ''), ('rollno', 'int', 'NO', 'PRI', None, 'auto_increment'), ('branch',
'varchar(255)', 'YES', '', None, ''), ('address', 'varchar(255)', 'YES', '', None, '')]
db = mysql.connect(
host = "localhost",
user = "root",
passwd = "himaniroot@99",
database = "studytonight"
cursor = db.cursor()
## We are going to add rollno field with primary key in table students
print(cursor.fetchall())
In the above code we have used the SQL Alter Query to add a new column to
an existing table. You can use the describe table query to see the new added
column.
[('name', 'varchar(255)', 'YES', '', None, ''), ('branch', 'varchar(255)', 'YES', '', None, ''), ('address',
'varchar(255)', 'YES', '', None, ''), ('rollno', 'int', 'NO', 'PRI', None, 'auto_increment')]
In this section, we will see the code example for inserting a single row of data
in a MySQL table:
We will be adding data to the students table we created in Python MySQL -
Create Table tutorial.
import mysql.connector as mysql
db = mysql.connect(
host = "localhost",
user = "root",
passwd = "himaniroot@99",
database = "studytonight"
cursor = db.cursor()
cursor.execute(query, values)
db.commit()
db = mysql.connect(
host = "localhost",
user = "root",
passwd = "himaniroot@99",
database = "studytonight"
cursor = db.cursor()
values = [
("Peter", "ME","Noida"),
cursor.executemany(query, values)
db.commit()
If all three rows were entered successfully, the output of the above code will
be:
3 records inserted
So in this tutorial we learned how we can insert data into MySQL table in
Python.
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
cursor.execute(query)
## fetching all records from the 'cursor' object
records = cursor.fetchall()
print(record)
Below we have a snapshot of the exact output when we run this python code:
In the next section we will learn how to retrieve data of certain columns from
a table.
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
cursor.execute(query)
names = cursor.fetchall()
print(name)
The above code will fetch the name column from the students table:
('Ramesh',)
('Peter',)
('Amy',)
('Michael',)
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
cursor.execute(query)
data = cursor.fetchall()
## Showing the data
print(pair)
The above code will fetch both name and branch column both from the
table students:
('Ramesh', 'CSE')
('Peter', 'ME')
('Amy', 'CE')
('Michael', 'CSE')
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
print(myresult)
Thus in the output the first row of the record will be fetched:
('Ramesh', 'CSE', '149 Indirapuram', 1)
The above syntax is used to update any specific row in the MySQL table.
And to specify which specific row of data we want to update, we use
the WHERE clause to provide the condition to be matched while looking for
the right row of data to update.
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
cursor.execute(query)
db.commit()
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
cursor.execute(query)
records = cursor.fetchall()
print(record)
Note: In the above syntax, WHERE clause is used to specify the condition to
pin point a particular record that you want to delete. If you don't specify the
condition, then all of the records of the table will be deleted.
db = mysql.connect(
host = "localhost",
user = "yoursername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
cursor.execute(query)
## final step to tell the database that we have changed the table data
db.commit()
If the above code runs without an error then it means that the row with rollno
= 4 is deleted successfully.
To check if it exists in the table or not you can use the code given below:
import mysql.connector as mysql
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
cursor.execute(query)
records = cursor.fetchall()
print(record)
As we can see in the output above, the row with rollno=4 is successfully
deleted.
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
)
cursor = db.cursor()
cursor.execute(sql)
If the above code will execute without any error it means table
named customers is deleted succesfully.
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
)
cursor = db.cursor()
cursor.execute(sql)
If the code executes without an error, then it means the customers table is
deleted if it existed.
Here is the snapshot of the actual output:
So in this tutorial we learned how to drop a MySQL table using Python. This
is useful when we have some application which creates some temporary
tables to store some data and then after the processing, deletes those tables.
9. Python MySQL
- WHERE Clause
We will learn how to filter rows from the fetched resultset, or update a
specific row, or delete a specific row from a MySQL table using
the WHERE clause to specify the condition to find the record/row of data on
which the operation is performed.
So, WHERE clause is nothing but a way to provide a condition(or multiple
conditions) to the SQL engine, which is applied on the query resultset to filter
out the required records of data.
FROM table_name
WHERE condition;
The above syntax is useful in fetching the records on the basis of some
conditions.
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
cursor.execute(query)
records = cursor.fetchall()
print(record)
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
cursor.execute(query)
records = cursor.fetchall()
print(record)
The above code will sort the names in ascending order and the output will be
as follows:
('Amy', 'CE', 'New Delhi', 3)
('Michael', 'CSE', 'London', 4)
('Peter', 'ME', 'Noida', 2)
('Ramesh', 'CSE', '149 Indirapuram', 1)
Now Let us sort the data in descending order by name column using
the DESC keyword with the ORDER BY clause. Let's see the code which is
given below:
import mysql.connector as mysql
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
cursor.execute(query)
records = cursor.fetchall()
## Showing the data
print(record)
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
myresult = cursor.fetchall()
for x in myresult:
print(x)
The output of the above code will be as follows:
('Ramesh', 'CSE', '149 Indirapuram', 1)
('Peter', 'ME', 'Noida', 2)
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
myresult = cursor.fetchall()
for x in myresult:
print(x)
In the above code LIMIT 2 means it will return the 2 rows and OFFSET
1 the resultset will start from the 1st row, means only row 2 is returned. Thus
the output is as follows:
('Peter', 'ME', 'Noida', 2)
('Navin', 'CE', 'New Delhi', 3)
The OFFSET keyword isused to specify the starting point. For example if a
query returnd 100 rows of data and we have specifed the OFFSET as 50, then
data starting from 51st row till the 100th row will be returned.
You can directly run the query to create the new table in MySQL command
line.
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
)
cursor = db.cursor()
cursor.execute(sql)
myresult = cursor.fetchall()
for x in myresult:
print(x)
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
cursor.execute(sql)
myresult = cursor.fetchall()
for x in myresult:
print(x)
In the above output, the status of rollno 5 and 6 is None because their result
is not mentioned in the results table. But as we have applied LEFT JOIN so
the query selects all the rows from the left table even if there is no match.
Here is the snapshot of the actual output:
db = mysql.connect(
host = "localhost",
user = "yourusername",
passwd = "yourpassword",
database = "studytonight"
cursor = db.cursor()
sql = "SELECT students.rollno, students.name,students.Branch,students.Address, results.status from
students RIGHT JOIN results ON students.rollno=results.rollno;"
cursor.execute(sql)
myresult = cursor.fetchall()
for x in myresult:
print(x)
And with this we have covered all the basics of Python MySQL. If you have
to develop an application in which you want to have a database with multipl
tables, in which you want to store data and retrieve data from the tables, then
we hope these tutorials help you.
Conclusion
This book is dedicated to the readers who take time to write me each day.
Every morning I’m greeted by various emails — some with requests, a few
with complaints, and then there are the very few that just say thank you. All
these emails encourage and challenge me as an author — to better both my
books and myself.
Thank you!