Mergesort again. 1. Split the list into two equal parts

Similar documents
Quicksort (Weiss chapter 8.6)

Better sorting algorithms (Weiss chapter )

Mergesort again. 1. Split the list into two equal parts

Data structures. More sorting. Dr. Alex Gerdes DIT961 - VT 2018

Sorting. Weiss chapter , 8.6

Real-world sorting (not on exam)

Algorithmic Analysis and Sorting, Part Two. CS106B Winter

Sorting. Sorting in Arrays. SelectionSort. SelectionSort. Binary search works great, but how do we create a sorted array in the first place?

Algorithmic Analysis and Sorting, Part Two

We can use a max-heap to sort data.

Lecture 8: Mergesort / Quicksort Steven Skiena

Divide and Conquer Sorting Algorithms and Noncomparison-based

CSE373: Data Structure & Algorithms Lecture 21: More Comparison Sorting. Aaron Bauer Winter 2014

Cosc 241 Programming and Problem Solving Lecture 17 (30/4/18) Quicksort

Quick Sort. CSE Data Structures May 15, 2002

CSC Design and Analysis of Algorithms

Sorting (Weiss chapter )

Sorting Algorithms. + Analysis of the Sorting Algorithms

CSE 332: Data Structures & Parallelism Lecture 12: Comparison Sorting. Ruth Anderson Winter 2019

CSE 373: Data Structures and Algorithms

Divide-and-Conquer. The most-well known algorithm design strategy: smaller instances. combining these solutions

CS 310 Advanced Data Structures and Algorithms

CS 171: Introduction to Computer Science II. Quicksort

CSC Design and Analysis of Algorithms. Lecture 6. Divide and Conquer Algorithm Design Technique. Divide-and-Conquer

COSC242 Lecture 7 Mergesort and Quicksort

Divide and Conquer. Algorithm D-and-C(n: input size)

DIVIDE AND CONQUER ALGORITHMS ANALYSIS WITH RECURRENCE EQUATIONS

Algorithms and Data Structures (INF1) Lecture 7/15 Hua Lu

CS 171: Introduction to Computer Science II. Quicksort

Chapter 4. Divide-and-Conquer. Copyright 2007 Pearson Addison-Wesley. All rights reserved.

CS 137 Part 8. Merge Sort, Quick Sort, Binary Search. November 20th, 2017

Sorting. Sorting. Stable Sorting. In-place Sort. Bubble Sort. Bubble Sort. Selection (Tournament) Heapsort (Smoothsort) Mergesort Quicksort Bogosort

Chapter 3:- Divide and Conquer. Compiled By:- Sanjay Patel Assistant Professor, SVBIT.

Data Structures and Algorithms

IS 709/809: Computational Methods in IS Research. Algorithm Analysis (Sorting)

BM267 - Introduction to Data Structures

CSE373: Data Structure & Algorithms Lecture 18: Comparison Sorting. Dan Grossman Fall 2013

17/05/2018. Outline. Outline. Divide and Conquer. Control Abstraction for Divide &Conquer. Outline. Module 2: Divide and Conquer

Divide-and-Conquer. Dr. Yingwu Zhu

SORTING AND SELECTION

Overview of Sorting Algorithms

Sorting. Task Description. Selection Sort. Should we worry about speed?

Faster Sorting Methods

Randomized Algorithms, Quicksort and Randomized Selection

CS2223: Algorithms Sorting Algorithms, Heap Sort, Linear-time sort, Median and Order Statistics

CS240 Fall Mike Lam, Professor. Quick Sort

CSE 373 MAY 24 TH ANALYSIS AND NON- COMPARISON SORTING

Searching, Sorting. part 1

Algorithms in Systems Engineering ISE 172. Lecture 12. Dr. Ted Ralphs

Sorting Pearson Education, Inc. All rights reserved.

Key question: how do we pick a good pivot (and what makes a good pivot in the first place)?

Sorting. CPSC 259: Data Structures and Algorithms for Electrical Engineers. Hassan Khosravi

