4
4
Sorting – Bubble sort – Selection sort – Insertion sort –Merge Sort – Quick Sort –Shell
sort —Radix sort. Searching – Linear Search – Binary Search.
SORTING
Preliminaries
A sorting algorithm is an algorithm that puts elements of a list in a certain order. The most used
orders are numerical order and lexicographical order. Efficient sorting is important to optimizing
the use of other algorithms that require sorted lists to work correctly and for producinghuman -
readable input.
Sorting algorithms are often classified by :
* Computational complexity (worst, average and best case) in terms of the size of the
list (N).
For typical sorting algorithms good behaviour is O(NlogN) and worst case behaviour
is O(N2) and the average case behaviour is O(N).
* Memory Utilization
* Stability - Maintaining relative order of records with equal keys.
* No. of comparisions.
* Methods applied like Insertion, exchange, selection, merging etc.
Sorting is a process of linear ordering of list of objects.
Sorting techniques are categorized into
Internal Sorting
External Sorting
Internal Sorting takes place in the main memory of a computer.
eg : - Bubble sort, Insertion sort, Shell sort, Quick sort, Heap sort, etc.
External Sorting, takes place in the secondary memory of a computer, Since the number of
objects to be sorted is too large to fit in main memory.
eg : - Merge Sort, Multiway Merge, Polyphase merge.
THE BUBBLE SORT
The bubble sort makes multiple passes through a list. It compares adjacent items and
exchanges those that are out of order. Each pass through the list places the next largest value in its
proper place. In essence, each item “bubbles” up to the location where it belongs.
Fig. 5.1 shows the first pass of a bubble sort. The shaded items are being compared to see
if they are out of order. If there are n items in the list, then there are n - 1n - 1 pairs of items that
need to be compared on the first pass. It is important to note that once the largest value in the list
is part of a pair, it will continually be moved along until the pass is complete.
First Pass
54 26 93 17 77 31 44 55 20 Exchange
26 54 93 17 77 31 44 55 20 No Exchange
26 54 93 17 77 31 44 55 20 Exchange
26 54 17 93 77 31 44 55 20 Exchange
26 54 17 77 93 31 44 55 20 Exchange
26 54 17 77 31 93 44 55 20 Exchange
26 54 17 77 31 44 93 55 20 Exchange
26 54 17 77 31 44 55 93 20 Exchange
26 54 17 77 31 44 55 20 93 93 in place
after first pass
Fig. Merge Sort
At the start of the second pass, the largest value is now in place. There are n - 1n - 1 items
left to sort, meaning that there will be n - 2 n - 2 pairs. Since each pass places the next largest
value in place, the total number of passes necessary will be n - 1 n - 1. After completing the n - 1
n - 1 passes, the smallest item must be in the correct position with no further processing required.
The exchange operation, sometimes called a “swap”.
Program for bubble sort:
def bubbleSort(alist):
for i in range(passnum):
if alist[i]>alist[i+1]:
temp = alist[i]
alist[i] = alist[i+1]
alist[i+1] = temp
alist = [54,26,93,17,77,31,44,55,20]
bubbleSort(alist)
print(alist)
Output:
[17, 20, 26, 31, 44, 54, 55, 77, 93]
Analysis:
To analyze the bubble sort, we should note that regardless of how the items are arranged in
the initial list, n”1n”1 passes will be made to sort a list of size n. Table -1 shows the number of
comparisons for each pass. The total number of comparisons is the sum of the first n - 1n - 1
integers. In the best case, if the list is already ordered, no exchanges will be made. However, in
the worst case, every comparison will cause an exchange. On average, we exchange half of the
time.
Pass Comparisons
1 n - 1n - 1
2 n - 2n - 2
3 n - 3n - 3
... ...
n - 1n - 1 11
Disadvantages:
Abubble sort is often considered the most inefficient sorting method since it must exchange
items before the final location is known. These “wasted” exchange operations are very costly.
However, because the bubble sort makes passes through the entire unsorted portion of the list, it
has the capability to do something most sorting algorithms cannot. In particular, if during a pass
there are no exchanges, then we know that the list must be sorted. A bubble sort can be modified
to stop early if it finds that the list has become sorted. This means that for lists that require just a
few passes, a bubble sort may have an advantage in that it will recognize the sorted list and stop
THE SELECTION SORT
The selection sort improves on the bubble sort by making only one exchange for every
pass through the list. In order to do this, a selection sort looks for the largest value as it makes a
pass and, after completing the pass, places it in the proper location. As with a bubble sort, after
the first pass, the largest item is in the correct place. After the second pass, the next largest is in
place. This process continues and requires n”1n”1passes to sort n items, since the final item must
be in place after the (n”1)(n”1) last pass.
Figure shows the entire sorting process. On each pass, the largest remaining item is selected
and then placed in its proper location. The first pass places 93, the second pass places 77, the third
places 55, and so on.
26 54 93 17 77 31 44 55 20 93 is largest
26 54 20 17 77 31 44 55 93 77 is largest
55 is largest
26 54 20 17 55 31 44 77 20
54 is largest
26 54 20 17 44 31 55 77 20
44 is largest
26 31 20 17 44 54 55 77 93 stays in place
31 is largest
26 31 20 17 44 54 55 77 93
26 is largest
26 17 20 31 44 54 55 77 93
20 17 26 31 44 54 55 77 93 20 is largest
20 26 44 54 55 77 93 17 ok
17 31
list is sorted
Program for Selection Sort:
def selectionSort(alist):
for fillslot in range(len(alist)-1,0,-1):
positionOfMax=0
for location in range(1,fillslot+1):
if alist[location]>alist[positionOfMax]:
positionOfMax = location
temp = alist[fillslot]
alist[fillslot] = alist[positionOfMax]
alist[positionOfMax] = temp
alist = [54,26,93,17,77,31,44,55,20]
selectionSort(alist)
print(alist)
Output:
[17, 20, 26, 31, 44, 54, 55, 77, 93]
INSERTION SORT
Insertion sorts works by taking elements from the list one by one and inserting them in their
current position into a new sorted list. Insertion sort consists of N - 1 passes, where N is the
number of elements to be sorted. The ith pass of insertion sort will insert the ith element A[i] into
its rightful place among A[1], A[2] --- A[i - 1]. After doing this insertion the records occupying
A[1] A[i] are in sorted order.
Insertion Sort Procedure
void Insertion_Sort (int a[ ], int n)
int i, j, temp ;
for (i = 0; i < n ; i++)
temp = a[i] ;
for (j = i ; j>0 && a[j-1] > temp ; j--)
a[j] = a[ j - 1] ;
a[j] = temp ;
Example
Consider an unsorted array as follows,
20 10 60 40 30 15
int i, j, k, temp;
for (k = N/2; k > 0 ; k = k/2)
for (i = k; i < N ; i++)
temp = A[i];
for (j = i; j >= k & & A [ j - k] > temp ; j = j - k)
A[j] = temp;
Example
Consider an unsorted array as follows.
81 94 11 96 12 35 17 95 28 58
Here N = 10, the first pass as K = 5 (10/2)
81 94 11 96 12 35 17 95 28 58
81 94 11 96 12 35 17 95 28 58
28 12 11 35 17 81 58 95 96 94
28 12 11 35 17 81 58 95 96 94
11 12 17 28 35 58 81 94 95 96
RADIX SORT
Radix sort is a small method used when alphabetizing a large list of names. Intuitively,
one might want to sort numbers on their most significant digit. However, Radix sort works counter-
intuitively by sorting on the least significant digits first. On the first pass, all the numbers are
sorted on the least significant digit and combined in an array. Then on the second pass, the entire
numbers are sorted again on the second least significant digits and combined in an array and so
on.
Algorithm: Radix-Sort (list, n)
shift = 1
for entry = 1 to n do
list = combinebuckets()
shift = shift * 10
Analysis
Each key is looked at once for each digit (or letter if the keys are alphabetic) of the longest
key. Hence, if the longest key has m digits and there are nkeys, radix sort has order O(m.n).
However, if we look at these two values, the size of the keys will be relatively small when
compared to the number of keys. For example, if we have six-digit keys, we could have a million
different records.
Here, we see that the size of the keys is not significant, and this algorithm is of linear
complexity O(n).
Example
Following example shows how Radix sort operates on seven 3-digits number.
Input 1st Pass 2nd Pass 3rd Pass
In the above example, the first column is the input. The remaining columns show the list
after successive sorts on increasingly significant digits position. The code for Radix sort assumes
that each element in an array A of nelements has d digits, where digit 1 is the lowest-order digit
and d is the highest-order digit.
Example
To show how radix sort works:
var array = [88, 410, 1772, 20]
Radix sort relies on the positional notation of integers, as shown here:
1 0 2 4
thousands hundreds tens ones
First, the array is divided into buckets based on the value of the least significant digit:
the ones digit.
0 - 410, 20
2 - 1772
8 - 88
These buckets are then emptied in order, resulting in the following partially-sorted array:
array = [410, 20, 1772, 88]
Next, repeat this procedure for the tens digit:
1 - 410
2 - 20
7 - 1772
8 - 88
The relative order of the elements didn‟t change this time, but you‟ve still got more
digits to inspect.
The next digit to consider is the hundreds digit:
- 20, 88
- 410
- 1772
For values that have no hundreds position (or any other position without a value), the digit
will be assumed to be zero.
Reassembling the array based on these buckets gives the following:
array = [20, 88, 410, 1772]
Finally consider the thousands digit:
- 1772
Reassembling the array from these buckets leads to the final sorted array:
array = [20, 88, 410, 1772]
When multiple numbers end up in the same bucket, their relative ordering doesn‟t change.
For example, in the zero bucket for the hundreds position, 20 comes before 88. This is because the
previous step put 20 in a lower bucket than 80, so 20 ended up before 88 in the array.
HASHING
Hash Table
The hash table data structure is an array of some fixed size, containing the keys. A key is a
value associated with each record.
Slot 1
Location
1
2
3 92
4 43
5
6 85
7
8
9
10
int Hashvalue = 0;
while (* key ! = „\0‟)
Hashval + = * key ++;
return Hashval % Tablesize;
0 10
1 11 81
4 34 94
7 17
9 29 89 99
Insertion
To perform the insertion of an element, traverse down the appropriate list to check whether
the element is already in place.
If the element is new one, the inserted it is either at the front of the list or at the end of the
list.
If it is a duplicate element, an extra field is kept and placed.
INSERT 10 :
Hash (k) = k% Tablesize
Hash (10) = 10 % 10
Hash (10) = 0
INSERT 11 :
Hash (11) = 11 % 10
Hash (11) = 1
INSERT 81 :
Hash (81) = 81% 10
Hash (81) = 1
The element 81 collides to the same hash value 1. To place the value 81 at this position
perform the following.
Traverse the list to check whether it is already present.
Since it is not already present, insert at end of the list. Similarly the rest of the elements are
inserted.
Routine To Perform Insertion
void Insert (int key, Hashtable H)
Find Routine
Position Find (int key, Hashtable H)
Position P;
List L;
L = H Thelists [Hash (key, H Tablesize)];
P = L Next;
while (P! = NULL && P Element ! = key)
P = p Next;
return p;
Advantage
More number of elements can be inserted as it uses array of linked lists.
Disadvantage of Seperate Chaining
* It requires pointers, which occupies more memory space.
* It takes more effort to perform a search, since it takes time to evaluate the hash function
and also to traverse the list.
1. Linear Search
2. Binary Search
LINEAR SEARCH
Linear search is a very basic and simple search algorithm. In Linear search, we search an
element or value in a given array by traversing the array from the starting, till the desired element
or value is found.
It compares the element to be searched with all the elements present in the array and when the
element is matched successfully, it returns the index of the element in the array, else it return -1.
Linear Search is applied on unsorted or unordered lists, when there are fewer elements in a list.
For Example,
Linear Search
10 14 19 26 27 31 33 35 42 44
=
33
Algorithm
10 14 19 26 27 31 33 35 42 44
0 1 2 3 4 5 6 7 8 9
First, we shall determine half of the array by using this formula -
mid = low + (high - low) / 2
Here it is, 0 + (9 - 0 ) / 2 = 4 (integer value of 4.5). So, 4 is the mid of the array.
10 14 19 26 27 31 33 35 42 44
0 1 2 3 4 5 6 7 8 9
Now we compare the value stored at location 4, with the value being searched, i.e. 31. We
find that the value at location 4 is 27, which is not a match. As the value is greater than 27 and we
have a sorted array, so we also know that the target value must be in the upper portion of the array.
10 14 19 26 27 31 33 35 42 44
0 1 2 3 4 5 6 7 8 9
We change our low to mid + 1 and find the new mid value again.
low = mid + 1
mid = low + (high - low) / 2
Our new mid is 7 now. We compare the value stored at location 7 with our target value 31.
10 14 19 26 27 31 33 35 42 44
0 1 2 3 4 5 6 7 8 9
The value stored at location 7 is not a match, rather it is more than what we are looking
for. So, the value must be in the lower part from this location.
10 14 19 26 27 31 33 35 42 44
0 1 2 3 4 5 6 7 8 9
Hence, we calculate the mid again. This time it is 5.
10 14 19 26 27 31 33 35 42 44
0 1 2 3 4 5 6 7 8 9
We compare the value stored at location 5 with our target value. We find that it is a match.
10 14 19 26 27 31 33 35 42 44
0 1 2 3 4 5 6 7 8 9
We conclude that the target value 31 is stored at location 5.
Binary search halves the searchable items and thus reduces the count of comparisons to be
made to very less numbers.
Pseudocode
The pseudocode of binary search algorithms should look like this “
Procedure binary_search
A ! sorted array
n ! size of array
x ! value to be searched
Set lowerBound = 1
Set upperBound = n
while x not found
if upperBound < lowerBound
EXIT: x does not exists.
set midPoint = lowerBound + ( upperBound - lowerBound ) / 2
if A[midPoint] < x
if A[midPoint] > x
if A[midPoint] = x
end while
end procedure