Data Structures Using Python (R20A0503)LECTURE NOTES B.TECHIIYEAR–I-SEM (R20)(2021-2022) MALLAREDDYCOLLEGEOFENGINEERING&TECHNOLOGY (AutonomousInstitution– UGC,Govt.ofIndia) Recognizedunder2(f)and12(B)ofUGCACT 1956 (AffiliatedtoJNTUH,Hyderabad,ApprovedbyAICTE-AccreditedbyNBA&NAAC –‘A’Grade-ISO9001:2015Certified) Maisammaguda,Dhulapally(PostVia.Hakimpet),Secunderabad–500100,TelanganaState,India 1 SYLLABUS MALLAREDDYCOLLEGEOFENGINEERING ANDTECHNOLOGY IIYearB.Tech CSE-I SEM L 3 T/P/DC -/ -/ -3 (R20A0503)DATA STRUCTURES USING PYTHON COURSE OBJECTIVES: This course will enable students to 1. Implement Object Oriented Programming concepts in Python. 2. Understand Lists, Dictionaries and Regular expressions in Python. 3. Understanding how searching and sorting is performed in Python. 4. Understanding how linear and non-linear data structures works. 5. To learn the fundamentals of writing Python scripts. UNIT – I Oops Concepts- class, object, constructors, types of variables, types of methods. Inheritance: single, multiple, multi-level, hierarchical, hybrid, Polymorphism: with functions and objects, with class methods, with inheritance,Abstraction: abstract classes. UNIT – II Data Structures – Definition,Linear Data Structures,Non-Linear Data Structures Python Specific Data Structures: List,Tuples, Set, Dictionaries, Comprehensions and its Types,Strings,slicing. UNIT -III Arrays - Overview, Types of Arrays, Operations on Arrays, Arrays vs List. Searching -Linear Search and Binary Search. Sorting - Bubble Sort, Selection Sort, Insertion Sort, Merge Sort, Quick Sort. UNIT -IV Linked Lists – Implementation ofSingly Linked Lists, Doubly Linked Lists, Circular Linked Lists. Stacks - Overview of Stack, Implementation of Stack (List & Linked list), Applications of Stack Queues:Overview of Queue, Implementation of Queue(List & Linked list), Applications of Queues, Priority Queues. UNIT -V Graphs -Introduction, Directed vs Undirected Graphs, Weighted vs Unweighted Graphs, Representations, Breadth First Search, Depth First Search. Trees - Overview of Trees, Tree Terminology, Binary Trees: Introduction, Implementation, Applications. Tree Traversals, Binary Search Trees: Introduction, Implementation, AVL Trees: Introduction, Rotations, Implementation. TEXTBOOKS: 1. Data structures and algorithms in python by Michael T. Goodrich 2. Data Structures and Algorithmic Thinking with Python by NarasimhaKarumanchi 2 REFERENCE BOOKS: 1. Hands-On Data Structures and Algorithms with Python: Write complex and powerful code using the latest features of Python 3.7, 2nd Edition by Dr. Basant Agarwal, Benjamin Baka. 2. Data Structures and Algorithms with Python by Kent D. Lee and Steve Hubbard. 3. Problem Solving with Algorithms and Data Structures Using Python by Bradley N Miller and David L. Ranum. 4. Core Python Programming -Second Edition,R. Nageswara Rao, Dreamtech Press COURSE OUTCOMES: The students should be able to: 1. Examine Python syntax and semantics and apply Python flow control and functions. 2. Create, run and manipulate Python Programs using core data structures like Lists, 3. Apply Dictionaries and use Regular Expressions. 4. Interpret the concepts of Object-Oriented Programming as used in Python. 5. Master object-oriented programming to create an entire python project using objects and classes 3 INDEX UNIT TOPIC PAGENO Class ,Objects, Constructors I 6-9 Types of Variables 10-12 Types of Methods 12-15 Inheritance & Types 16-21 Polymorphism 22-26 Abstract Classes 27-28 Introduction to Data Structures, Types 29-31 Python Specific Data Structures 31-43 Sequences 44-49 Comprehensions-List,Tuples,Set,Dictonary 50-52 String Slicing 53-55 Arrays&operations on Arrays 56-61 Searching: Linear Search, Binary search 62-68 Sorting:Bubble,Selction,Insertion,Merge,Quick Sort Techniques 68-77 Linked List: Single ,Double, Circular Linked List 78-85 Stacks:Operations,Implementation using List ,Linked List 86-92 Queues:Operations,Implementation using List ,Linked List Priority Queues 93-105 II III IV i 4 106-111 V Introduction to Graphs, Types of Graphs Breadth First Search Depth First Search Introduction to Trees, Types of Trees Binary Search Tree:Operations,Implementation AVL Tree : Operations, Implementation 5 112-117 117-119 120-122 123-135 144-151 152-159 UNIT – I Oops Concepts- class, object, constructors, types of variables, types of methods. Inheritance: single, multiple, multi-level, hierarchical, hybrid, Polymorphism: with functions and objects, with class methods, with inheritance,Abstraction: abstract classes. OOPs in Python OOPs in Python is a programming approach that focuses on using objects and classes as same as other general programming languages. The objects can be any real-world entities. Python allows developers to develop applications using the OOPs approach with the major focus on code reusability. 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() Here, obj is an object of class Parrot. Suppose we have details of parrots. Now, we are going to show how to build the class and objects of parrots. 6 Example: class Parrot: # class attribute species = "bird" # instance attribute def __init__(self, name, age): self.name = name self.age = age # instantiate the Parrot class blu = Parrot("Blu", 10) woo = Parrot("Woo", 15) # access the class attributes print("Blu is a {}".format(blu.__class__.species)) print("Woo is also a {}".format(woo.__class__.species)) # access the instance attributes print("{} is {} years old".format( blu.name, blu.age)) print("{} is {} years old".format( woo.name, woo.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. 7 constructor The constructor is a method that is called when an object is created. This method is defined in the class and can be used to initialize basic variables. If you create four objects, the class constructor is called four times. Every class has a constructor, but its not required to explicitly define it. Example: Each time an object is created a method is called. That methods is named the constructor. The constructor is created with the function init. As parameter we write the self keyword, which refers to itself (the object). The process visually is: Inside the constructor we initialize two variables: legs and arms. Sometimes variables are named properties in the context of object oriented programming. We create one object (bob) and just by creating it, its variables are initialized. classHuman: def__init__(self): self.legs = 2 self.arms = 2 bob = Human() print(bob.legs) The newly created object now has the variables set, without you having to define them manually. You could create tens or hundreds of objects without having to set the values each time. python __init__ The function init(self) builds your object. Its not just variables you can set here, you can call class methods too. Everything you need to initialize the object(s). Lets say you have a class Plane, which upon creation should start flying. There are many steps involved in taking off: accelerating, changing flaps, closing the wheels and so on. 8 The default actions can be defined in methods. These methods can be called in the constructor. classPlane: def__init__(self): self.wings = 2 # fly self.drive() self.flaps() self.wheels() defdrive(self): print('Accelerating') defflaps(self): print('Changing flaps') defwheels(self): print('Closing wheels') ba = Plane() To summarize: A constructor is called if you create an object. In the constructor you can set variables and call methods. Default value The constructor of a class is unique: initiating objects from different classes will call different constructors. Default values of newly created objects can be set in the constructor. The example belwo shows two classes with constructors. Then two objects are created but different constructors are called. classBug: def__init__(self): self.wings = 4 classHuman: def__init__(self): self.legs = 2 self.arms = 2 bob = Human() tom = Bug() print(tom.wings) print(bob.arms) 9 But creating multiple objects from one class, will call the same constructor. Python Variable Types: Local & Global There are two types of variables in Python, Global variable and Local variable. When you want to use the same variable for rest of your program or module you declare it as a global variable, while if you want to use the variable in a specific function or method, you use a local variable while Python variable declaration. Let's understand this Python variable types with the difference between local and global variables in the below program. 1. Let us define variable in Python where the variable "f" is global in scope and is assigned value 101 which is printed in output 2. Variable f is again declared in function and assumes local scope. It is assigned value "I am learning Python." which is printed out as an output. This Python declare variable is different from the global variable "f" defined earlier 3. Once the function call is over, the local variable f is destroyed. At line 12, when we again, print the value of "f" is it displays the value of global variable f=101 Python 2 Example # Declare a variable and initialize it f = 101 print f # Global vs. local variables in functions def someFunction(): # global f f = 'I am learning Python' print f 10 someFunction() print f Python 3 Example # Declare a variable and initialize it f = 101 print(f) # Global vs. local variables in functions def someFunction(): # global f f = 'I am learning Python' print(f) someFunction() print(f) While Python variable declaration using the keyword global, you can reference the global variable inside a function. 1. Variable "f" is global in scope and is assigned value 101 which is printed in output 2. Variable f is declared using the keyword global. This is NOT a local variable, but the same global variable declared earlier. Hence when we print its value, the output is 101 3. We changed the value of "f" inside the function. Once the function call is over, the changed value of the variable "f" persists. At line 12, when we again, print the value of "f" is it displays the value "changing global variable" Python 2 Example f = 101; print f # Global vs.local variables in functions def someFunction(): 11 global f print f f = "changing global variable" someFunction() print f Python 3 Example f = 101; print(f) # Global vs.local variables in functions def someFunction(): global f print(f) f = "changing global variable" someFunction() print(f) Types of methods: Generally, there are three types of methods in Python: 1. Instance Methods. 2. Class Methods 3. Static Methods Before moving on with the topic, we have to know some key concepts. Class Variable: A class variable is nothing but a variable that is defined outside the constructor. A class variable is also called as a static variable. Accessor(Getters): If you want to fetch the value from an instance variable we call them accessors. Mutator(Setters): If you want to modify the value we call them mutators. 1. Instance Method This is a very basic and easy method that we use regularly when we create classes in python. If we want to print an instance variable or instance method we must create an object of that required class. If we are using self as a function parameter or in front of a variable, that is nothing but the calling instance itself. As we are working with instance variables we use self keyword. Note: Instance variables are used with instance methods. 12 Look at the code below # Instance Method Example in Python classStudent: def__init__(self, a, b): self.a = a self.b = b defavg(self): return(self.a + self.b)/2 s1 = Student(10,20) print( s1.avg()) Copy Output: 15.0 In the above program, a and b are instance variables and these get initialized when we create an object for the Student class. If we want to call avg() function which is an instance method, we must create an object for the class. If we clearly look at the program, the self keyword is used so that we can easily say that those are instance variables and methods. 2. Class Method classsmethod() function returns a class method as output for the given function. Here is the syntax for it: classmethod(function) The classmethod() method takes only a function as an input parameter and converts that into a class method. 13 There are two ways to create class methods in python: 1. Using classmethod(function) 2. Using @classmethod annotation A class method can be called either using the class (such as C.f()) or using an instance (such as C().f()). The instance is ignored except for its class. If a class method is called from a derived class, the derived class object is passed as the implied first argument. As we are working with ClassMethod we use the cls keyword. Class variables are used with class methods. Look at the code below. # Class Method Implementation in python classStudent: name ='Student' def__init__(self, a, b): self.a = a self.b = b @classmethod definfo(cls): return cls.name print(Student.info()) Copy Output: Student In the above example, name is a class variable. If we want to create a class method we must use @classmethod decorator and cls as a parameter for that function. 14 3. Static Method A static method can be called without an object for that class, using the class name directly. If you want to do something extra with a class we use static methods. For example, If you want to print factorial of a number then we don't need to use class variables or instance variables to print the factorial of a number. We just simply pass a number to the static method that we have created and it returns the factorial. Look at the below code # Static Method Implementation in python classStudent: name ='Student' def__init__(self, a, b): self.a = a self.b = b @staticmethod definfo(): return"This is a student class" print(Student.info()) Copy Output This a student class 15 Types of inheritances: The inheritance is a very useful and powerful concept of object-oriented programming. Using the inheritance concept, we can use the existing features of one class in another class. The inheritance is the process of acquiring the properties of one class to another class. In inheritance, we use the terms like parent class, child class, base class, derived class, superclass, and subclass. The Parent class is the class which provides features to another class. The parent class is also known as Base class or Superclass. The Child class is the class which receives features from another class. The child class is also known as the Derived Class or Subclass. In the inheritance, the child class acquires the features from its parent class. But the parent class never acquires the features from its child class. There are five types of inheritances, and they are as follows. • • • • • Simple Inheritance (or) Single Inheritance Multiple Inheritance Multi-Level Inheritance Hierarchical Inheritance Hybrid Inheritance The following picture illustrates how various inheritances are implemented. 16 Creating a Child Class In Python, we use the following general structure to create a child class from a parent class. Syntax classChildClassName(ParentClassName): ChildClass implementation . . Let's look at individual inheritance type with an example. Simple Inheritance (or) Single Inheritance In this type of inheritance, one child class derives from one parent class. Look at the following example code. Example classParentClass: deffeature_1(self): print('feature_1 from ParentClass is running...') deffeature_2(self): print('feature_2 from ParentClass is running...') classChildClass(ParentClass): deffeature_3(self): print('feature_3 from ChildClass is running...') obj = ChildClass() obj.feature_1() obj.feature_2() 17 obj.feature_3() When we run the above example code, it produces the following output. Multiple Inheritance In this type of inheritance, one child class derives from two or more parent classes. Look at the following example code. Example classParentClass_1: deffeature_1(self): print('feature_1 from ParentClass_1 is running...') classParentClass_2: deffeature_2(self): print('feature_2 from ParentClass_2 is running...') 18 classChildClass(ParentClass_1, ParentClass_2): deffeature_3(self): print('feature_3 from ChildClass is running...') obj = ChildClass() obj.feature_1() obj.feature_2() obj.feature_3() When we run the above example code, it produces the following output. Multi-Level Inheritance In this type of inheritance, the child class derives from a class which already derived from another class. Look at the following example code. Example classParentClass: 19 deffeature_1(self): print('feature_1 from ParentClass is running...') classChildClass_1(ParentClass): deffeature_2(self): print('feature_2 from ChildClass_1 is running...') classChildClass_2(ChildClass_1): deffeature_3(self): print('feature_3 from ChildClass_2 is running...') obj = ChildClass_2() obj.feature_1() obj.feature_2() obj.feature_3() When we run the above example code, it produces the following output. 20 Hierarchical Inheritance In this type of inheritance, two or more child classes derive from one parent class. Look at the following example code. Example classParentClass_1: deffeature_1(self): print('feature_1 from ParentClass_1 is running...') classParentClass_2: deffeature_2(self): print('feature_2 from ParentClass_2 is running...') classChildClass(ParentClass_1, ParentClass_2): deffeature_3(self): print('feature_3 from ChildClass is running...') obj = ChildClass() obj.feature_1() obj.feature_2() obj.feature_3() When we run the above example code, it produces the following output. 21 Hybrid Inheritance The hybrid inheritance is the combination of more than one type of inheritance. We may use any combination as a single with multiple inheritances, multi-level with multiple inheritances, etc., Polymorphism: Polymorphism is a concept of object oriented programming, which means multiple forms or more than one form. Polymorphism enables using a single interface with input of different datatypes, different class or may be for different number of inputs. In python as everything is an object hence by default a function can take anything as an argument but the execution of the function might fail as every function has some logic that it follows. For example, len("hello")# returns 5 as result len([1,2,3,4,45,345,23,42])# returns 8 as result In this case the function len is polymorphic as it is taking string as input in the first case and is taking list as input in the second case. 22 In python, polymorphism is a way of making a function accept objects of different classes if they behave similarly. Method overriding is a type of polymorphism in which a child class which is extending the parent class can provide different definition to any function defined in the parent class as per its own requirements. Method Overloading Method overriding or function overloading is a type of polymorphism in which we can define a number of methods with the same name but with a different number of parameters as well as parameters can be of different types. These methods can perform a similar or different function. Python doesn't support method overloading on the basis of different number of parameters in functions. Defining Polymorphic Classes Imagine a situation in which we have a different class for shapes like Square, Triangle etc which serves as a resource to calculate the area of that shape. Each shape has a different number of dimensions which are used to calculate the area of the respective shape. Now one approach is to define different functions with different names to calculate the area of the given shapes. The program depicting this approach is shown below: classSquare: side =5 defcalculate_area_sq(self): return self.side * self.side classTriangle: base =5 height =4 defcalculate_area_tri(self): return0.5* self.base * self.height 23 sq = Square() tri = Triangle() print("Area of square: ", sq.calculate_area_sq()) print("Area of triangle: ", tri.calculate_area_tri()) Area of square: 25 Area of triangle: 10.0 The problem with this approach is that the developer has to remember the name of each function separately. In a much larger program, it is very difficult to memorize the name of the functions for every small operation. Here comes the role of method overloading. Now let's change the name of functions to calculate the area and give them both same name calculate_area() while keeping the function separately in both the classes with different definitions. In this case the type of object will help in resolving the call to the function. The program below shows the implementation of this type of polymorphism with class methods: classSquare: side =5 defcalculate_area(self): return self.side * self.side classTriangle: base =5 height =4 defcalculate_area(self): return0.5* self.base * self.height sq = Square() 24 tri = Triangle() print("Area of square: ", sq.calculate_area()) print("Area of triangle: ", tri.calculate_area()) Area of square: 25 Area of triangle: 10.0 As you can see in the implementation of both the classes i.e. Square as well as Triangle has the function with same name calculate_area(), but due to different objects its call get resolved correctly, that is when the function is called using the object sq then the function of class Square is called and when it is called using the object tri then the function of class Triangle is called. Polymorphism with Class Methods What we saw in the example above is again obvious behaviour. Let's use a loop which iterates over a tuple of objects of various shapes and call the area function to calculate area for each shape object. sq = Square() tri = Triangle() for(obj in(sq, tri)): obj.calculate_area() Now this is a better example of polymorphism because now we are treating objects of different classes as an object on which same function gets called. Here python doesn't care about the type of object which is calling the function hence making the class method polymorphic in nature. Polymorphism with Functions Just like we used a loop in the above example, we can also create a function which takes an object of some shape class as input and then calls the function to calculate area for it. For example, find_area_of_shape(obj): obj.calculate_area() 25 sq = Square() tri = Triangle() # calling the method with different objects find_area_of_shape(sq) find_area_of_shape(tri) In the example above we have used the same function find_area_of_shape to calculate area of two different shape classes. The same function takes different class objects as arguments and executes perfectly to return the result. This is polymorphism. Polymorphism with Inheritance Polymorphism in python defines methods in the child class that have the same name as the methods in the parent class. In inheritance, the child class inherits the methods from the parent class. Also, it is possible to modify a method in a child class that it has inherited from the parent class. This is mostly used in cases where the method inherited from the parent class doesn’t fit the child class. This process of re-implementing a method in the child class is known as Method Overriding. Here is an example that shows polymorphism with inheritance: 26 Output: There are different types of birds Most of the birds can fly but some cannot There are different types of bird Parrots can fly There are many types of birds Penguins do not fly These are different ways to define polymorphism in Python. With this, we have come to the end of our article. I hope you understood what is polymorphism and how it is used in Python. Abstraction Abstraction is one of the most important features of object-oriented programming. It is used to hide the background details or any unnecessary implementation. Pre-defined functions are similar to data abstraction. For example, when you use a washing machine for laundry purposes. What you do is you put your laundry and detergent inside the machine and wait for the machine to perform its task. How does it perform it? What mechanism does it use? A user is not required to know the engineering behind its work. This process is typically known as data abstraction, when all the unnecessary information is kept hidden from the users. Code In Python, we can achieve abstraction by incorporating abstract classes and methods. Any class that contains abstract method(s) is called an abstract class. Abstract methods do not include any implementations – they are always defined and implemented as part of the methods of the sub-classes inherited from the abstract class. Look at the sample syntax below for an abstract class: from abc import ABC // abc is a library from where a class ABC is being imported. However, a separate class can also be created. The importing from the library has nothing to do with abstraction. Class type_shape(ABC): The class type_shape is inherited from the ABC class. Let’s define an abstract method area inside the class type_shape: from abc import ABC class type_shape(ABC): // abstract method area def area(self): pass The implementation of an abstract class is done in the sub-classes, which will inherit the class type_shape. We have defined four classes that inherit the abstract class type_shape in the code below 27 Example: from abc import ABC class type_shape(ABC): def area(self): #abstract method pass class Rectangle(type_shape): length = 6 breadth = 4 def area(self): return self.length * self.breadth class Circle(type_shape): radius = 7 def area(self): return 3.14 * self.radius * self.radius class Square(type_shape): length = 4 def area(self): return self.length*self.length class triangle: length = 5 width = 4 def area(self): return 0.5 * self.length * self.width r = Rectangle() # object created for the class 'Rectangle' c = Circle() # object created for the class 'Circle' s = Square() # object created for the class 'Square' t = triangle() # object created for the class 'triangle' print("Area of a rectangle:", r.area()) # call to 'area' method defined inside the class. print("Area of a circle:", c.area()) # call to 'area' method defined inside the class. print("Area of a square:", s.area()) # call to 'area' method defined inside the class. print("Area of a triangle:", t.area()) # call to 'area' method defined inside the class. Output 1.14s Area of a rectangle: 24 Area of a circle: 153.86 Area of a square: 16 Area of a triangle: 10.0 28 UNIT – II Data Structures – Definition,Linear Data Structures,Non-Linear Data Structures,Python Specific Data Structures, List,Tuples, Set, Dictionaries, Comprehensions and its Types,Strings,slicing. Data Structure Organizing, managing and storing data is important as it enables easier access and efficient modifications. Data Structures allows you to organize your data in such a way that enables you to store collections of data, relate them and perform operations on them accordingly. A data structure is classified into two categories: o Linear data structure o Non-linear data structure Now let's have a brief look at both these data structures. What is the Linear data structure? A linear data structure is a structure in which the elements are stored sequentially, and the elements are connected to the previous and the next element. As the elements are stored sequentially, so they can be traversed or accessed in a single run. The implementation of linear data structures is easier as the elements are sequentially organized in memory. The data elements in an array are traversed one after another and can access only one element at a time. The types of linear data structures are Array, Queue, Stack, Linked List. Let's discuss each linear data structure in detail. o Array: An array consists of data elements of a same data type. For example, if we want to store the roll numbers of 10 students, so instead of creating 10 integer type variables, we will create an array having size 10. Therefore, we can say that an array saves a lot of memory and reduces the length of the code. o Stack: It is linear data structure that uses the LIFO (Last In-First Out) rule in which the data added last will be removed first. The addition of data element in a stack is known as a push operation, and the deletion of data element form the list is known as pop operation. o Queue: It is a data structure that uses the FIFO rule (First In-First Out). In this rule, the element which is added first will be removed first. There are two terms used in the queue front end and rear The insertion operation performed at the back end is known ad enqueue, and the deletion operation performed at the front end is known as dequeue. o Linked list: It is a collection of nodes that are made up of two parts, i.e., data element and reference to the next node in the sequence. 29 What is a Non-linear data structure? A non-linear data structure is also another type of data structure in which the data elements are not arranged in a contiguous manner. As the arrangement is nonsequential, so the data elements cannot be traversed or accessed in a single run. In the case of linear data structure, element is connected to two elements (previous and the next element), whereas, in the non-linear data structure, an element can be connected to more than two elements. Trees and Graphs are the types of non-linear data structure. Let's discuss both the data structures in detail. o Tree It is a non-linear data structure that consists of various linked nodes. It has a hierarchical tree structure that forms a parent-child relationship. The diagrammatic representation of a tree data structure is shown below: For example, the posts of employees are arranged in a tree data structure like managers, officers, clerk. In the above figure, A represents a manager, B and C represent the officers, and other nodes represent the clerks. o Graph A graph is a non-linear data structure that has a finite number of vertices and edges, and these edges are used to connect the vertices. The vertices are used to store the data elements, while the edges represent the relationship between the vertices. A graph is used in various real-world problems like telephone networks, circuit networks, social networks like LinkedIn, Facebook. In the case of facebook, a single user can be considered as a node, and the connection of a user with others is known as edges. 30 Python Specific Data Structures: List: • It is a general purpose most widely used in data structures • List is a collection which is ordered and changeable and allows duplicate members. (Grow and shrink as needed, sequence type, sortable). • To use a list, you must declare it first. Do this using square brackets and separate values with commas. • We can construct / create list in many ways. Ex: >>> list1=[1,2,3,'A','B',7,8,[10,11]] >>> print(list1) [1, 2, 3, 'A', 'B', 7, 8, [10, 11]] --------------------->>> x=list() >>> x [] ------------------------->>> tuple1=(1,2,3,4) >>> x=list(tuple1) >>> x [1, 2, 3, 4] The list data type has some more methods. Here are all of the methods of list objects: List Operations: • Del() • Append() • Extend() • Insert() • Pop() • Remove() • Reverse() • Sort() Delete: Delete a list or an item from a list >>> x=[5,3,8,6] >>> del(x[1]) #deletes the index position 1 in a list >>> x 31 [5, 8, 6] ----------->>> del(x) >>> x # complete list gets deleted Append: Append an item to a list >>> x=[1,5,8,4] >>> x.append(10) >>> x [1, 5, 8, 4, 10] Extend: Append a sequence to a list. >>> x=[1,2,3,4] >>> y=[3,6,9,1] >>> x.extend(y) >>> x [1, 2, 3, 4, 3, 6, 9, 1] Insert: To add an item at the specified index, use the insert () method: >>> x=[1,2,4,6,7] >>> x.insert(2,10) #insert(index no, item to be inserted) >>> x [1, 2, 10, 4, 6, 7] ------------------------>>> x.insert(4,['a',11]) >>> x [1, 2, 10, 4, ['a', 11], 6, 7] Pop: The pop() method removes the specified index, (or the last item if index is not specified) or simply pops the last item of list and returns the item. >>> x=[1, 2, 10, 4, 6, 7] >>> x.pop() 7 >>> x [1, 2, 10, 4, 6] ---------------------------------->>> x=[1, 2, 10, 4, 6] >>> x.pop(2) 10 >>> x [1, 2, 4, 6] Remove: The remove() method removes the specified item from a given list. >>> x=[1,33,2,10,4,6] >>> x.remove(33) >>> x [1, 2, 10, 4, 6] >>> x.remove(4) >>> x [1, 2, 10, 6] Reverse: Reverse the order of a given list. >>> x=[1,2,3,4,5,6,7] >>> x.reverse() >>> x 32 [7, 6, 5, 4, 3, 2, 1] Sort: Sorts the elements in ascending order >>> x=[7, 6, 5, 4, 3, 2, 1] >>> x.sort() >>> x [1, 2, 3, 4, 5, 6, 7] ---------------------->>> x=[10,1,5,3,8,7] >>> x.sort() >>> x [1, 3, 5, 7, 8, 10] Slicing: Slice out substrings, sub lists, sub Tuples using index. [Start: stop: steps] • Slicing will start from index and will go up to stop in step of steps. • Default value of start is 0, • Stop is last index of list • And for step default is 1 Example: >>> x='computer' >>> x[1:4] 'omp' >>> x[1:6:2] 'opt' >>> x[3:] 'puter' >>> x[:5] 'compu' >>> x[-1] 'r' >>> x[-3:] 'ter' >>> x[:-2] 'comput' >>> x[::-2] 33 'rtpo' >>> x[::-1] 'retupmoc' List: >>> list1=range(1,6) >>> list1 range(1, 6) >>> print(list1) range(1, 6) >>> list1=[1,2,3,4,5,6,7,8,9,10] >>> list1[1:] [2, 3, 4, 5, 6, 7, 8, 9, 10] >>> list1[:1] [1] >>> list1[2:5] [3, 4, 5] >>> list1[:6] [1, 2, 3, 4, 5, 6] >>> list1[1:2:4] [2] >>> list1[1:8:2] [2, 4, 6, 8] Tuple: >>> list1=(11,12,13,14) >>> list1[:2] (11, 12) To create a slice: >>> print(slice(3)) slice(None, 3, None) >>> print(slice(2)) 34 slice(None, 2, None) >>> print(slice(1,6,4)) slice(1, 6, 4) To get substring from a given string using slice object: >>> pystr='python' >>> x=slice(3) >>> print(pystr[x]) Pyt Using –ve index: >>> pystr='python' >>> x=slice(1,-3,1) >>> print(pystr[x]) >>> yt To get sublist and sub-tuple from a given list and tuple respectively: >>> list1=['m','r','c','e','t'] >>> tup1=('c','o','l','l','e','g','e') >>> x=slice(1,4,1) >>> print(tup1[x]) ('o', 'l', 'l') >>> print(list1[x]) ['r', 'c', 'e'] >>> x=slice(1,5,2) >>> print(list1[x]) ['r', 'e'] >>> print(tup1[x]) ('o', 'l') >>> x=slice(-1,-4,-1) #negative index >>> print(list1[x]) ['t', 'e', 'c'] >>> x=slice(-1,-4,-1) #negative index 35 >>> print(tup1[x]) ('e', 'g', 'e') >>> print(list1[0:3]) #extending indexing syntax ['m', 'r', 'c'] Tuples: A tuple is a collection which is ordered and unchangeable. In Python tuples are written with round brackets. • Supports all operations for sequences. • Immutable, but member objects may be mutable. • If the contents of a list shouldn’t change, use a tuple to prevent items from accidently being added, changed, or deleted. • Tuples are more efficient than list due to python’s implementation. We can construct tuple in many ways: X=() #no item tuple X=(1,2,3) X=tuple(list1) X=1,2,3,4 Example: >>> x=(1,2,3) >>> print(x) (1, 2, 3) >>> x (1, 2, 3) ---------------------->>> x=() >>> x () --------------------------->>> x=[4,5,66,9] >>> y=tuple(x) >>> y (4, 5, 66, 9) ---------------------------->>> x=1,2,3,4 >>> x (1, 2, 3, 4) Some of the operations of tuple are: 36 Access tuple items Change tuple items Loop through a tuple Count() Index() Length() • • • • • • Access tuple items:Access tuple items by referring to the index number, inside square brackets >>> x=('a','b','c','g') >>> print(x[2]) c Change tuple items: Once a tuple is created, you cannot change its values. Tuples are unchangeable. >>> x=(2,5,7,'4',8) >>> x[1]=10 Traceback (most recent call last): File "<pyshell#41>", line 1, in <module> x[1]=10 TypeError: 'tuple' object does not support item assignment >>> x (2, 5, 7, '4', 8) # the value is still the same Loop through a tuple: We can loop the values of tuple using for loop >>> x=4,5,6,7,2,'aa' >>> for i in x: print(i) 4 5 6 7 2 aa Count ():Returns the number of times a specified value occurs in a tuple >>> x=(1,2,3,4,5,6,2,10,2,11,12,2) >>> x.count(2) 4 Index (): Searches the tuple for a specified value and returns the position of where it was found >>> x=(1,2,3,4,5,6,2,10,2,11,12,2) 37 >>> x.index(2) 1 (Or) >>> x=(1,2,3,4,5,6,2,10,2,11,12,2) >>> y=x.index(2) >>> print(y) 1 Length (): To know the number of items or values present in a tuple, we use len(). >>> x=(1,2,3,4,5,6,2,10,2,11,12,2) >>> y=len(x) >>> print(y) 12 Set: A set is a collection which is unordered and unindexed with no duplicate elements. In Python sets are written with curly brackets. • • To create an empty set we use set() Curly braces ‘{}’ or the set() function can be used to create sets We can construct tuple in many ways: X=set() X={3,5,6,8} X=set(list1) Example: >>> x={1,3,5,6} >>> x {1, 3, 5, 6} --------------------->>> x=set() >>> x set() -------------------->>> list1=[4,6,"dd",7] >>> x=set(list1) >>> x {4, 'dd', 6, 7} • • We cannot access items in a set by referring to an index, since sets are unordered the items has no index. But you can loop through the set items using a for loop, or ask if a specified value is present in a set, by using the in keyword. 38 Some of the basic set operations are: • Add() • Remove() • Len() • Item in x • Pop • Clear Add (): To add one item to a set use the add () method. To add more than one item to a set use the update () method. >>> x={"mrcet","college","cse","dept"} >>> x.add("autonomous") >>> x {'mrcet', 'dept', 'autonomous', 'cse', 'college'} --->>> x={1,2,3} >>> x.update("a","b") >>> x {1, 2, 3, 'a', 'b'} --------------->>> x={1,2,3} >>> x.update([4,5],[6,7,8]) >>> x {1, 2, 3, 4, 5, 6, 7, 8} Remove (): To remove an item from the set we use remove or discard methods. >>> x={1, 2, 3, 'a', 'b'} >>> x.remove(3) >>> x {1, 2, 'a', 'b'} Len (): To know the number of items present in a set, we use len(). >>> z={'mrcet', 'dept', 'autonomous', 'cse', 'college'} >>> len(z) 5 Item in X: you can loop through the set items using a for loop. >>> x={'a','b','c','d'} >>> for item in x: print(item) c d a b pop ():This method is used to remove an item, but this method will remove the last item. Remember that sets are unordered, so you will not know what item that gets removed. 39 >>> x={1, 2, 3, 4, 5, 6, 7, 8} >>> x.pop() 1 >>> x {2, 3, 4, 5, 6, 7, 8} Clear (): This method will the set as empty. >>> x={2, 3, 4, 5, 6, 7, 8} >>> x.clear() >>> x set() The set also consist of some mathematical operations like: Intersection AND & Union OR Symmetric Diff XOR Diff In set1 but not in set2 Subset set2 contains set1 Superset set1 contains set2 Some examples: | ^ set1-set2 set1<=set2 set1>=set2 >>> x={1,2,3,4} >>> y={4,5,6,7} >>> print(x|y) {1, 2, 3, 4, 5, 6, 7} ---------------------------->>> x={1,2,3,4} >>> y={4,5,6,7} >>> print(x&y) {4} --------------------------->>> A = {1, 2, 3, 4, 5} >>> B = {4, 5, 6, 7, 8} >>> print(A-B) {1, 2, 3} -------------------------->>> B = {4, 5, 6, 7, 8} >>> A = {1, 2, 3, 4, 5} >>> print(B^A) {1, 2, 3, 6, 7, 8} Dictionaries: A dictionary is a collection which is unordered, changeable and indexed. In Python dictionaries are written with curly brackets, and they have keys and values. • Key-value pairs • Unordered 40 We can construct or create dictionary like: X={1:’A’,2:’B’,3:’c’} X=dict([(‘a’,3) (‘b’,4)] X=dict(‘A’=1,’B’ =2) Examples: >>> dict1 = {"brand":"mrcet","model":"college","year":2004} >>> dict1 {'brand': 'mrcet', 'model': 'college', 'year': 2004} ------------------To access specific value of a dictionary, we must pass its key, >>> dict1 = {"brand":"mrcet","model":"college","year":2004} >>> x=dict1["brand"] >>> x 'mrcet' --------------------To access keys and values and items of dictionary: >>> dict1 = {"brand":"mrcet","model":"college","year":2004} >>> dict1.keys() dict_keys(['brand', 'model', 'year']) >>> dict1.values() dict_values(['mrcet', 'college', 2004]) >>> dict1.items() dict_items([('brand', 'mrcet'), ('model', 'college'), ('year', 2004)]) ---------------------------------------------->>> for items in dict1.values(): print(items) mrcet college 2004 >>> for items in dict1.keys(): print(items) brand model year >>> for i in dict1.items(): print(i) ('brand', 'mrcet') ('model', 'college') 41 ('year', 2004) • • • • Some of the operations are: Add/change Remove Length Delete Add/change values:You can change the value of a specific item by referring to its key name >>> dict1 = {"brand":"mrcet","model":"college","year":2004} >>> dict1["year"]=2005 >>> dict1 {'brand': 'mrcet', 'model': 'college', 'year': 2005} Remove(): It removes or pop the specific item of dictionary. >>> dict1 = {"brand":"mrcet","model":"college","year":2004} >>> print(dict1.pop("model")) college >>> dict1 {'brand': 'mrcet', 'year': 2005} Delete: Deletes a particular item. >>> x = {1:1, 2:4, 3:9, 4:16, 5:25} >>> del x[5] >>> x Length: we use len() method to get the length of dictionary. >>>{1: 1, 2: 4, 3: 9, 4: 16} {1: 1, 2: 4, 3: 9, 4: 16} >>> y=len(x) >>> y 4 Iterating over (key, value) pairs: >>> x = {1:1, 2:4, 3:9, 4:16, 5:25} >>> for key in x: print(key, x[key]) 11 24 39 4 16 5 25 >>> for k,v in x.items(): print(k,v) 42 11 24 39 4 16 5 25 List of Dictionaries: >>> customers = [{"uid":1,"name":"John"}, {"uid":2,"name":"Smith"}, {"uid":3,"name":"Andersson"}, ] >>>>>> print(customers) [{'uid': 1, 'name': 'John'}, {'uid': 2, 'name': 'Smith'}, {'uid': 3, 'name': 'Andersson'}] ## Print the uid and name of each customer >>> for x in customers: print(x["uid"], x["name"]) 1 John 2 Smith 3 Andersson ##Modify an entry, This will change the name of customer 2 from Smith to Charlie >>> customers[2]["name"]="charlie" >>> print(customers) [{'uid': 1, 'name': 'John'}, {'uid': 2, 'name': 'Smith'}, {'uid': 3, 'name': 'charlie'}] ##Add a new field to each entry >>> for x in customers: x["password"]="123456" # any initial value >>> print(customers) [{'uid': 1, 'name': 'John', 'password': '123456'}, {'uid': 2, 'name': 'Smith', 'password': '123456'}, {'uid': 3, 'name': 'charlie', 'password': '123456'}] ##Delete a field >>> del customers[1] >>> print(customers) [{'uid': 1, 'name': 'John', 'password': '123456'}, {'uid': 3, 'name': 'charlie', 'password': '123456'}] >>> del customers[1] >>> print(customers) [{'uid': 1, 'name': 'John', 'password': '123456'}] 43 ##Delete all fields >>> for x in customers: del x["uid"] >>> x {'name': 'John', 'password': '123456'} Sequences: A sequence is a succession of values bound together by a container that reflects their type. Almost every stream that you put in python is a sequence. Some of them are: • • • • String List Tuples Range object String: A string is a group of characters. Since Python has no provision for arrays, we simply use strings. This is how we declare a string. We can use a pair of single or double quotes. Every string object is of the type ‘str’. >>> type("name") <class 'str'> >>> name=str() >>> name '' >>> a=str('mrcet') >>> a 'mrcet' >>> a=str(mrcet) >>> a[2] 'c' List: A list is an ordered group of items. To declare it, we use square brackets. >>> college=["cse","it","eee","ece","mech","aero"] >>> college[1] 'it' >>> college[:2] ['cse', 'it'] >>> college[:3] ['cse', 'it', 'eee'] >>> college[3:] ['ece', 'mech', 'aero'] >>> college[0]="csedept" 44 >>> college ['csedept', 'it', 'eee', 'ece', 'mech', 'aero'] Tuple: It is an immutable group of items. When we say immutable, we mean we cannot change a single value once we declare it. >>> x=[1,2,3] >>> y=tuple(x) >>> y (1, 2, 3) >>> hello=tuple(["mrcet","college"]) >>> hello ('mrcet', 'college') Range object: A range() object lends us a range to iterate on; it gives us a list of numbers. >>> a=range(4) >>> type(a) <class 'range'> >>> for i in range(1,6,2): print(i) 1 3 5 Some of the python sequence operations and functions are: 1. Indexing 2. Slicing 3. Adding/Concatenation 4. Multiplying 5. Checking membership 6. Iterating 7. Len() 8. Min() 9. Max() 10. Sum() 11. Sorted() 12. Count() 13. Index() 1. Indexing Access any item in the sequence using its index. string List 45 >>> x='mrcet' >>> print(x[2]) c >>> x=['a','b','c'] >>> print(x[1]) b 2. Slicing Slice out substrings, sub lists, sub tuples using index [start : stop : step size] >>> x='computer' >>> x[1:4] 'omp' >>> x[1:6:2] 'opt' >>> x[3:] 'puter' >>> x[:5] 'compu' >>> x[-1] 'r' >>> x[-3:] 'ter' >>> x[:-2] 'comput' >>> x[::-2] 'rtpo' >>> x[::-1] 'retupmoc' 3. Adding/concatenation: Combine 2 sequences of same type using +. string List >>> x='mrcet' + 'college' >>> x=['a','b'] + ['c'] >>> print(x) >>> print(x) Mrcetcollege ['a', 'b', 'c'] 46 4. Multiplying: Multiply a sequence using *. string >>> x='mrcet'*3 >>> x 'mrcetmrcetmrcet' List >>> x=[3,4]*2 >>> x [3, 4, 3, 4] 5. Checking Membership: Test whether an item is in or not in a sequence. string >>> x='mrcet' >>> print('c' in x) True List >>> x=['a','b','c'] >>> print('a' not in x) False 6. Iterating: Iterate through the items in asequence >>> x=[1,2,3] >>> for item in x: print(item*2) 2 4 6 If we want to display the items of a given list with index then we have to use “enumerate” keyword. >>> x=[5,6,7] >>> for item,index in enumerate(x): print(item,index) 05 16 27 7. len(): It will count the number of items in a given sequence. string List 47 >>> x="mrcet" >>> print(len(x)) 5 >>> x=["aa","b",'c','cc'] >>> print(len(x)) 4 8. min(): Finds the minimum item in a given sequence lexicographically. string >>> x="mrcet" >>> print(min(x)) c List >>> x=["apple","ant1","ant"] >>> print(min(x)) ant It is an alpha-numeric type but cannot mix types. >>> x=["apple","ant1","ant",11] >>> print(min(x)) Traceback (most recent call last): File "<pyshell#73>", line 1, in <module> print(min(x)) TypeError: '<' not supported between instances of 'int' and 'str' 9. max(): Finds the maximum item in a given sequence string >>> x='cognizant' >>> print(max(x)) z List >>> x=["hello","yummy","zebra"] >>> print(max(x)) zebra It is an alpha-numeric type but cannot mix types. >>> x=["hello","yummy1","zebra1",22] >>> print(max(x)) Traceback (most recent call last): File "<pyshell#79>", line 1, in <module> print(max(x)) TypeError: '>' not supported between instances of 'int' and 'str' 10. Sum: Finds the sum of items in a sequence >>> x=[1,2,3,4,5] >>> print(sum(x)) 48 15 >>> print(sum(x[-2:])) 9 Entire string must be numeric type. >>> x=[1,2,3,4,5,"mrcet"] >>> print(sum(x)) Traceback (most recent call last): File "<pyshell#83>", line 1, in <module> print(sum(x)) TypeError: unsupported operand type(s) for +: 'int' and 'str' 11. Sorted(): Returns a new list of items in sorted order but does not change the original list. string >>> x='college' >>> print(sorted(x)) ['c', 'e', 'e', 'g', 'l', 'l', 'o'] List >>> x=['a','r','g','c','j','z'] >>> print(sorted(x)) ['a', 'c', 'g', 'j', 'r', 'z'] 12. Count(): It returns the count of an item string >>> x='college' >>> print(x.count('l')) 2 >>> 'college'.count('l') 2 List >>> x=['a','b','a','a','c','a'] >>> print(x.count('a')) 4 13. Index() Returns the index of first occurrence string >>> x='college' >>> print(x.index('l')) 2 List >>> x=['a','b','a','a','c','a'] >>> print(x.index('a')) 0 49 Comprehensions: List: List comprehensions provide a concise way to create lists. Common applications are to make new lists where each element is the result of some operations applied to each member of another sequence or iterable, or to create a subsequence of those elements that satisfy a certain condition. For example, assume we want to create a list of squares, like: >>> list1=[] >>> for x in range(10): list1.append(x**2) >>> list1 [0, 1, 4, 9, 16, 25, 36, 49, 64, 81] (or) This is also equivalent to >>> list1=list(map(lambda x:x**2, range(10))) >>> list1 [0, 1, 4, 9, 16, 25, 36, 49, 64, 81] (or) Which is more concise and redable. >>> list1=[x**2 for x in range(10)] >>> list1 [0, 1, 4, 9, 16, 25, 36, 49, 64, 81] Similarily some examples: 50 >>> x=[m for m in range(8)] >>> print(x) [0, 1, 2, 3, 4, 5, 6, 7] >>> x=[z**2 for z in range(10) if z>4] >>> print(x) [25, 36, 49, 64, 81] >>> x=[x ** 2 for x in range (1, 11) if x % 2 == 1] >>> print(x) [1, 9, 25, 49, 81] >>> a=5 >>> table = [[a, b, a * b] for b in range(1, 11)] >>> for i in table: print(i) [5, 1, 5] [5, 2, 10] [5, 3, 15] [5, 4, 20] [5, 5, 25] [5, 6, 30] [5, 7, 35] [5, 8, 40] [5, 9, 45] [5, 10, 50] Tuple: Tuple Comprehensions are special: The result of a tuple comprehension is special. You might expect it to produce a tuple, but what it does is produce a special "generator" object that we can iterate over. For example: >>> x = (i for i in 'abc') #tuple comprehension >>> x <generator object <genexpr> at 0x033EEC30> >>> print(x) <generator object <genexpr> at 0x033EEC30> You might expect this to print as ('a', 'b', 'c') but it prints as <generator object <genexpr> at 0x02AAD710> The result of a tuple comprehension is not a tuple: it is actually a generator. The only thing that you need to know now about a generator now is that you can iterate over it, but 51 ONLY ONCE. So, given the code >>> x = (i for i in 'abc') >>> for i in x: print(i) a b c Create a list of 2-tuples like (number, square): >>> z=[(x, x**2) for x in range(6)] >>> z [(0, 0), (1, 1), (2, 4), (3, 9), (4, 16), (5, 25)] Set: Similarly to list comprehensions, set comprehensions are also supported: >>> a = {x for x in 'abracadabra' if x not in 'abc'} >>> a {'r', 'd'} >>> x={3*x for x in range(10) if x>5} >>> x {24, 18, 27, 21} Dictionary: Dictionary comprehensions can be used to create dictionaries from arbitrary key and value expressions: >>> z={x: x**2 for x in (2,4,6)} >>> z {2: 4, 4: 16, 6: 36} >>> dict11 = {x: x*x for x in range(6)} >>> dict11 {0: 0, 1: 1, 2: 4, 3: 9, 4: 16, 5: 25} 52 String Sliceing in Python If you want to slice strings in Python, it’s going to be as simple as this one line below. res_s =s[ start_pos:end_pos:step ] Here, • res_s stores the returned sub-string, • s is the given string, • start_pos is the starting index from which we need to slice the string s, • end_pos is the ending index, before which the slicing operation would end, • step is the steps the slicing process would take from start_pos to end_pos. Note: All of the above three parameters are optional. By default, start_pos is set to 0, end_pos is considered equal to the length of the string, and step is set to 1. Now let us take some examples to understand how to slice strings in Python in a better way. Slice Strings in Python – Examples Slicing Python strings can be done in different ways. Usually, we access string elements(characters) using simple indexing that starts from 0 up to n-1(n is the length of the string). Hence for accessing the 1st element of a string string1, we can simply use the below code. s1 =String1[0] Again, there is another way to access these characters, that is, using negative indexing. Negative indexing starts from -1 to -n(n is the length for the given string). Note, negative indexing is done from the other end of the string. Hence, to access the first character this time we need to follow the below-given code. s1 =String1[-n] Now let us look at some ways following which we can slice a string using the above concept. 1. Slice Strings in Python with Start and End We can easily slice a given string by mentioning the starting and ending indices for the desired sub-string we are looking for. Look at the example given below, it explains string slicing using starting and ending indices for both usual and negative indexing method. #string slicing with two parameters 53 s ="Hello World!" res1 =s[2:8] res2 =s[-4:-1] #using negative indexing print("Result1 = ",res1) print("Result2 = ",res2) Output: Result1 = llo Wo Result2 = rld Here, • We initialize a string, s as “Hello World!”, • At first, we slice the given string with starting index 2 and ending index as 8. This means that the resultant sub-string would contain the characters from s[2] to s[8-1], • Similarly, for the next one, the resultant sub-string should contain characters from s[-4] to s[(-1)-1]. Hence, our output is justified. 2. Slice Strings Using Only the Start or End As mentioned earlier, all of the three parameters for string slicing are optional. Hence, we can easily accomplish our tasks using one parameter. Look at the code below to get a clear understanding. #string slicing with one parameter s1="Charlie" s2="Jordan" res1 =s1[2:] #default value of ending position is set to the length of string res2 =s2[:4] #default value of starting position is set to 0 print("Result1 = ",res1) print("Result2 = ",res2) Output: Result1 = arlie Result2 = Jord Here, • We firstly initialize two strings, s1 and s2, • For slicing both of them we just mention the start_pos for s1, and end_pos for s2 only, • Hence for res1, it contains a sub-string of s1 from index 2(as mentioned) up to the last(by default it is set to n-1). Whereas for res2, the range of the indices lies from 0 upto 4(mentioned). 54 3. Slice Strings in Python with Step Parameter The step value decides the jump the slicing operation would take from one index to the other. Look at the example below carefully. #string slicing with step parameter s="Python" s1="Kotlin" res =s[0:5:2] res1 =s1[-1:-4:-2] #using negative parameters print("Resultant sliced string = ",res) print("Resultant sliced string(negative parameters) = ",res1) Output: Resultant sliced string = Pto Resultant sliced string(negative parameters) = nl In the code above, • We initialize two strings s and s1, and try to slice them for the given starting and ending indices as we did for our first example, • But this time we have mentioned a step value which was set to 1 by default for the previous examples, • For res, having a step size 2 means, that while the traversal to get the substring from index 0 to 4, each time the index would be increased by a value of 2. That is the first character is s[0] (‘P’), the next characters in the sub-string would be s[0+2], and s[2+2], until the index is just less than 5. • For the next one i.e. res1, the step mentioned is (-2). Hence, similar to the previous case, the characters in the substring would be s1[-1] , then s1[(-1)+(-2)] or s1[-3] until the index is just less than (-4). 4. Reversing a String using Slicing in Python With the use of negative index string slicing in Python, we can also reverse the string and store it in another variable. For doing so we just need to mention a step size of (-1). Let us see how that works in the example given below. #reversing string using string slicing s="AskPython" rev_s =s[::-1] #reverse string stored into rev_s print(rev_s) Output: nohtyPksA As we can see, the string s is reversed and stored into rev_s. Note: For this case too, the original string remains intact and untouched. 55 UNIT – III Arrays - Array Data Structure Overview, Types of Arrays, Operations on Arrays, Arrays vs List.Searching - Linear Search and Binary Search.Sorting - Bubble Sort, Selection Sort, Insertion Sort, Merge Sort, Quick Sort Python arrays: Array is a container which can hold a fix number of items and these items should be of the same type. Most of the data structures make use of arrays to implement their algorithms. Following are the important terms to understand the concept of Array. • Element− Each item stored in an array is called an element. • Index − Each location of an element in an array has a numerical index, which is used to identify the element. Array Representation Arrays can be declared in various ways in different languages. Below is an illustration. Elements Int array [10] = {10, 20, 30, 40, 50, 60, 70, 80, 85, 90} Type Name Size Index 0 As per the above illustration, following are the important points to be considered. • Index starts with 0. • Array length is 10 which means it can store 10 elements. • Each element can be accessed via its index. For example, we can fetch an element at index 6 as 70 Basic Operations Following are the basic operations supported by an array. • Traverse − print all the array elements one by one. • Insertion − Adds an element at the given index. • Deletion − Deletes an element at the given index. • Search − Searches an element using the given index or by the value. • Update − Updates an element at the given index. Array is created in Python by importing array module to the python program. Then the array is declared as shown below. from array import * arrayName=array(typecode, [initializers]) 56 Typecode are the codes that are used to define the type of value the array will hold. Some common typecodes used are: Typecode Value b Represents signed integer of size 1 byte/td> B Represents unsigned integer of size 1 byte c Represents character of size 1 byte i Represents signed integer of size 2 bytes I Represents unsigned integer of size 2 bytes f Represents floating point of size 4 bytes d Represents floating point of size 8 bytes Creating an array: from array import * array1 = array('i', [10,20,30,40,50]) for x in array1: print(x) Output: >>> RESTART: C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/arr.py 10 20 30 40 50 Accessing Array Element We can access each element of an array using the index of the element. from array import * array1 = array('i', [10,20,30,40,50]) print (array1[0]) print (array1[2]) Output: RESTART: C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/arr2.py 10 57 30 Insertion Operation Insert operation is to insert one or more data elements into an array. Based on the requirement, a new element can be added at the beginning, end, or any given index of array. Here, we add a data element at the middle of the array using the python in-built insert() method. from array import * array1 = array('i', [10,20,30,40,50]) array1.insert(1,60) for x in array1: print(x) Output: ============================================ C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/arr3.py =========================================== 10 60 20 30 40 50 >>> Deletion Operation Deletion refers to removing an existing element from the array and re-organizing all elements of an array. Here, we remove a data element at the middle of the array using the python in-built remove() method. from array import * array1 = array('i', [10,20,30,40,50]) array1.remove(40) for x in array1: print(x) Output: ============================================ C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/arr4.py =========================================== 10 20 30 50 Search Operation You can perform a search for an array element based on its value or its index. Here, we search a data element using the python in-built index() method. from array import * array1 = array('i', [10,20,30,40,50]) print (array1.index(40)) Output: ============================================ 58 C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/arr5.py =========================================== 3 >>> Update Operation Update operation refers to updating an existing element from the array at a given index. Here, we simply reassign a new value to the desired index we want to update. from array import * array1 = array('i', [10,20,30,40,50]) array1[2] = 80 for x in array1: print(x) Output: ============================================ C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/arr6.py =========================================== 10 20 80 40 50 Arrays vs List Both list and array and list are used to store the data in Python. These data structures allow us to indexing, slicing, and iterating. But they have little dissimilarity with each other. Python List A list is a built-in, linear data structure of Python. It is used to store the data in a sequence manner. We can perform several operations to list, such as indexing, iterating, and slicing. The list comes with the following features. o The list elements are enclosed with a square bracket, and each element is separated by a comma (,). o It is a mutable type which means we can modify the list items after their creation. o The lists are ordered, which means items are stored in a specific order. We can use indexing to access the list element. o We can store the items of different data types. We can combine strings, integers, and objects in the same list. Below is an example of a list. Example 1. list = [31, 60, 19, 12] 59 2. print(list) 3. print(type(list)) Output: [31, 60, 19, 12] <class 'list'> Example - 2 1. # creating a list containing elements 2. # belonging to different data types 3. list1 = [1,"Yash",['a','e']] 4. print(list1) Output: [1, 'Yash', ['a', 'e']] In the above list, the first element is an integer; the second is a string and third is a list of characters. Array in Python An array is also a linear data structure that stores the data. It is also ordered, mutable, and enclosed in square brackets. It can store the non-unique items. But there are restrictions to store the different data type values. To work with the Array in Python, we need to import either an array module or a Numpy. 1. import array as arr 2. or 3. import numpy as np Elements are allocated in contiguous memory location that allows us to easy modification, addition, deletion, accessing of element. Moreover, we need to specify the data type. Let's understand the following example. Example 1. Import array as arr 2. array_1 = arr.array("i", [31, 60,19, 12]) 3. print(array_1) 4. print(type(array_1)) Output: 60 array('i', [31, 60, 19, 12]) <class 'array.array'> Example - 2: Using Numpy array 1. import numpy as np 2. array_sample = np.array(["str", 'sunil', 'sachin', 'megha', 'deepti']) 3. print (array_sample) 4. print(type(array_sample)) Output: ['numbers' 'sunil' 'sachin' 'megha' 'deepti'] <class 'numpy.ndarray'> We have specified the string type and stored the string value. Difference between Array and List Searching - Linear Search and Binary Search. 61 There are two types of searching o Linear Search o Binary Search Both techniques are widely used to search an element in the given list. Linear Search What is a Linear Search? Linear search is a method of finding elements within a list. It is also called a sequential search. It is the simplest searching algorithm because it searches the desired element in a sequential manner. It compares each and every element with the value that we are searching for. If both are matched, the element is found, and the algorithm returns the key's index position. Concept of Linear Search Let's understand the following steps to find the element key = 7 in the given list. Step - 1: Start the search from the first element and Check key = 7 with each element of list x. Step - 2: If element is found, return the index position of the key. Step - 3: If element is not found, return element is not present. 62 Linear Search Algorithm There is list of n elements and key value to be searched. Below is the linear search algorithm. 1. LinearSearch(list, key) 2. for each item in the list 3. if item == value 4. return its index position 5. return -1 Python Program Let's understand the following Python implementation of the linear search algorithm. Program 1. def linear_Search(list1, n, key): 2. 3. # Searching list1 sequentially 4. for i in range(0, n): 5. 6. 7. if (list1[i] == key): return i return -1 8. 9. 10. list1 = [1 ,3, 5, 4, 7, 9] 11. key = 7 12. 13. n = len(list1) 14. res = linear_Search(list1, n, key) 15. if(res == -1): 16. print("Element not found") 63 17. else: 18. print("Element found at index: ", res) Output: Element found at index: 4 Explanation: In the above code, we have created a function linear_Search(), which takes three arguments - list1, length of the list, and number to search. We defined for loop and iterate each element and compare to the key value. If element is found, return the index else return -1 which means element is not present in the list. Linear Search Complexity Time complexity of linear search algorithm o Base Case - O(1) o Average Case - O(n) o Worst Case -O(n) Linear search algorithm is suitable for smaller list (<100) because it check every element to get the desired number. Suppose there are 10,000 element list and desired element is available at the last position, this will consume much time by comparing with each element of the list. To get the fast result, we can use the binary search algorithm. Binary Search: A binary search is an algorithm to find a particular element in the list. Suppose we have a list of thousand elements, and we need to get an index position of a particular element. We can find the element's index position very fast using the binary search algorithm. There are many searching algorithms but the binary search is most popular among them. The elements in the list must be sorted to apply the binary search algorithm. If elements are not sorted then sort them first. Let's understand the concept of binary search. Concept of Binary Search The divide and conquer approach technique is followed by the recursive method. In this method, a function is called itself again and again until it found an element in the list. 64 A set of statements is repeated multiple times to find an element's index position in the iterative method. The while loop is used for accomplish this task. Binary search is more effective than the linear search because we don't need to search each list index. The list must be sorted to achieve the binary search algorithm. Let's have a step by step implementation of binary search. We have a sorted list of elements, and we are looking for the index position of 45. [12, 24, 32, 39, 45, 50, 54] So, we are setting two pointers in our list. One pointer is used to denote the smaller value called low and the second pointer is used to denote the highest value called high. Next, we calculate the value of the middle element in the array. 1. mid = (low+high)/2 2. Here, the low is 0 and the high is 7. 3. mid = (0+7)/2 4. mid = 3 (Integer) Now, we will compare the searched element to the mid index value. In this case, 32 is not equal to 45. So we need to do further comparison to find the element. If the number we are searching equal to the mid. Then return mid otherwise move to the further comparison. The number to be search is greater than the middle number, we compare the n with the middle element of the elements on the right side of mid and set low to low = mid + 1. Otherwise, compare the n with the middle element of the elements on the left side of mid and set high to high = mid - 1. 65 Repeat until the number that we are searching for is found. Implement a Binary Search in Python First, we implement a binary search with the iterative method. We will repeat a set of statements and iterate every item of the list. We will find the middle value until the search is complete. Let's understand the following program of the iterative method. Python Implementation 1. # Iterative Binary Search Function method Python Implementation 2. # It returns index of n in given list1 if present, 3. # else returns -1 4. def binary_search(list1, n): 5. low = 0 6. high = len(list1) - 1 7. mid = 0 66 8. 9. while low <= high: 10. # for get integer result 11. mid = (high + low) // 2 12. 13. # Check if n is present at mid 14. if list1[mid] < n: 15. low = mid + 1 16. 17. # If n is greater, compare to the right of mid 18. elif list1[mid] > n: 19. high = mid - 1 20. 21. # If n is smaller, compared to the left of mid 22. else: 23. return mid 24. 25. 26. # element was not present in the list, return -1 return -1 27. 28. 29. # Initial list1 30. list1 = [12, 24, 32, 39, 45, 50, 54] 31. n = 45 32. 33. # Function call 34. result = binary_search(list1, n) 35. 36. if result != -1: 37. print("Element is present at index", str(result)) 38. else: 39. print("Element is not present in list1") 67 Output: Element is present at index 4 Explanation: In the above program o We have created a function called binary_search() function which takes two arguments - a list to sorted and a number to be searched. o We have declared two variables to store the lowest and highest values in the list. The low is assigned initial value to 0, high to len(list1) - 1 and mid as 0. o Next, we have declared the while loop with the condition that the lowest is equal and smaller than the highest The while loop will iterate if the number has not been found yet. o In the while loop, we find the mid value and compare the index value to the number we are searching for. o If the value of the mid-index is smaller than n, we increase the mid value by 1 and assign it to The search moves to the left side. o Otherwise, decrease the mid value and assign it to the high. The search moves to the right side. o If the n is equal to the mid value then return mid. o This will happen until the low is equal and smaller than the high. o If we reach at the end of the function, then the element is not present in the list. We return -1 to the calling function. Sorting - Bubble Sort, Selection Sort, Insertion Sort, Merge Sort, Quick Sort. Bubble Sort: It is a simple sorting algorithm which sorts ‘n’ number of elements in the list by comparing the ach pair of adjacent items and swaps them if they are in wrong order. Algorithm: 1. Starting with the first element (index=0), compare the current element with the next element of a list. 2. If the current element is greater (>) than the next element of the list then swap them. 3. If the current element is less (<) than the next element of the list move to the next element. 4. Repeat step 1 until it correct order is framed. 68 so here we are comparing values again For ex: list1= [10, 15, 4, 23, 0] If > --- yes ---- swap and again, so we use loops. If < --- No ---- Do nothing/remains same #Write a python program to arrange the elements in ascending order using bubble sort: list1=[9,16,6,26,0] print("unsorted list1 is", list1) for j in range(len(list1)-1): for i in range(len(list1)-1): if list1[i]>list1[i+1]: list1[i],list1[i+1]=list1[i+1],list1[i] print(list1) else: print(list1) print( ) print("sorted list is",list1) Output: unsorted list1 is [9, 16, 6, 26, 0] [9, 16, 6, 26, 0] [9, 6, 16, 26, 0] [9, 6, 16, 26, 0] [9, 6, 16, 0, 26] [6, 9, 16, 0, 26] [6, 9, 16, 0, 26] [6, 9, 0, 16, 26] [6, 9, 0, 16, 26] [6, 9, 0, 16, 26] [6, 0, 9, 16, 26] [6, 0, 9, 16, 26] [6, 0, 9, 16, 26] [0, 6, 9, 16, 26] [0, 6, 9, 16, 26] [0, 6, 9, 16, 26] [0, 6, 9, 16, 26] sorted list is [0, 6, 9, 16, 26] #If we want to reduce no of iterations/steps in output: C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/bubb.py 69 list1=[9,16,6,26,0] print("unsorted list1 is", list1) for j in range(len(list1)-1,0,-1): for i in range(j): if list1[i]>list1[i+1]: list1[i],list1[i+1]=list1[i+1],list1[i] print(list1) else: print(list1) print( ) print("sorted list is",list1) Output: C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/bubb2.py unsorted list1 is [9, 16, 6, 26, 0] [9, 16, 6, 26, 0] [9, 6, 16, 26, 0] [9, 6, 16, 26, 0] [9, 6, 16, 0, 26] [6, 9, 16, 0, 26] [6, 9, 16, 0, 26] [6, 9, 0, 16, 26] [6, 9, 0, 16, 26] [6, 0, 9, 16, 26] [0, 6, 9, 16, 26] sorted list is [0, 6, 9, 16, 26] # In a different way: list1=[9,16,6,26,0] print("unsorted list1 is", list1) for j in range(len(list1)-1): for i in range(len(list1)-1-j): if list1[i]>list1[i+1]: list1[i],list1[i+1]=list1[i+1],list1[i] print(list1) else: print(list1) print( ) print("sorted list is",list1) Output: C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/bubb3.py 70 unsorted list1 is [9, 16, 6, 26, 0] [9, 16, 6, 26, 0] [9, 6, 16, 26, 0] [9, 6, 16, 26, 0] [9, 6, 16, 0, 26] [6, 9, 16, 0, 26] [6, 9, 16, 0, 26] [6, 9, 0, 16, 26] [6, 9, 0, 16, 26] [6, 0, 9, 16, 26] [0, 6, 9, 16, 26] sorted list is [0, 6, 9, 16, 26] # Program to give input from the user to sort the elements list1=[] num=int(input("enter how many numbers:")) print("enter values") for k in range(num): list1.append(int(input())) print("unsorted list1 is", list1) for j in range(len(list1)-1): for i in range(len(list1)-1): if list1[i]>list1[i+1]: list1[i],list1[i+1]=list1[i+1],list1[i] print(list1) else: print(list1) print( ) print("sorted list is",list1) Output: C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/bubb4.py enter how many numbers:5 enter values 5 77 4 66 30 unsorted list1 is [5, 77, 4, 66, 30] [5, 77, 4, 66, 30] 71 [5, 4, 77, 66, 30] [5, 4, 66, 77, 30] [5, 4, 66, 30, 77] [4, 5, 66, 30, 77] [4, 5, 66, 30, 77] [4, 5, 30, 66, 77] [4, 5, 30, 66, 77] [4, 5, 30, 66, 77] [4, 5, 30, 66, 77] [4, 5, 30, 66, 77] [4, 5, 30, 66, 77] [4, 5, 30, 66, 77] [4, 5, 30, 66, 77] [4, 5, 30, 66, 77] [4, 5, 30, 66, 77] sorted list is [4, 5, 30, 66, 77] #bubble sort program for descending order list1=[9,16,6,26,0] print("unsorted list1 is", list1) for j in range(len(list1)-1): for i in range(len(list1)-1): if list1[i]<list1[i+1]: list1[i],list1[i+1]=list1[i+1],list1[i] print(list1) else: print(list1) print( ) print("sorted list is",list1) Output: C:/Users/MRCET/AppData/Local/Programs/Python/Python38-2/pyyy/bubbdesc.py unsorted list1 is [9, 16, 6, 26, 0] [16, 9, 6, 26, 0] [16, 9, 6, 26, 0] [16, 9, 26, 6, 0] [16, 9, 26, 6, 0] [16, 9, 26, 6, 0] [16, 26, 9, 6, 0] [16, 26, 9, 6, 0] [16, 26, 9, 6, 0] [26, 16, 9, 6, 0] 72 [26, 16, 9, 6, 0] [26, 16, 9, 6, 0] [26, 16, 9, 6, 0] [26, 16, 9, 6, 0] [26, 16, 9, 6, 0] [26, 16, 9, 6, 0] [26, 16, 9, 6, 0] sorted list is [26, 16, 9, 6, 0] Selection Sort: Sort (): Built-in list method Sorted (): built in function • • • Generally this algorithm is called as in-place comparison based algorithm. We compare numbers and place them in correct position. Search the list and find out the min value, this we can do it by min () method. We can take min value as the first element of the list and compare with the next element until we find small value. Algorithm: 1. Starting from the first element search for smallest/biggest element in the list of numbers. 2. Swap min/max number with first element 3. Take the sub-list (ignore sorted part) and repeat step 1 and 2 until all the elements are sorted. #Write a python program to arrange the elements in ascending order using selection sort: list1=[5,3,7,1,9,6] print(list1) for i in range(len(list1)): min_val=min(list1[i:]) min_ind=list1.index(min_val) list1[i],list1[min_ind]=list1[min_ind],list1[i] print(list1) Output: C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/selectasce.py [5, 3, 7, 1, 9, 6] [1, 3, 7, 5, 9, 6] [1, 3, 7, 5, 9, 6] [1, 3, 5, 7, 9, 6] [1, 3, 5, 6, 9, 7] [1, 3, 5, 6, 7, 9] [1, 3, 5, 6, 7, 9] #Write a python program to arrange the elements in descending order using selection sort: list1=[5,3,7,1,9,6] print(list1) for i in range(len(list1)): min_val=max(list1[i:]) min_ind=list1.index(min_val) 73 list1[i],list1[min_ind]=list1[min_ind],list1[i] print(list1) Output: C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/selecdecs.py [5, 3, 7, 1, 9, 6] [9, 7, 6, 5, 3, 1] Note: If we want the elements to be sorted in descending order use max () method in place of min (). Insertion Sort: • Insertion sort is not a fast sorting algorithm. It is useful only for small datasets. • It is a simple sorting algorithm that builds the final sorted list one item at a time. Algorithm: 1. Consider the first element to be sorted & the rest to be unsorted. 2. Take the first element in unsorted order (u1) and compare it with sorted part elements(s1) 3. If u1<s1 then insert u1 in the correct order, else leave as it is. 4. Take the next element in the unsorted part and compare with sorted element. 5. Repeat step 3 and step 4 until all the elements get sorted. # Write a python program to arrange the elements in ascending order using insertion sort (with functions) def insertionsort(my_list): #we need to sorrt the unsorted part at a time. for index in range(1,len(my_list)): current_element=my_list[index] pos=index while current_element<my_list[pos-1]and pos>0: my_list[pos]=my_list[pos-1] pos=pos-1 my_list[pos]=current_element list1=[3,5,1,0,10,2] num=int(input(“enter how many elements to be in list”)) insertionsort(list1) list1=[int(input()) for i in range (num)] print(list1) Output: C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/inserti.py [0, 1, 2, 3, 5, 10] # Write a python program to arrange the elements in descending order using insertion sort (with functions) def insertionsort(my_list): #we need to sorrt the unsorted part at a time. for index in range(1,len(my_list)): current_element=my_list[index] pos=index while current_element>my_list[pos-1]and pos>0: my_list[pos]=my_list[pos-1] pos=pos-1 my_list[pos]=current_element #list1=[3,5,1,0,10,2] 74 #insertionsort(list1) #print(list1) num=int(input("enter how many elements to be in list")) list1=[int(input())for i in range(num)] insertionsort(list1) print(list1) Output: C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/insertdesc.py enter how many elements to be in list 5 8 1 4 10 2 [10, 8, 4, 2, 1] Merge Sort: Generally this merge sort works on the basis of divide and conquer algorithm. The three steps need to be followed is divide, conquer and combine. We will be dividing the unsorted list into sub list until the single element in a list is found. Algorithm: 1. Split the unsorted list. 2. Compare each of the elements and group them 3. Repeat step 2 until whole list is merged and sorted. # Write a python program to arrange the elements in ascending order using Merge sort (with functions) def mergesort(list1): if len(list1)>1: mid=len(list1)//2 left_list=list1[:mid] right_list=list1[mid:] mergesort(left_list) mergesort(right_list) i=0 j=0 k=0 while i<len(left_list) and j<len(right_list): if left_list[i]<right_list[j]: list1[k]=left_list[i] i=i+1 k=k+1 else: list1[k]=right_list[j] j=j+1 k=k+1 while i<len(left_list): 75 list1[k]=left_list[i] i=i+1 k=k+1 while j<len(right_list): list1[k]=right_list[j] j=j+1 k=k+1 num=int(input("how many numbers in list1")) list1=[int(input()) for x in range(num)] mergesort(list1) print("sorted list1",list1) Output: C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/merg.py how many numbers in list15 5 9 10 1 66 sorted list1 [1, 5, 9, 10, 66] Quick Sort: Algorithm: 1. Select the pivot element 2. Find out the correct position of pivot element in the list by rearranging it. 3. Divide the list based on pivot element 4. Sort the sub list recursively Note: Pivot element can be first, last, random elements or median of three values. In the following program we are going to write 3 functions. The first function is to find pivot element and its correct position. In second function we divide the list based on pivot element and sort the sub list and third function (main fun) is to print input and output. # Write a python program to arrange the elements in ascending order using Quick sort (with functions) #To get the correct position of pivot element: def pivot_place(list1,first,last): pivot=list1[first] left=first+1 right=last while True: while left<=right and list1[left]<=pivot: left=left+1 while left<=right and list1[right]>=pivot: right=right-1 if right<left: break else: list1[left],list1[right]=list1[right],list1[left] 76 list1[first],list1[right]=list1[right],list1[first] return right #second function def quicksort(list1,first,last): if first<last: p=pivot_place(list1,first,last) quicksort(list1,first,p-1) quicksort(list1,p+1,last) #main fun list1=[56,25,93,15,31,44] n=len(list1) quicksort(list1,0,n-1) print(list1) Output: C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/qucksort.py [15, 25, 31, 44, 56, 93] # Write a python program to arrange the elements in descending order using Quick sort (with functions) #To get the correct position of pivot element: def pivot_place(list1,first,last): pivot=list1[first] left=first+1 right=last while True: while left<=right and list1[left]>=pivot: left=left+1 while left<=right and list1[right]<=pivot: right=right-1 if right<left: break else: list1[left],list1[right]=list1[right],list1[left] list1[first],list1[right]=list1[right],list1[first] return right def quicksort(list1,first,last): if first<last: p=pivot_place(list1,first,last) quicksort(list1,first,p-1) quicksort(list1,p+1,last) #main fun list1=[56,25,93,15,31,44] n=len(list1) quicksort(list1,0,n-1) print(list1) Output: 77 C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/qukdesc.py [93, 56, 44, 31, 25, 15] UNIT - IV Linked Lists - Singly Linked Lists, Doubly Linked Lists, Circular Linked Lists. Stacks - Overview of Stack, Implementation of Stack (List & Linked list), Applications of Stack Queues: Overview of Queue, Implementation of Queue(List & Linked list), Applications of Queues, Priority Queues. Linked Lists: Linked lists are one of the most commonly used data structures in any programming language.Linked Lists, on the other hand, are different. Linked lists, do not store data at contiguous memory locations. For each item in the memory location, linked list stores value of the item and the reference or pointer to the next item. One pair of the linked list item and the reference to next item constitutes a node. The following are different types of linked lists. • Single Linked List A single linked list is the simplest of all the variants of linked lists. Every node in a single linked list contains an item and reference to the next item and that's it. • • • Doubly Linked List Circular Linked List Linked List with Header # Python program to create a linked list and display its elements. The program creates a linked list using data items input from the user and displays it. Solution: 1. Create a class Node with instance variables data and next. 2. Create a class Linked List with instance variables head and last_node. 3. The variable head points to the first element in the linked list while last_node points to the last. 4. Define methods append and display inside the class Linked List to append data and display the linked list respectively. 5. Create an instance of Linked List, append data to it and display the list. Program: class Node: def__init__(self, data): 78 self.data= data self.next=None class LinkedList: def__init__(self): self.head=None self.last_node=None def append(self, data): ifself.last_nodeisNone: self.head= Node(data) self.last_node=self.head else: self.last_node.next= Node(data) self.last_node=self.last_node.next def display(self): current =self.head while current isnotNone: print(current.data, end =' ') current = current.next a_llist = LinkedList() n =int(input('How many elements would you like to add? ')) for i inrange(n): data =int(input('Enter data item: ')) a_llist.append(data) print('The linked list: ', end ='') a_llist.display() Program Explanation 1. An instance of Linked List is created. 2. The user is asked for the number of elements they would like to add. This is stored in n. 3. Using a loop, data from the user is appended to the linked list n times. 4. The linked list is displayed. Output: C:/Users/MRCET/AppData/Local/Programs/Python/Python38-32/pyyy/link1.py How many elements would you like to add? 5 Enter data item: 4 Enter data item: 4 Enter data item: 6 Enter data item: 8 79 Enter data item: 9 The linked list: 4 4 6 8 9 Doubly Linked List A doubly linked list is a linked data structure that consists of a set of sequentially linked records called nodes. Each node contains three fields: two link fields (references to the previous and to the next node in the sequence of nodes) and one data field. Advantages of Doubly Linked List 1. Traversal can be done on either side means both in forward as well as backward. 2. Deletion Operation is more efficient if the pointer to delete node is given. Disadvantages of Linked List 1. Since it requires extra pointer that is the previous pointer to store previous node reference. 2. After each operation like insertion-deletion, it requires an extra pointer that is a previous pointer which needs to be maintained. So, a typical node in the doubly linked list consists of three fields: o Data represents the data value stored in the node. o Previous represents a pointer that points to the previous node. o Next represents a pointer that points to the next node in the list. The above picture represents a doubly linked list in which each node has two pointers to point to previous and next node respectively. Here, node 1 represents the head of the list. The previous pointer of the head node will always point to NULL. Next pointer of node one will point to node 2. Node 5 represents the tail of the list whose previous pointer will point to node 4, and the next will point to NULL. 80 ALGORITHM: 1. Define a Node class which represents a node in the list. It will have three properties: data, previous which will point to the previous node and next which will point to the next node. 2. Define another class for creating a doubly linked list, and it has two nodes: head and tail. Initially, head and tail will point to null. 3. addNode() will add node to the list: o It first checks whether the head is null, then it will insert the node as the head. o Both head and tail will point to a newly added node. o Head's previous pointer will point to null and tail's next pointer will point to null. o If the head is not null, the new node will be inserted at the end of the list such that new node's previous pointer will point to tail. o The new node will become the new tail. Tail's next pointer will point to null. a. display() will show all the nodes present in the list. o Define a new node 'current' that will point to the head. o Print current.data till current points to null. o Current will point to the next node in the list in each iteration. PROGRAM: 1. #Represent a node of doubly linked list 2. class Node: 3. def __init__(self,data): 4. self.data = data; 5. self.previous = None; 6. self.next = None; 7. 8. class DoublyLinkedList: 9. #Represent the head and tail of the doubly linked list 10. def __init__(self): 11. self.head = None; 81 12. self.tail = None; 13. 14. #addNode() will add a node to the list 15. def addNode(self, data): 16. #Create a new node 17. newNode = Node(data); 18. 19. #If list is empty 20. if(self.head == None): 21. #Both head and tail will point to newNode 22. self.head = self.tail = newNode; 23. #head's previous will point to None 24. self.head.previous = None; 25. #tail's next will point to None, as it is the last node of the list 26. self.tail.next = None; 27. else: 28. #newNode will be added after tail such that tail's next will point to newNode 29. self.tail.next = newNode; 30. #newNode's previous will point to tail 31. newNode.previous = self.tail; 32. #newNode will become new tail 33. self.tail = newNode; 34. #As it is last node, tail's next will point to None 35. self.tail.next = None; 36. 37. #display() will print out the nodes of the list 38. def display(self): 39. #Node current will point to head 40. current = self.head; 41. if(self.head == None): 42. print("List is empty"); 43. return; 44. print("Nodes of doubly linked list: "); 82 45. while(current != None): 46. #Prints each node by incrementing pointer. 47. print(current.data),; 48. current = current.next; 49. 50. dList = DoublyLinkedList(); 51. #Add nodes to the list 52. dList.addNode(1); 53. dList.addNode(2); 54. dList.addNode(3); 55. dList.addNode(4); 56. dList.addNode(5); 57. 58. #Displays the nodes present in the list 59. dList.display(); Output: Nodes of doubly linked list: 12345 Circular Linked List The circular linked list is a kind of linked list. First thing first, the node is an element of the list, and it has two parts that are, data and next. Data represents the data stored in the node, and next is the pointer that will point to the next node. Head will point to the first element of the list, and tail will point to the last element in the list. In the simple linked list, all the nodes will point to their next element and tail will point to null. The circular linked list is the collection of nodes in which tail node also point back to head node. The diagram shown below depicts a circular linked list. Node A represents head and node D represents tail. So, in this list, A is pointing to B, B is pointing to C and C is pointing to D but what makes it circular is that node D is pointing back to node A. 83 ALGORITHM: 1. Define a Node class which represents a node in the list. It has two properties data and next which will point to the next node. 2. Define another class for creating the circular linked list, and it has two nodes: head and tail. It has two methods: add() and display() . 3. add() will add the node to the list: o It first checks whether the head is null, then it will insert the node as the head. o Both head and tail will point to the newly added node. o If the head is not null, the new node will be the new tail, and the new tail will point to the head as it is a circular linked list. a. display() will show all the nodes present in the list. o Define a new node 'current' that will point to the head. o Print current.data till current will points to head o Current will point to the next node in the list in each iteration. PROGRAM: 1. #Represents the node of list. 2. class Node: 3. def __init__(self,data): 4. self.data = data; 5. self.next = None; 6. 84 7. class CreateList: 8. #Declaring head and tail pointer as null. 9. def __init__(self): 10. self.head = Node(None); 11. self.tail = Node(None); 12. self.head.next = self.tail; 13. self.tail.next = self.head; 14. 15. #This function will add the new node at the end of the list. 16. def add(self,data): 17. newNode = Node(data); 18. #Checks if the list is empty. 19. if self.head.data is None: 20. #If list is empty, both head and tail would point to new node. 21. self.head = newNode; 22. self.tail = newNode; 23. newNode.next = self.head; 24. else: 25. #tail will point to new node. 26. self.tail.next = newNode; 27. #New node will become new tail. 28. self.tail = newNode; 29. #Since, it is circular linked list tail will point to head. 30. self.tail.next = self.head; 31. 32. #Displays all the nodes in the list 33. def display(self): 34. current = self.head; 35. if self.head is None: 36. print("List is empty"); 37. return; 38. else: 39. print("Nodes of the circular linked list: "); 85 40. #Prints each node by incrementing pointer. 41. print(current.data), 42. while(current.next != self.head): 43. current = current.next; 44. print(current.data), 45. 46. 47. class CircularLinkedList: 48. cl = CreateList(); 49. #Adds data to the list 50. cl.add(1); 51. cl.add(2); 52. cl.add(3); 53. cl.add(4); 54. #Displays all the nodes present in the list 55. cl.display(); Output: Nodes of the circular linked list: 1234 Stacks: Stack works on the principle of “Last-in, first-out”. Also, the inbuilt functions in Python make the code short and simple. To add an item to the top of the list, i.e., to push an item, we use append() function and to pop out an element we use pop() function. #Python code to demonstrate Implementing stack using list stack = ["Amar", "Akbar", "Anthony"] stack.append("Ram") stack.append("Iqbal") print(stack) print(stack.pop()) print(stack) print(stack.pop()) print(stack) 86 Output: ['Amar', 'Akbar', 'Anthony', 'Ram', 'Iqbal'] Iqbal ['Amar', 'Akbar', 'Anthony', 'Ram'] Ram ['Amar', 'Akbar', 'Anthony'] Stack Using Linked List A stack using a linked list is just a simple linked list with just restrictions that any element will be added and removed using push and pop respectively. In addition to that, we also keep top pointer to represent the top of the stack. This is described in the picture given below. Stack Operations: 1. push() : Insert the element into linked list nothing but which is the top node of Stack. 2. pop() : Return top element from the Stack and move the top pointer to the second node of linked list or Stack. 3. peek(): Return the top element. 4. display(): Print all element of Stack. A stack will be empty if the linked list won’t have any node i.e., when the top pointer of the linked list will be null. So, let’s start by making a function to check whether a stack is empty or not. IS_EMPTY(S) if S.top == null return TRUE return FALSE Now, to push any node to the stack (S) - PUSH(S, n), we will first check if the stack is empty or not. If the stack is empty, we will make the new node head of the linked list and also point the top pointer to it. 87 PUSH(S, n) if IS_EMPTY(S) //stack is empty S.head = n //new node is the head of the linked list S.top = n //new node is the also the top If the stack is not empty, we will add the new node at the last of the stack. For that, we will point next of the top to the new node - (S.top.next = n) and the make the new node top of the stack - (S.top = n). PUSH(S, n) if IS_EMPTY(S) //stack is empty ... else S.top.next = n S.top = n PUSH(S, n) if IS_EMPTY(S) //stack is empty S.head = n //new node is the head of the linked list S.top = n //new node is the also the top else S.top.next = n 88 S.top = n Similarly, to remove a node (pop), we will first check if the stack is empty or not as we did in the implementation with array. POP(S) if IS_EMPTY(S) Error “Stack Underflow” In the case when the stack is not empty, we will first store the value in top node in a temporary variable because we need to return it after deleting the node. POP(S) if IS_EMPTY(S) ... else x = S.top.data Now if the stack has only one node (top and head are same), we will just make both top and head null. POP(S) if IS_EMPTY(S) ... else ... if S.top == S.head //only one node S.top = NULL S.head = NULL If the stack has more than one node, we will move to the node previous to the top node and make the next of point it to null and also point the top to it. POP(S) ... ... if S.top == S.head //only one node ... 89 else tmp = S.head while tmp.next != S.top //iterating to the node previous to top tmp = tmp.next tmp.next = NULL //making the next of the node null S.top = tmp //changing the top pointer We first iterated to the node previous to the top node and then we marked its next to null - tmp.next = NULL. After this, we pointed the top pointer to it - S.top = tmp. At last, we will return the data stored in the temporary variable - return x. POP(S) if IS_EMPTY(S) Error "Stack Underflow" else x = S.top.data if S.top == S.head //only one node S.top = NULL S.head = NULL else tmp = S.head while tmp.next != S.top //iterating to the node previous to top tmp = tmp.next tmp.next = NULL //making the next of the node null S.top = tmp //changing the top pointer return x Stack using linked list 90 class Node(): def __init__(self, data): self.data = data self.next = None class Stack(): def __init__(self): self.head = None self.top = None def traversal(s): temp = s.head #temporary pointer to point to head a = '' while temp != None: #iterating over stack a = a+str(temp.data)+'\t' temp = temp.next print(a) def is_empty(s): if s.top == None: return True return False 91 def push(s, n): if is_empty(s): #empty s.head = n s.top = n else: s.top.next = n s.top = n def pop(s): if is_empty(s): print("Stack Underflow") return -1000 else: x = s.top.data if s.top == s.head: # only one node s.top = None s.head = None else: temp = s.head while temp.next != s.top: #iterating to the last element temp = temp.next temp.next = None del s.top s.top = temp return x 92 if __name__ == '__main__': s = Stack() a = Node(10) b = Node(20) c = Node(30) pop(s) push(s, a) push(s, b) push(s, c) traversal(s) pop(s) traversal(s) Applications of Stack There are many applications of a stack. Some of them are: • • • • Stacks are used in backtracking algorithms. They are also used to implement undo/redo functionality in a software. Stacks are also used in syntax parsing for many compilers. Stacks are also used to check proper opening and closing of parenthesis. Queue Similar to stacks, a queue is also an Abstract Data Type or ADT. A queue follows FIFO (First-in, First out) policy. It is equivalent to the queues in our general life. For example, a new person enters a queue at the last and the person who is at the front (who must have entered the queue at first) will be served first. 93 Similar to a queue of day to day life, in Computer Science also, a new element enters a queue at the last (tail of the queue) and removal of an element occurs from the front (head of the queue). Similar to the stack, we will implement the queue using a linked list as well as with an array. But let’s first discuss the operations which are done on a queue. Enqueue → Enqueue is an operation which adds an element to the queue. As stated earlier, any new item enters at the tail of the queue, so Enqueue adds an item to the tail of a queue. Dequeue → It is similar to the pop operation of stack i.e., it returns and deletes the front element from the queue. isEmpty → It is used to check whether the queue has any element or not. isFull → It is used to check whether the queue is full or not. Front → It is similar to the top operation of a stack i.e., it returns the front element of the queue (but don’t delete it). Before moving forward to code up these operations, let’s discuss the applications of a queue. 94 Applications of Queue Queues are used in a lot of applications, few of them are: • • • Queue is used to implement many algorithms like Breadth First Search (BFS), etc. It can be also used by an operating system when it has to schedule jobs with equal priority Customers calling a call center are kept in queues when they wait for someone to pick up the calls Queue Using an Array We will maintain two pointers - tail and head to represent a queue. head will always point to the oldest element which was added and tail will point where the new element is going to be added. To insert any element, we add that element at tail and increase the tail by one to point to the next element of the array. Suppose tail is at the last element of the queue and there are empty blocks before head as shown in the picture given below. In this case, our tail will point to the first element of the array and will follow a circular order. 95 Initially, the queue will be empty i.e., both head and tail will point to the same location i.e., at index 1. We can easily check if a queue is empty or not by checking if head and tail are pointing to the same location or not at any time. IS_EMPTY(Q) If Q.tail == Q.head return True return False Similarly, we will say that if the head of a queue is 1 more than the tail, the queue is full. IS_FULL(Q) if Q.head = Q.tail+1 return True Return False Now, we have to deal with the enqueue and the dequeue operations. To enqueue any item to the queue, we will first check if the queue is full or not i.e., Enqueue(Q, x) if isFull(Q) Error “Queue Overflow” 96 else … If the queue is not full, we will add the element to the tail i.e, Q[Q.tail] = x. While adding the element, it might be possible that we have added the element at the last of the array and in this case, the tail will go to the first element of the array. Otherwise, we will just increase the tail by 1. Enqueue(Q, x) if isFull(Q) Error “Queue Overflow” else Q[Q.tail] = x if Q.tail == Q.size Q.tail = 1 else 97 Q.tail = Q.tail+1 To dequeue, we will first check if the queue is empty or not. If the queue is empty, then we will throw an error. Dequeue(Q, x) if isEmpty(Q) Error “Queue Underflow” else … To dequeue, we will first store the item which we are going to delete from the queue in a variable because we will be returning it at last. Dequeue(Q, x) if isEmpty(Q) Error “Queue Underflow” else x = Q[Q.head] … Now, we just have to increase the head pointer by 1. And in the case when the head is at the last element of the array, it will go 1. Dequeue(Q, x) if isEmpty(Q) Error “Queue Underflow” else x = Q[Q.head] if Q.head == Q.size Q.head = 1 98 else Q.head = Q.head+1 return x class Queue: def __init__(self, size): self.head = 1 self.tail = 1 self.Q = [0]*(size) self.size = size def is_empty(self): if self.tail == self.head: return True return False def is_full(self): if self.head == self.tail+1: return True return False def enqueue(self, x): if self.is_full(): print("Queue Overflow") else: 99 self.Q[self.tail] = x if self.tail == self.size: self.tail = 1 else: self.tail = self.tail+1 def dequeue(self): if self.is_empty(): print("Underflow") else: x = self.Q[self.head] if self.head == self.size: self.head = 1 else: self.head = self.head+1 return x def display(self): i = self.head while(i < self.tail): print(self.Q[i]) if(i == self.size): i=0 i = i+1 100 if __name__ == '__main__': q = Queue(10) q.enqueue(10) q.enqueue(20) q.enqueue(30) q.enqueue(40) q.enqueue(50) q.display() print("") q.dequeue() q.dequeue() q.display() print("") q.enqueue(60) q.enqueue(70) q.display() Queue Using Linked List As we know that a linked list is a dynamic data structure and we can change the size of it whenever it is needed. So, we are not going to consider that there is a maximum size of the queue and thus the queue will never overflow. However, one can set a maximum size to restrict the linked list from growing more than that size. As told earlier, we are going to maintain a head and a tail pointer to the queue. In the case of an empty queue, head will point to NULL. 101 IS_EMPTY(Q) if Q.head == null return True return False We will point the head pointer to the first element of the linked list and the tail pointer to the last element of it as shown in the picture given below. The enqueue operation simply adds a new element to the last of a linked list. However, if the queue is empty, we will simply make the new node head and tail of the queue. 102 ENQUEUE(Q, n) if IS_EMPTY(Q) Q.head = n Q.tail = n else Q.tail.next = n Q.tail = n To dequeue, we need to remove the head of the linked list. To do so, we will first store its data in a variable because we will return it at last and then point head to its next element. x = Q.head.data Q.head = Q.head.next return x We will execute the above codes when the queue is not empty. If it is, we will throw the "Queue Underflow" error. DEQUEUE(Q, n) if IS_EMPTY(Q) Error "Queue Underflow" else x = Q.head.data Q.head = Q.head.next return x Queue using Linked List class Node(): def __init__(self, data): self.data = data 103 self.next = None class Queue(): def __init__(self): self.head = None self.tail = None def traversal(q): temp = q.head #temporary pointer to point to head a = '' while temp != None: #iterating over queue a = a+str(temp.data)+'\t' temp = temp.next print(a) def is_empty(q): if q.head == None: return True return False def enqueue(q, n): if is_empty(q): #empty q.head = n 104 q.tail = n else: q.tail.next = n q.tail = n def dequeue(q): if is_empty(q): print("Queue Underflow") return -1000 else: x = q.head.data temp = q.head q.head = q.head.next del temp return x if __name__ == '__main__': q = Queue() a = Node(10) b = Node(20) c = Node(30) dequeue(q) enqueue(q, a) 105 enqueue(q, b) enqueue(q, c) traversal(q) dequeue(q) traversal(q) Priority QueuesA queue has FIFO (first-in-first-out) ordering where items are taken out or accessed on a first-come-first-served basis. Examples of queues include a queue at a movie ticket stand, as shown in the illustration above. But, what is a priority queue? A priority queue is an abstract data structure (a data structure defined by its behaviour) that is like a normal queue but where each item has a special “key” to quantify its “priority”. For example, if the movie cinema decides to serve loyal customers first, it will order them by their loyalty (points or number of tickets purchased). In such a case, the queue for tickets will no longer be first-come-first-served, but most-loyalfirst-served. The customers will be the “items” of this priority queue while the “priority” or “key” will be their loyalty. A priority queue can be of two types: 1. Max Priority Queue: Which arranges the data as per descending order of their priority. 2. Min Priority Queue: Which arranges the data as per ascending order of their priority. In a priority queue, following factors come into play: 1. In priority queue, data when inserted, is stored based on its priority. 106 2. When we remove a data from a priority queue(min), the data at the top, which will be the data with least priority, will get removed. 3. But, this way priority queue will not be following the basic priniciple of a queue, First in First Out(FIFO). Yes, it won't! Because a priority queue is an advanced queue used when we have to arrange and manipulate data as per the given priority. Implementing Priority Queue So now we will design our very own minimum priority queue using python list and object oriented concept. Below are the algorithm steps: 1. Node: The Node class will be the element inserted in the priority queue. You can modify the Node class as per your requirements. 2. insert: To add a new data element(Node) in the priority queue. o If the priority queue is empty, we will insert the element to it. o If the priority queue is not empty, we will traverse the queue, comparing the priorities of the existing nodes with the new node, and we will add the new node before the node with priority greater than the new node. o If the new node has the highest priority, then we will add the new node at the end. 3. delete: To remove the element with least priority. 4. size: To check the size of the priority queue, in other words count the number of elements in the queue and return it. 5. show: To print all the priority queue elements. Priority Queue Program # class for Node with data and priority class Node: def __init__(self, info, priority): 107 self.info = info self.priority = priority # class for Priority queue class PriorityQueue: def __init__(self): self.queue = list() # if you want you can set a maximum size for the queue def insert(self, node): # if queue is empty if self.size() == 0: # add the new node self.queue.append(node) else: # traverse the queue to find the right place for new node for x in range(0, self.size()): 108 # if the priority of new node is greater if node.priority >= self.queue[x].priority: # if we have traversed the complete queue if x == (self.size()-1): # add new node at the end self.queue.insert(x+1, node) else: continue else: self.queue.insert(x, node) return True def delete(self): # remove the first node from the queue return self.queue.pop(0) def show(self): for x in self.queue: 109 print str(x.info)+" - "+str(x.priority) def size(self): return len(self.queue) pQueue = PriorityQueue() node1 = Node("C", 3) node2 = Node("B", 2) node3 = Node("A", 1) node4 = Node("Z", 26) node5 = Node("Y", 25) node6 = Node("L", 12) pQueue.insert(node1) pQueue.insert(node2) pQueue.insert(node3) pQueue.insert(node4) pQueue.insert(node5) 110 pQueue.insert(node6) pQueue.show() print("--------") pQueue.delete() pQueue.show() UNIT -V Graphs - Introduction, Directed vs Undirected Graphs, Weighted vs Unweighted Graphs, Representations, Applications: Breadth First Search, Depth First Search. Trees - Overview of Trees, Tree Terminology, Binary Trees: Introduction, Applications, Implementation. Tree Traversals, Binary Search Trees: Introduction, Implementation, AVL Trees: Introduction, Rotations, 111 Implementation. Graphs – Introduction: A graph is an advanced data structure that is used to organize items in an interconnected network. Each item in a graph is known as a node(or vertex) and these nodes are connected by edges. In the figure below, we have a simple graph where there are five nodes in total and six edges. The nodes in any graph can be referred to as entities and the edges that connect different nodes define the relationships between these entities. In the above graph we have a set of nodes {V} = {A, B, C, D, E} and a set of edges, {E} = {A-B, A-D, B-C, B-D, C-D, D-E} respectively Types of Graphs Let's cover various different types of graphs. 1. Null Graphs A graph is said to be null if there are no edges in that graph. A pictorial representation of the null graph is given below: 2. Undirected Graphs If we take a look at the pictorial representation that we had in the Real-world example above, we can clearly see that different nodes are connected by a link (i.e. edge) and that edge doesn't have any kind of direction 112 associated with it. For example, the edge between "Anuj" and "Deepak" is bi-directional and hence the relationship between them is two ways, which turns out to be that "Anuj" knows "Deepak" and "Deepak" also knows about "Anuj". These types of graphs where the relation is bi-directional or there is not a single direction, are known as Undirected graphs. A pictorial representation of another undirected graph is given below: 3. Directed Graphs What if the relation between the two people is something like, "Shreya" know "Abhishek" but "Abhishek" doesn't know "Shreya". This type of relationship is one-way, and it does include a direction. The edges with arrows basically denote the direction of the relationship and such graphs are known as directed graphs. A pictorial representation of the graph is given below: It can also be noted that the edge from "Shreya" to "Abhishek" is an outgoing edge for "Shreya" and an incoming edge for "Abhishek". 4. Cyclic Graph A graph that contains at least one node that traverses back to itself is known as a cyclic graph. In simple words, a graph should have at least one cycle formation for it to be called a cyclic graph. A pictorial representation of a cyclic graph is given below: 113 It can be easily seen that there exists a cycle between the nodes (a, b, c) and hence it is a cyclic graph. 5. Acyclic Graph A graph where there's no way we can start from one node and can traverse back to the same one, or simply doesn't have a single cycle is known as an acyclic graph. A pictorial representation of an acyclic graph is given below: 6. Weighted Graph When the edge in a graph has some weight associated with it, we call that graph as a weighted graph. The weight is generally a number that could mean anything, totally dependent on the relationship between the nodes of that graph. A pictorial representation of the weighted graph is given below: 114 It can also be noted that if any graph doesn't have any weight associated with it, we simply call it an unweighted graph. 7. Connected Graph A graph where we have a path between every two nodes of the graph is known as a connected graph. A path here means that we are able to traverse from a node "A" to say any node "B". In simple terms, we can say that if we start from one node of the graph we will always be able to traverse to all the other nodes of the graph from that node, hence the connectivity. A pictorial representation of the connected graph is given below: 8. Disconnected Graph A graph that is not connected is simply known as a disconnected graph. In a disconnected graph, we will not be able to find a path from between every two nodes of the graph. A pictorial representation of the disconnected graph is given below: 9. Complete Graph A graph is said to be a complete graph if there exists an edge for every pair of vertices(nodes) of that graph. A pictorial representation of the complete graph is given below: 115 10. Multigraph A graph is said to be a multigraph if there exist two or more than two edges between any pair of nodes in the graph. A pictorial representation of the multigraph is given below: Commonly Used Graph Terminologies • • • • • • • • Path - A sequence of alternating nodes and edges such that each of the successive nodes are connected by the edge. Cycle - A path where the starting and the ending node is the same. Simple Path - A path where we do not encounter a vertex again. Bridge - An edge whose removal will simply make the graph disconnected. Forest - A forest is a graph without cycles. Tree - A connected graph that doesn't have any cycles. Degree - The degree in a graph is the number of edges that are incident on a particular node. Neighbour - We say vertex "A" and "B" are neighbours if there exists an edge between them. Weighted vs UnWeighted Graph To understand difference between weighted vs unweighted graph, first we need to understand what weight represent ? A weight is a numerical value attached to each individual edge in the graph. Weighted Graph will contains weight on each edge where as unweighted does not. Following is an example, where both graphs looks exactly the same but one is weighted another is not. 116 What difference does it make ? Let’s take the same example to find shortest path from A to F. Result is different, just because one is weighted another doesn’t. But how? Because when you doesn’t have weight, all edges are considered equal. Shortest distance means less number of nodes you travel. But in case of weighted graph, calculation happens on the sum of weights of the travelled edges. Breadth First Search: BFS is an algorithm that is used to graph data or searching tree or traversing structures. The algorithm efficiently visits and marks all the key nodes in a graph in an accurate breadthwise fashion. This algorithm selects a single node (initial or source point) in a graph and then visits all the nodes adjacent to the selected node. Once the algorithm visits and marks the starting node, then it moves towards the nearest unvisited nodes and analyses them. Once visited, all nodes are marked. These iterations continue until all the nodes of the graph have been successfully visited and marked. The full form of BFS is the Breadth-first search. Steps in Breadth First Search 117 1. Start by putting any one of the graph’s vertices at the back of a queue. 2. Take the front item of the queue and add it to the visited list. 3. Create a list of that vertex’s adjacent nodes. Add the ones which aren’t in the visited list to the back of the queue. 4. Keep repeating steps 2 and 3 until the queue is empty. Example of BFS In the following example of DFS, we have used graph having 6 vertices. Example of BFS Step 1) 5. You have a graph of seven numbers ranging from 0 – 6. Step 2) 118 6. 0 or zero has been marked as a root node. Step 3) 7. 0 is visited, marked, and inserted into the queue data structure. Step 4) 8. Remaining 0 adjacent and unvisited nodes are visited, marked, and inserted into the queue. Step 5) 119 9. Traversing iterations are repeated until all nodes are visited. Depth First Search. DFS is an algorithm for finding or traversing graphs or trees in depth-ward direction. The execution of the algorithm begins at the root node and explores each branch before backtracking. It uses a stack data structure to remember, to get the subsequent vertex, and to start a search, whenever a dead-end appears in any iteration. The full form of DFS is Depth-first search. Steps in Depth First Search 1. Start by putting any one of the graph’s vertices on top of a stack. 2. Take the top item of the stack and add it to the visited list. 3. Create a list of that vertex’s adjacent nodes. Add the ones which aren’t in the visited list to the top of the stack. 4. Keep repeating steps 2 and 3 until the stack is empty. Example of DFS In the following example of DFS, we have used an undirected graph having 5 vertices. 120 Step 1) We have started from vertex 0. The algorithm begins by putting it in the visited list and simultaneously putting all its adjacent vertices in the data structure called stack. Step 2) You will visit the element, which is at the top of the stack, for example, 1 and go to its adjacent nodes. It is because 0 has already been visited. Therefore, we visit vertex 2. Step 3) 121 Vertex 2 has an unvisited nearby vertex in 4. Therefore, we add that in the stack and visit it. Step 4) Finally, we will visit the last vertex 3, it doesn't have any unvisited adjoining nodes. We have completed the traversal of the graph using DFS algorithm. Tree Data Structures Trees are non-linear data structures that represent nodes connected by edges. Each tree consists of a root node as the Parent node, and the left node and right node as Child nodes. 122 • • • • • • • • • • Root → The topmost node of the hierarchy is called the root of the tree. Child → Nodes next in the hierarchy are the children of the previous node. Parent → The node just previous to the current node is the parent of the current node. Siblings → Nodes with the same parent are called siblings. Ancestors → Nodes which are higher in the hierarchy are ancestors of a given node. Descendents → Nodes which are lower in the hierarchy are descendants of a given node. Internal Nodes → Nodes with at least one child are internal nodes. External Nodes/Leaves → Nodes which don't have any child are called leaves of a tree. Edge → The link between two nodes is called an edge. Level → The root of a tree is at level 0 and the nodes whose parent is root are at level 1 and so on. 123 • Height → The height of a node is the number of nodes (excluding the node) on the longest path from the node to a leaf. • • Height of Tree → Height of a tree is the height of its root. Depth → The depth of a node is the number of nodes (excluding the node) on the path from the root to the node. • Node Degree → It is the maximum number of children a node has. • Tree Degree → Tree degree is the maximum of the node degrees. So, the tree degree in the above picture is 3. Till now, we have an idea of what a tree is and the terminologies we use with a tree. But we don't know yet what the specific properties of a tree are or which structure should qualify as a tree. So, let's see the properties of a tree. Properties of a Tree 124 A tree must have some properties so that we can differentiate from other data structures. So, let's look at the properties of a tree. The numbers of nodes in a tree must be a finite and nonempty set. There must exist a path to every node of a tree i.e., every node must be connected to some other node. There must not be any cycles in the tree. It means that the number of edges is one less than the number of nodes. Binary Trees A binary tree is a tree in which every node has at most two children. As you can see in the picture given above, a node can have less than 2 children but not more than that. We can also classify a binary tree into different categories. Let's have a look at them: Full Binary Tree → A binary tree in which every node has 2 children except the leaves is known as a full binary tree. 125 Complete Binary Tree → A binary tree which is completely filled with a possible exception at the bottom level i.e., the last level may not be completely filled and the bottom level is filled from left to right. Let's look at this picture to understand the difference between a full and a complete binary tree. A complete binary tree also holds some important properties. So, let's look at them. • The parent of node i is ⌊i2⌋⌊i2⌋. For example, the parent of node 4 is 2 and the parent of node 5 is also 2. • The left child of node i is 2i2i. • The right child of node i is 2i+12i+1 126 Perfect Binary Tree → In a perfect binary tree, each leaf is at the same level and all the interior nodes have two children. Thus, a perfect binary tree will have the maximum number of nodes for all alternative binary trees of the same height and it will be 2h+1−12h+1−1 which we are going to prove next. Maximum Number of Nodes in a Binary Tree We know that the maximum number of nodes will be in a perfect binary tree. So, let's assume that the height of a perfect binary tree is hh. 127 Number of nodes at level 0 = 20=120=1 Number of nodes at level 1 = 21=221=2 Similarly, the number of nodes at level h = 2h2h Thus, the total number of nodes in the tree = 20+21+...+2h20+21+...+2h The above sequence is a G.P. with common ratio 2 and first term 1 and total number of terms are h+1. So, the value of the summation will be 2h+1−12−1=2h+1−12h+1−12−1=2h+1−1. Thus, the total number of nodes in a perfect binary tree = 2h+1−12h+1−1. Height of a Perfect Binary Tree We know that the number of nodes (n) for height (h) of a perfect binary tree = 2h+1−12h+1−1. =>n=2∗2h−1=>n=2∗2h−1 or,2h=n+12or,2h=n+12 h=lgn+12h=lg n+12 Thus, the height of a perfect binary tree with n nodes = lgn+12lg n+12. We know that the number of nodes at level i in a perfect binary tree = 2i2i. Thus, the number of leaves (nodes at level h) = 2h2h. Thus, the total number of non-leaf nodes = 2h+1−1−2h=2h−12h+1−1−2h=2h−1 i.e., number of leaf nodes - 1. Thus, the maximum number of nodes will be in a perfect binary tree and the minimum number of nodes will be in a tree in which nodes are linked just like a linked list. Array Representation of Binary Tree 128 In the previous chapter, we have already seen to make a node of a tree. We can easily use those nodes to make a linked representation of a binary tree. For now, let's discuss the array representation of a binary tree. We start by numbering the nodes of the tree from 1 to n(number of nodes). As you can see, we have numbered from top to bottom and left to right for the same level. Now, these numbers represent the indices of an array (starting from 1) as shown in the picture given below. We can also get the parent, the right child and the left child using the properties of a complete binary tree we have discussed above i.e., for a node i, the parent is ⌊i2⌋⌊i2⌋, the left child is 2i2i and the right child is 2i+12i+1. 129 So, we represented a complete binary tree using an array and saw how to get the parent and children of any node. Let's discuss about doing the same for an incomplete binary tree. Array Representation of Incomplete Binary Tree To represent an incomplete binary tree with an array, we first assume that all the nodes are present to make it a complete binary tree and then number the nodes as shown in the picture given below. Now according to these numbering, we fill up the array. Coding a Binary Tree For the linked representation, we can easily access the parent, right child and the left child with T.parent, T.right and T.left respectively. So, we will first write explain the codes for the array representation and then write the full codes in C, Java and Python for both array and linked representation. Let's start by writing the code to get the right child of a node. We will pass the index and the tree to the function - RIGHT_CHILD(index). 130 After this, we will check if there is a node at the index or not (if (T[index] != null)) and also if the index of the right child (2∗index+12∗index+1) lies in the size of the tree or not i.e., if (T[index] != null and (2*index + 1) <= T.size). If the above condition is true, we will return the index of the right child i.e., return (2*index + 1). RIGHT_CHILD(index) if (T[index] != null and (2*index + 1) <= T.size) return (2*index + 1) else return null Similarly, we can get the left child. LEFT_CHILD(index) if (T[index] != null and (2*index) <= T.size) return (2*index) else return null Similarly, we can also write the code to get the parent. PARENT(index) if (T[index] != null and (floor(index/2)) =! null) return floor(index/2) else return null 131 Code Using Array ''' D / \ / \ / \ A F / \ / \ / \ / \ E B R T / \ / / \ G Q V J L ''' complete_node=15 tree=[None,'D','A','F','E','B','R','T','G','Q',None,None,'V',None,'J','L'] defget_right_child(index): # node is not null # and the result must lie within the number of nodes for a complete binary tree iftree[index]!=Noneand((2*index)+1)<=complete_node: return(2*index)+1 # right child doesn't exist 132 return-1 defget_left_child(index): # node is not null # and the result must lie within the number of nodes for a complete binary tree iftree[index]!=Noneand(2*index)<=complete_node: return2*index # left child doesn't exist return-1 defget_parent(index): iftree[index]!=Noneandindex/2!=None: returnindex//2 return-1 Linked List Representation of Binary Tree We use a double linked list to represent a binary tree. In a double linked list, every node consists of three fields. First field for storing left child address, second for storing actual data and third for storing right child address. In this linked list representation, a node has the following structure... 133 Code Using Linked Representation classTreeNode: def__init__(self,data): self.data=data self.right=None self.left=None self.parent=None classTree: def__init__(self,n): self.root=n if__name__=='__main__': ''' D / \ / / A \ \ F 134 / \ / / \ \ / \ E B R T / \ / / \ G Q V J L ''' d=TreeNode('D') a=TreeNode('A') f=TreeNode('F') e=TreeNode('E') b=TreeNode('B') r=TreeNode('R') t1=TreeNode('T') g=TreeNode('G') q=TreeNode('Q') v=TreeNode('V') j=TreeNode('J') l=TreeNode('L') t=Tree(d) t.root.right=f t.root.left=a ''' D / \ / \ 135 / \ A F ''' a.right=b a.left=e f.right=t1 f.left=r e.right=q e.left=g r.left=v t1.right=l t1.left=j Applications of Binary tree Binary trees are used to represent a nonlinear data structure. There are various forms of Binary trees. Binary trees play a vital role in a software application. One of the most important applications of the Binary tree is in the searching algorithm. A general tree is defined as a nonempty finite set T of elements called nodes such that: • The tree contains the root element • The remaining elements of the tree form an ordered collection of zeros and more disjoint trees T1, T2, T3, T4 …. Tn which are called subtrees. Binary Tree Traversal We are ready with a binary tree. Our next task would be to visit each node of it i.e., to traverse over the entire tree. In a linear data structure like linked list, it was a simple task, we just had to visit the next pointer of the node. But since a tree is a non-linear data structure, we follow different approaches. Generally, there are three types of traversals: • Preorder Traversal 136 • • Postorder Traversal Inorder Traversal Basically, each of these traversals gives us a sequence in which we should visit the nodes. For example, in preorder traversal we first visit the root, then the left subtree and then the right subtree. Each traversal is useful in solving some specific problems. So, we choose the method of traversal accroding to the need of the problem we are going to solve. Let's discuss each of them one by one. Preorder Traversal In preorder traversal, we first visit the root of a tree, then its left subtree and after visiting the left subtree, the right subtree. PREORDER(n) if(n != null) print(n.data) // visiting root PREORDER(n.left) // visiting left subtree PREORDER(n.right) // visiting right subtree So, we are first checking if the node is null or not - if(n != null). After this, we are visiting the root i.e., printing its data - print(n.data). Then we are visiting the left subtree - PREORDER(n.left). At last, we are visiting the right subtree - PREORDER(n.right). So, we will first visit the root as shown in the picture given below. 137 Then, we will visit the left subtree. In this left subtree, again we will visit its root and then its left subtree. 138 At last, we will visit the right subtree. 139 Postorder Traversal In postorder traversal, we first visit the left subtree, then the right subtree and at last, the root. POSTORDER(n) if(n != null) PREORDER(n.left) // visiting left subtree PREORDER(n.right) // visiting right subtree print(n.data) // visiting root We will first visit the left subtree. 140 When there is no left subtree, we will visit the right subtree. Since the right subtree is null, we will visit the root. 141 Similarly, we will visit every other node. Inorder Traversal In inorder traversal, we first visit the left subtree, then the root and lastly, the right subtree. INORDER(n) if(n != null) INORDER(n.left) print(n.data) 142 INORDER(n.right) We can also see the inorder traversal as projection of the tree on an array as shown in the picture given below. 143 Binary Search Tree In a binary tree, every node can have a maximum of two children but there is no need to maintain the order of nodes basing on their values. In a binary tree, the elements are arranged in the order they arrive at the tree from top to bottom and left to right. A binary tree has the following time complexities... 1. Search Operation - O(n) 2. Insertion Operation - O(1) 3. Deletion Operation - O(n) To enhance the performance of binary tree, we use a special type of binary tree known as Binary Search Tree. Binary search tree mainly focuses on the search operation in a binary tree. Binary search tree can be defined as follows... Binary Search Tree is a binary tree in which every node contains only smaller values in its left subtree and only larger values in its right subtree. In a binary search tree, all the nodes in the left subtree of any node contains smaller values and all the nodes in the right subtree of any node contains larger values as shown in the following figure... 144 Example The following tree is a Binary Search Tree. In this tree, left subtree of every node contains nodes with smaller values and right subtree of every node contains larger values. Every binary search tree is a binary tree but every binary tree need not to be binary search tree. Operations on a Binary Search Tree The following operations are performed on a binary search tree... 1. Search 2. Insertion 3. Deletion Search Operation in BST In a binary search tree, the search operation is performed with O(log n) time complexity. The search operation is performed as follows... 145 • Step 1 - Read the search element from the user. • Step 2 - Compare the search element with the value of root node in the tree. • Step 3 - If both are matched, then display "Given node is found!!!" and terminate the function • Step 4 - If both are not matched, then check whether search element is smaller or larger than that node value. • Step 5 - If search element is smaller, then continue the search process in left subtree. • Step 6- If search element is larger, then continue the search process in right subtree. • Step 7 - Repeat the same until we find the exact element or until the search element is compared with the leaf node • Step 8 - If we reach to the node having the value equal to the search value then display "Element is found" and terminate the function. • Step 9 - If we reach to the leaf node and if it is also not matched with the search element, then display "Element is not found" and terminate the function. Insertion Operation in BST In a binary search tree, the insertion operation is performed with O(log n) time complexity. In binary search tree, new node is always inserted as a leaf node. The insertion operation is performed as follows... • Step 1 - Create a newNode with given value and set its left and right to NULL. • Step 2 - Check whether tree is Empty. • Step 3 - If the tree is Empty, then set root to newNode. • Step 4 - If the tree is Not Empty, then check whether the value of newNode is smaller or larger than the node (here it is root node). • Step 5 - If newNode is smaller than or equal to the node then move to its left child. If newNode is larger than the node then move to its right child. • Step 6- Repeat the above steps until we reach to the leaf node (i.e., reaches to NULL). • Step 7 - After reaching the leaf node, insert the newNode as left child if the newNode is smaller or equal to that leaf node or else insert it as right child. Deletion Operation in BST 146 In a binary search tree, the deletion operation is performed with O(log n) time complexity. Deleting a node from Binary search tree includes following three cases... • Case 1: Deleting a Leaf node (A node with no children) • Case 2: Deleting a node with one child • Case 3: Deleting a node with two children Case 1: Deleting a leaf node We use the following steps to delete a leaf node from BST... • Step 1 - Find the node to be deleted using search operation • Step 2 - Delete the node using free function (If it is a leaf) and terminate the function. Case 2: Deleting a node with one child We use the following steps to delete a node with one child from BST... • Step 1 - Find the node to be deleted using search operation • Step 2 - If it has only one child then create a link between its parent node and child node. • Step 3 - Delete the node using free function and terminate the function. Case 3: Deleting a node with two children We use the following steps to delete a node with two children from BST... • Step 1 - Find the node to be deleted using search operation • Step 2 - If it has two children, then find the largest node in its left subtree (OR) the smallest node in its right subtree. • Step 3 - Swap both deleting node and node which is found in the above step. • Step 4 - Then check whether deleting node came to case 1 or case 2 or else goto step 2 • Step 5 - If it comes to case 1, then delete using case 1 logic. • Step 6- If it comes to case 2, then delete using case 2 logic. • Step 7 - Repeat the same process until the node is deleted from the tree. 147 Example Construct a Binary Search Tree by inserting the following sequence of numbers... 10,12,5,4,20,8,7,15 and 13 Above elements are inserted into a Binary Search Tree as follows... classNode: def __init__(self,data): 148 self.data=data self.right=None self.left=None self.parent=None classBinarySearchTree: def __init__(self): self.root=None defminimum(self,x): whilex.left!=None: x=x.left returnx definsert(self,n): y=None temp=self.root whiletemp!=None: y=temp ifn.data<temp.data: temp=temp.left else: temp=temp.right n.parent=y ify==None:#newly added node is root self.root=n elifn.data<y.data: y.left=n 149 else: y.right=n deftransplant(self,u,v): ifu.parent==None: self.root=v elifu==u.parent.left: u.parent.left=v else: u.parent.right=v ifv!=None: v.parent=u.parent defdelete(self,z): ifz.left==None: self.transplant(z,z.right) elifz.right==None: self.transplant(z,z.left) else: y=self.minimum(z.right)#minimum element in right subtree ify.parent!=z: self.transplant(y,y.right) y.right=z.right y.right.parent=y self.transplant(z,y) y.left=z.left 150 y.left.parent=y definorder(self,n): ifn!=None: self.inorder(n.left) print(n.data) self.inorder(n.right) if __name__ =='__main__': t=BinarySearchTree() a=Node(10) b=Node(20) c=Node(30) d=Node(100) e=Node(90) f=Node(40) g=Node(50) h=Node(60) i=Node(70) j=Node(80) k=Node(150) l=Node(110) m=Node(120) t.insert(a) t.insert(b) t.insert(c) t.insert(d) t.insert(e) 151 t.insert(f) t.insert(g) t.insert(h) t.insert(i) t.insert(j) t.insert(k) t.insert(l) t.insert(m) t.delete(a) t.delete(m) t.inorder(t.root) AVL Tree AVL tree is a height-balanced binary search tree. That means, an AVL tree is also a binary search tree but it is a balanced tree. A binary tree is said to be balanced if, the difference between the heights of left and right subtrees of every node in the tree is either -1, 0 or +1. In other words, a binary tree is said to be balanced if the height of left and right children of every node differ by either -1, 0 or +1. In an AVL tree, every node maintains an extra information known as balance factor. The AVL tree was introduced in the year 1962 by G.M. Adelson-Velsky and E.M. Landis. An AVL tree is defined as follows... An AVL tree is a balanced binary search tree. In an AVL tree, balance factor of every node is either 1, 0 or +1. Balance factor of a node is the difference between the heights of the left and right subtrees of that node. The balance factor of a node is calculated either height of left subtree - height of right subtree (OR) height of right subtree - height of left subtree. In the following explanation, we calculate as follows... 152 Balance factor = heightOfLeftSubtree - heightOfRightSubtree Example of AVL Tree The above tree is a binary search tree and every node is satisfying balance factor condition. So this tree is said to be an AVL tree. Every AVL Tree is a binary search tree but every Binary Search Tree need not be AVL tree. AVL Tree Rotations In AVL tree, after performing operations like insertion and deletion we need to check the balance factor of every node in the tree. If every node satisfies the balance factor condition then we conclude the operation otherwise we must make it balanced. Whenever the tree becomes imbalanced due to any operation we use rotation operations to make 153 the tree balanced. Rotation operations are used to make the tree balanced. Rotation is the process of moving nodes either to left or to right to make the tree balanced. There are four rotations and they are classified into two types. Single Left Rotation (LL Rotation) In LL Rotation, every node moves one position to left from the current position. To understand LL Rotation, let us consider the following insertion operation in AVL Tree... Single Right Rotation (RR Rotation) 154 In RR Rotation, every node moves one position to right from the current position. To understand RR Rotation, let us consider the following insertion operation in AVL Tree... Left Right Rotation (LR Rotation) The LR Rotation is a sequence of single left rotation followed by a single right rotation. In LR Rotation, at first, every node moves one position to the left and one position to right from the current position. To understand LR Rotation, let us consider the following insertion operation in AVL Tree... Right Left Rotation (RL Rotation) The RL Rotation is sequence of single right rotation followed by single left rotation. In RL Rotation, at first every node moves one position to right and one position to left from the current position. To understand RL Rotation, let us consider the following insertion operation in AVL Tree... 155 Operations on an AVL Tree The following operations are performed on AVL tree... 1. Search 2. Insertion 3. Deletion Search Operation in AVL Tree In an AVL tree, the search operation is performed with O(log n) time complexity. The search operation in the AVL tree is similar to the search operation in a Binary search tree. We use the following steps to search an element in AVL tree... • Step 1 - Read the search element from the user. • Step 2 - Compare the search element with the value of root node in the tree. • Step 3 - If both are matched, then display "Given node is found!!!" and terminate the function • Step 4 - If both are not matched, then check whether search element is smaller or larger than that node value. • Step 5 - If search element is smaller, then continue the search process in left subtree. 156 • Step 6 - If search element is larger, then continue the search process in right subtree. • Step 7 - Repeat the same until we find the exact element or until the search element is compared with the leaf node. • Step 8 - If we reach to the node having the value equal to the search value, then display "Element is found" and terminate the function. • Step 9 - If we reach to the leaf node and if it is also not matched with the search element, then display "Element is not found" and terminate the function. Insertion Operation in AVL Tree In an AVL tree, the insertion operation is performed with O(log n) time complexity. In AVL Tree, a new node is always inserted as a leaf node. The insertion operation is performed as follows... • Step 1 - Insert the new element into the tree using Binary Search Tree insertion logic. • Step 2 - After insertion, check the Balance Factor of every node. • Step 3 - If the Balance Factor of every node is 0 or 1 or -1 then go for next operation. • Step 4 - If the Balance Factor of any node is other than 0 or 1 or -1 then that tree is said to be imbalanced. In this case, perform suitable Rotation to make it balanced and go for next operation. Example: Construct an AVL Tree by inserting numbers from 1 to 8. 157 158 Deletion Operation in AVL Tree The deletion operation in AVL Tree is similar to deletion operation in BST. But after every deletion operation, we need to check with the Balance Factor condition. If the tree is balanced after deletion go for next operation otherwise perform suitable rotation to make the tree Balanced. 159