Algorithmic Analysis. Go go Big O(h)!

ITEC2620 Introduction to Data Structures

Divide and Conquer 4-0

Data Structures and Algorithms

Unit-2 Divide and conquer 2016

SAMPLE OF THE STUDY MATERIAL PART OF CHAPTER 6. Sorting Algorithms

Sorting Algorithms Day 2 4/5/17

CS 61B Summer 2005 (Porter) Midterm 2 July 21, SOLUTIONS. Do not open until told to begin

Sorting Goodrich, Tamassia Sorting 1

2/26/2016. Divide and Conquer. Chapter 6. The Divide and Conquer Paradigm

Programming II (CS300)

Lecture 19 Sorting Goodrich, Tamassia

Problem. Input: An array A = (A[1],..., A[n]) with length n. Output: a permutation A of A, that is sorted: A [i] A [j] for all. 1 i j n.

Sorting. Bubble Sort. Pseudo Code for Bubble Sorting: Sorting is ordering a list of elements.

Sorting: Quick Sort. College of Computing & Information Technology King Abdulaziz University. CPCS-204 Data Structures I

7. Sorting I. 7.1 Simple Sorting. Problem. Algorithm: IsSorted(A) 1 i j n. Simple Sorting

ECE 2574: Data Structures and Algorithms - Basic Sorting Algorithms. C. L. Wyatt

Run Times. Efficiency Issues. Run Times cont d. More on O( ) notation

CSE 373: Data Structures and Algorithms

CS1 Lecture 30 Apr. 2, 2018

SORTING, SETS, AND SELECTION

DIVIDE & CONQUER. Problem of size n. Solution to sub problem 1

CPSC 311 Lecture Notes. Sorting and Order Statistics (Chapters 6-9)

MPATE-GE 2618: C Programming for Music Technology. Unit 4.2

Presentation for use with the textbook, Algorithm Design and Applications, by M. T. Goodrich and R. Tamassia, Wiley, Merge Sort & Quick Sort

Quicksort. Repeat the process recursively for the left- and rightsub-blocks.

9/10/12. Outline. Part 5. Computational Complexity (2) Examples. (revisit) Properties of Growth-rate functions(1/3)

4. Sorting and Order-Statistics

CS125 : Introduction to Computer Science. Lecture Notes #40 Advanced Sorting Analysis. c 2005, 2004 Jason Zych

Practical Session #11 - Sort properties, QuickSort algorithm, Selection

Sorting (I) Hwansoo Han

Test 1 Review Questions with Solutions

Searching in General

SEARCHING, SORTING, AND ASYMPTOTIC COMPLEXITY. Lecture 11 CS2110 Spring 2016

CS Divide and Conquer

Quick-Sort fi fi fi 7 9. Quick-Sort Goodrich, Tamassia

CSC Design and Analysis of Algorithms. Lecture 6. Divide and Conquer Algorithm Design Technique. Divide-and-Conquer

CS S-11 Sorting in Θ(nlgn) 1. Base Case: A list of length 1 or length 0 is already sorted. Recursive Case:

Lecture 7 Quicksort : Principles of Imperative Computation (Spring 2018) Frank Pfenning

Sorting Algorithms. CptS 223 Advanced Data Structures. Larry Holder School of Electrical Engineering and Computer Science Washington State University

Sorting. Data structures and Algorithms

Sorting. CSE 143 Java. Insert for a Sorted List. Insertion Sort. Insertion Sort As A Card Game Operation. CSE143 Au

Sorting. Riley Porter. CSE373: Data Structures & Algorithms 1

Quick-Sort. Quick-Sort 1

Question 7.11 Show how heapsort processes the input:

Divide and Conquer Algorithms: Advanced Sorting

Quick-Sort. Quick-sort is a randomized sorting algorithm based on the divide-and-conquer paradigm:

Sorting. CSC 143 Java. Insert for a Sorted List. Insertion Sort. Insertion Sort As A Card Game Operation CSC Picture

Transcription:

Quicksort

Mergesort again 1. Split the list into two equal parts 5 3 9 2 8 7 3 2 1 4 5 3 9 2 8 7 3 2 1 4

Mergesort again 2. Recursively mergesort the two parts 5 3 9 2 8 7 3 2 1 4 2 3 5 8 9 1 2 3 4 7

Mergesort again 3. Merge the two sorted lists together 2 3 5 8 9 1 2 3 4 7 1 2 2 3 3 4 5 7 8 9

Quicksort Mergesort is great... except that it's not in-place So it needs to allocate memory And it has a high constant factor Quicksort: let's do divide-and-conquer sorting, but do it in-place

Quicksort Pick an element from the array, called the pivot Partition the array: First come all the elements smaller than the pivot, then the pivot, then all the elements greater than the pivot Recursively quicksort the two partitions

Quicksort 5 3 9 2 8 7 3 2 1 4 Say the pivot is 5. Partition the array into: all elements less than 5, then 5, then all elements greater than 5 3 3 2 2 1 4 5 9 8 7 Less than the pivot Greater than the pivot

Quicksort Now recursively quicksort the two partitions! 3 3 2 2 1 4 5 9 8 7 Quicksort Quicksort 1 2 2 3 3 4 5 7 8 9

Pseudocode // call as sort(a, 0, a.length-1); void sort(int[] a, int low, int high) { if (low >= high) return; int pivot = partition(a, low, high); // assume that partition returns the // index where the pivot now is sort(a, low, pivot-1); sort(a, pivot+1, high); } Common optimisation: switch to insertion sort when the input array is small

Quicksort's performance Mergesort is fast because it splits the array into two equal halves Quicksort just gives you two halves of whatever size! So does it still work fast?

Complexity of quicksort In the best case, partitioning splits an array of size n into two halves of size n/2: n n/2 n/2

Complexity of quicksort The recursive calls will split these arrays into four arrays of size n/4: n n/2 n/2 n/4 n/4 n/4 n/4

n n/2 n/2 Total time is O(n log n)! n/4 n/4 n/4 n/4 log n levels n/8 n/8 n/8 n/8 n/8 n/8 n/8 n/8 O(n) time per level

Complexity of quicksort But that's the best case! In the worst case, everything is greater than the pivot (say) The recursive call has size n-1 Which in turn recurses with size n-2, etc. Amount of time spent in partitioning: n + (n-1) + (n-2) + + 1 = O(n 2 )

n n-1 Total time is O(n 2 )! n-2 n levels n-3 O(n) time per level

Worst cases When we simply use the first element as the pivot, we get this worst case for: Sorted arrays Reverse-sorted arrays The best pivot to use is the median value of the array, but in practice it's too expensive to compute... Most important decision in QuickSort: what to use as the pivot

Complexity of quicksort You don't need to split the array into exactly equal parts, it's enough to have some balance e.g. 10%/90% split still gives O(n log n) runtime Median-of-three: pick first, middle and last element of the array and pick the median of those three gives O(n log n) in practice Pick pivot at random: gives O(n log n) expected (probabilistic) complexity Introsort: detect when we get into the O(n 2 ) case and switch to a different algorithm (e.g. heapsort)

Partitioning algorithm 1. Pick a pivot (here 5) 5 3 9 2 8 7 3 2 1 4

Partitioning algorithm 2. Set two indexes, low and high 5 3 9 2 8 7 3 2 1 4 low high Idea: everything to the left of low is less than the pivot (coloured yellow), everything to the right of high is greater than the pivot (green)

Partitioning algorithm 3. Move low right until you find something greater than the pivot 5 3 9 2 8 7 3 2 1 4 low high

Partitioning algorithm 3. Move low right until you find something greater or equal to the pivot 5 3 9 2 8 7 3 2 1 4 low high while (a[low] < pivot) low++;

Partitioning algorithm 3. Move low right until you find something greater than the pivot 5 3 9 2 8 7 3 2 1 4 low high while (a[low] < pivot) low++;

Partitioning algorithm 3. Move high left until you find something less than the pivot 5 3 9 2 8 7 3 2 1 4 low high while (a[high] < pivot) high--;

Partitioning algorithm 4. Swap them! 5 3 4 2 8 7 3 2 1 9 low swap(a[low], a[high]); high

Partitioning algorithm 5. Advance low and high and repeat 5 3 4 2 8 7 3 2 1 9 low low++; high--; high

Partitioning algorithm 5. Advance low and high and repeat 5 3 4 2 8 7 3 2 1 9 low high while (a[low] < pivot) low++;

Partitioning algorithm 5. Advance low and high and repeat 5 3 4 2 8 7 3 2 1 9 low high

Partitioning algorithm 5. Advance low and high and repeat 5 3 4 2 8 7 3 2 1 9 low high while (a[high] < pivot) high++;

Partitioning algorithm 5. Advance low and high and repeat 5 3 4 2 1 7 3 2 8 9 low swap(a[low], a[high]); high

Partitioning algorithm 5. Advance low and high and repeat 5 3 4 2 1 7 3 2 8 9 low++; high--; low high

Partitioning algorithm 5. Advance low and high and repeat 5 3 4 2 1 7 3 2 8 9 low high

Partitioning algorithm 5. Advance low and high and repeat 5 3 4 2 1 7 3 2 8 9 low high

Partitioning algorithm 5. Advance low and high and repeat 5 3 4 2 1 2 3 7 8 9 low high

Partitioning algorithm 5. Advance low and high and repeat 5 3 4 2 1 2 3 7 8 9 low high

Partitioning algorithm 5. Advance low and high and repeat 5 3 4 2 1 2 3 7 8 9 low high

Partitioning algorithm 6. When low and high have crossed, we are finished! 5 3 4 2 1 2 3 7 8 9 But the pivot is in the wrong place. low high

Partitioning algorithm 7. Last step: swap pivot with high 3 3 4 2 1 2 5 7 8 9 low high

Details 1. What to do if we want to use a different element (not the first) for the pivot? Swap the pivot with the first element before starting partitioning!

Details 2. What happens if the array contains many duplicates? Notice that we only advance a[low] as long as a[low] < pivot If a[low] == pivot we stop, same for a[high] If the array contains just one element over and over again, low and high will advance at the same rate Hence we get equal-sized partitions

Pivot Which pivot should we pick? First element: gives O(n 2 ) behaviour for alreadysorted lists Median-of-three: pick first, middle and last element of the array and pick the median of those three Pick pivot at random: gives O(n log n) expected (probabilistic) complexity

Quicksort Typically the fastest sorting algorithm......but very sensitive to details! Must choose a good pivot to avoid O(n 2 ) case Must take care with duplicates Switch to insertion sort for small arrays to get better constant factors If you do all that right, you get an inplace sorting algorithm, with low constant factors and O(n log n) complexity

Mergesort vs quicksort Quicksort: In-place O(n log n) but O(n 2 ) if you are not careful Works on arrays only (random access) Compared to mergesort: Not in-place O(n log n) Only requires sequential access to the list this makes it good in functional programming Both the best in their fields! Quicksort best imperative algorithm Mergesort best functional algorithm

Sorting Why is sorting important? Because sorted data is much easier to deal with! Searching use binary instead of linear search Finding duplicates takes linear instead of quadratic time etc. Most sorting algorithms are based on comparisons Compare elements is one bigger than the other? If not, do something about it! Advantage: they can work on all sorts of data Disadvantage: specialised algorithms for e.g. sorting lists of integers can be faster

Complexity of recursive functions

Calculating complexity Let T(n) be the time mergesort takes on a list of size n Mergesort does O(n) work to split the list in two, two recursive calls of size n/2 and O(n) work to merge the two halves together, so... T(n) = O(n) + 2T(n/2) Time to sort a list of size n Linear amount of time spent in splitting + merging Plus two recursive calls of size n/2

Calculating complexity Procedure for calculating complexity of a recursive algorithm: Write down a recurrence relation e.g. T(n) = O(n) + 2T(n/2) Solve the recurrence relation to get a formula for T(n) (difficult!) There isn't a general way of solving any recurrence relation we'll just see a few families of them

Approach 1: draw a diagram

n T(n) n/2 n/2 Total time is O(n log n)! n/4 n/4 n/4 n/4 2T(n/2) O(log n) levels 4T(n/4) n/8 n/8 n/8 n/8 n/8 n/8 n/8 n/8 8T(n/8) O(n) time per level

Another example: T(n) = O(1) + 2T(n-1)

1 T(n) 1 1 1 1 1 1 2T(n-1) O(n) levels 4T(n-2) 1 1 1 1 1 1 1 1 8T(n-3) amount of work doubles at each level

1 T(n) 1 1 Total time is O(2 n )! 1 1 1 1 2T(n-1) O(n) levels 4T(n-2) 1 1 1 1 1 1 1 1 8T(n-3) amount of work doubles at each level

This approach Good for building an intuition Maybe a bit error-prone Approach 2: expand out the definition Example: solving T(n) = O(n) + 2T(n/2)

Expanding out recurrence relations T(n) = n + 2T(n/2) Get rid of big-o before expanding out (n instead of O(n)) the big O just gets in the way here

Expanding out recurrence relations T(n) = n + 2T(n/2) = n + 2(n/2 + 2T(n/4)) Expand out T(n/2) = n + n + 4T(n/4) = n + n + n + 8T(n/8) = = n + n + n + + n + T(1) (log n times) = O(n log n) (Note that T(1) is a constant so O(1))

If you prefer it a bit more formally... T(n) = n + 2T(n/2) = 2n + 4T(n/4) = 3n + 8T(n/8) =... General form is kn + 2 k T(n/2 k ) When k = log n, this is n log n + nt(1) which is O(n log n)

Divide-and-conquer algorithms T(n) = O(n) + 2T(n/2): T(n) = O(n log n) This is mergesort! T(n) = 2T(n-1): T(n) = O(2 n ) Because 2 n recursive calls of depth n (exercise: show this) Other cases: master theorem (Wikipedia) Kind of fiddly best to just look it up if you need it

Another example: T(n) = O(n) + T(n-1) T(n) = n + T(n-1) = n + (n-1) + T(n-2) = n + (n-1) + (n-2) + T(n-3) = = n + (n-1) + (n-2) + + 1 + T(0) = n(n+1) / 2 + T(0) = O(n 2 )

Another example: T(n) = O(1) + T(n-1) T(n) = 1 + T(n-1) = 2 + T(n-2) = 3 + T(n-3) = = n + T(0) = O(n)

Another example: T(n) = O(1) + T(n/2) T(n) = 1 + T(n/2) = 2 + T(n/4) = 3 + T(n/8) = = log n + T(1) = O(log n)

Another example: T(n) = O(n) + T(n/2) T(n) = n + T(n/2): T(n) = n + T(n/2) = n + n/2 + T(n/4) = n + n/2 + n/4 + T(n/8) = = n + n/2 + n/4 + < 2n = O(n)

Functions that recurse once T(n) = O(1) + T(n-1): T(n) = O(n) T(n) = O(n) + T(n-1): T(n) = O(n 2 ) T(n) = O(1) + T(n/2): T(n) = O(log n) T(n) = O(n) + T(n/2): T(n) = O(n) An almost-rule-of-thumb: Solution is maximum recursion depth times amount of work in one call (except that this rule of thumb would give O(n log n) for the last case)

Complexity of recursive functions Basic idea recurrence relations Easy enough to write down, hard to solve One technique: expand out the recurrence and see what happens Another rule of thumb: multiply work done per level with number of levels Drawing a diagram might help Master theorem for divide and conquer Luckily, in practice you come across the same few recurrence relations, so you just need to know how to solve those