Preface: this chapter will introduce eight common sorts, including direct insertion sort, Hill sort, selection sort, heap sort, bubble sort, fast sort, merge sort and count sort (cardinal sort). The content of this chapter is the focus of the key points!!! Iron children must master all!!!

# 1. Insert sort

## 1.1 direct insertion sort

basic thought

Insert a number into the ordered interval to keep the interval in order. The current n+1 number is inserted into the front. The front arr[0] to arr[n-1] have been ordered. At this time, use the values of arr[n] and the front arr[n-1], arr[n-2]. By comparison, find the appropriate position to insert arr[n], and move the elements in the original position backward to realize the insertion

code implementation

//Insert sort void InsertSort(int* a, int n){ //Control start condition //Pay attention to controlling the termination conditions. The position of end here is the penultimate position, so - 1 is required for(int i=0; i<n-1;i++){ //Single pass insertion int end = i; int temp = a[end + 1]; //After the ordered interval while(end>=0){ //end to - 1 if(a[end]>temp){ a[end+1] = a[end]; --end; }else{ break; } } //There are two situations: the first is on the far right, the second is on the far left, and end is - 1, always after end a[end+1] = temp; } }

Time complexity

The time complexity of insertion sorting is also o (N^2). When it is close to order, its time complexity is O (N), because the result can be obtained after traversal. The space complexity is O (1).

## 1.2 Hill sorting

Shell's sort is a kind of insertion sort, also known as "narrowing increment sort". It is a more efficient improved version of direct insertion sort algorithm.

basic thought

Hill sorting is more efficient in dealing with some extreme cases. For example, in the above insertion sorting, if we rank the original array in ascending order in descending order, then we exchange a lot of times, which can also be said to be the worst case of insertion sorting. At this time, the intelligent ancestors thought of hill sorting and divided the array into gap groups, Then, it can be understood that each group is processed separately. In the process of gap from 5 – 2 – 1, in the case of descending order, the numbers with small values in the back can be arranged to the front in larger steps. When gap is 1, it is actually an insertion sort. The process of setting gap is also called pre sorting.

The smaller gap is, the closer it is to order, and the larger gap is, the less it is to order;

However, the smaller the gap, the slower it moves, and the larger the gap, the faster it moves;

code implementation

void ShellSort1(int* a, int n) { int gap = n; while (gap>1)//Don't be silly to add an equal sign, dead circle { gap = gap / 3 + 1;end The scope is[0,n-gap) for (int i = 0; i < n - gap; i++)//Side by side { int end = i; int temp = a[end + gap]; while (end>=0) { //If the current end value is greater than tmp, move to the end+gap position //So save the end+gap value in advance if (temp < a[end]) { a[end + gap] = a[end]; end = end-gap; } else { break; } } a[end + gap] = temp; } } }

Time complexity

O(N^1.3). Generally, gap/3+1 is recommended.

# 2. Select Sorting

The simple selection sorting idea is shown in the following dynamic diagram, that is, only one number in the head is compared. You can knock on the code yourself!

## 2.1 selection sorting (binary improved version)

basic thought

The optimized selection sorting can select the largest and the smallest at a time, and then put them in the appropriate position, that is, the smallest is placed in the first position and the largest is placed in the last position.

Then select the next small and the next large, and do so successively until there is only one value or no value left in the interval.

code implementation

void SelectSort(int* a, int n) { assert(a); int begin = 0, end = n - 1; while (begin < end) { int min = begin, max = begin; for (int i = begin; i <= end; i++)//Note that the starting point is begin { if (a[i] >= a[max]) max = i; if (a[i] < a[min]) min = i; } //The smallest is placed in Swap(&a[begin], &a[min]); //If the maximum position is in the begin position //Note that min is the maximum exchange position //At this time, the position of max changes //max changes to min //So update the position of max if (begin == max) max = min; Swap(&a[end], &a[max]); ++begin; --end; } }

Time complexity

O(N^2), worst sort

## 2.2 heap sorting

The article has been introduced in detail before stacking. Welcome to the specific details Click to view

basic thought

For details, please refer to the previous articles, and now emphasize that a large pile should be built in ascending order, and a small pile should be built in descending order

Take ascending order as an example: build a heap first, arrange a heap in ascending order, select the largest number and put it at the back, and then adjust it downward after meeting the size of the heap.

code implementation

//Heap sort void AdjustDown(int* a, int n, int parent){ int child = parent*2 + 1; while(child < n){ if(child+1<n && a[child+1] > a[child]){ ++child; } if(a[child]>a[parent]){ Swap(&a[child], &a[parent]); parent = child; child = parent*2+1; }else{ break; } } } void HeapSort(int* a, int n){ //Row in ascending order for(int i=(n-1-1)/2; i>=0; i--){ AdjustDown(a, n, i); } //O(N*logN) int end = n - 1; while(end > 0){ Swap(&a[0], &a[end]); AdjustDown(a, end, 0); //Isn't it wonderful hhh! end--; } }

Record the mistakes made when writing the heap (readers can skip it, which is left for the author to review for his own use):

Boundary problem, drawing, calm analysis!!!

Time complexity

The time complexity is O (N*logN), and the space complexity is O (1)

# 3. Exchange sorting

## 3.1 bubble sorting

basic thought

Taking ascending order as an example, the bubble sorting of each trip is to put a maximum number at the end. If a [I-1] > a [i], we exchange the values of I-1 and I, and repeat the cycle in turn.

code implementation

void BubbleSort(int* a, int n){ for(int j=0; j<n; j++){ int flag = 0; for(int i=1; i<n-j; ++i){ if(a[i] < a[i-1] ){ Swap(&a[i], &a[i-1]); flag = 1; } } if(flag == 0){ break; } } }

Time complexity

O(N^2)

## 3.2 quick sort

### 3.2.1 Hoare

basic thought

Select a key, usually a header.

Single trip: the key is placed in its correct position. The value on the left of the key is smaller than the key, and the value on the right of the key is larger than the key (this is the final position to be placed after the key is lined up)

After a single shot, find a way to make the left section orderly and the right section of key orderly

Then there are optimization solutions:

The first is to take random values as subscripts

The second is to get the subscript of the value that is not the largest or the smallest of the three numbers. In this case, there will be no worst case, because there are three arrays to get the middle

code implementation

Triple median (for optimization)

//Triple median int MidIndex(int* a, int left, int right) { int mid = left + (right - left) / 2; if (a[left] < a[mid]) { if (a[mid] < a[right]) { return mid; } else if (a[left] < a[right]) { return right; } else { return left; } } else //a[left] > a[mid] { if (a[mid] > a[right]) { return mid; } else if (a[left] < a[right]) { return left; } else { return right; } } } void Swap(int* a, int* b) { int tmp = *a; *a = *b; *b = tmp; } //We might as well think about what is the meaning of "three data fetching"?

Single pass sorting:

//What is the time complexity of a single sort operation? Think about how many times such a single sequence will be required for the next complete fast queue? int PartSort1(int* a, int left, int right) { int midi = MidIndex(a, left, right); Swap(&a[left], &a[midi]); //Take the leftmost key as an example int key = left; while (left<right) { //Because we take the key on the far left, we must go first on the right to find the one smaller than the key and think about why? //Go first on the right while (left < right && a[right] >= a[key]) { --right; } //Then go left while (left < right && a[left] < a[key]) { ++left; } Swap(&a[left], &a[right]); } Swap(&a[left], &a[key]);//At this time, left and right have met, the same return left; }

Full pass sorting (recursive):

void QuickSort(int* a, int left, int right) { //When the interval is divided to one or none, it returns if (left >= right) return; //Determine a position and divide the interval recursively //It is divided into [left, key-1] and [Key + 1, right] //int key = PartSort1(a, left, right); int key = PartSort2(a, left, right); QuickSort(a, left, key - 1); QuickSort(a, key + 1, right); }

First question: what is the meaning of "three data extraction"?

If we do not perform "three data fetching", if we encounter the worst case of fast scheduling - order, the time complexity will become O (N^2). If "three data fetching" is added, the worst case will no longer exist (the same is true for the latter two single pass sorting). Of course, there is not enough time in the actual interview process. There is no need to write another "three numbers in the middle". The interview is racing against time!

Second question: what is the time complexity of a single pass sorting operation? Think about how many times such a single sequence will be required for the next complete fast queue?

The time complexity of a single trip is O (N). A complete fast scheduling requires a single trip sort such as O (logN) trips.

The third question: why does the key choose the leftmost value, first let the number on the right go first and find the small value first?

In order to ensure a [left] < a [key] at the last meeting, just let the number on the right go first. When stopping at the end, whether it is "left meets right" or "right meets left", it meets a [left] < a [key].

Time complexity

A whole fast platoon: O (N*logN)

### 3.2.2 front and rear pointer method

basic thought

1.cur moves forward and finds data smaller than key

2. After finding data smaller than key, stop and + + prev

3. Exchange the values of prev and cur pointing to the position

Until cur reaches the rightmost position!

Before cur encounters data larger than key, prev follows cur. After cur encounters a value larger than key, there is a section of data larger than key between prev and cur.

code implementation

int PartSort2(int* a, int left, int right) { int midi = MidIndex(a, left, right); Swap(&a[midi], &a[left]); //Here, take the leftmost element as an example int key = left; int prev = left, cur = prev + 1; while (cur<=right) { if (a[cur] < a[key] && ++prev != cur)//Prevent yourself from exchanging with yourself { Swap(&a[cur], &a[prev]); } cur++; } //cur has come to the end. Exchange it. Swap(&a[prev], &a[key]);//Here we can ensure that a[prev] must be less than a[key] before the exchange. Why? return prev; }

Answer: a[prev] that jumps out of the while loop has just been exchanged with a[cur] before jumping out of the loop, and the exchange condition between a[prev] and a[cur] is that a[cur] is less than a[key], so it can be guaranteed that a[prev] must be less than a[key] before the last exchange after jumping out of the while loop.

Full pass sorting (recursive):

void QuickSort(int* a, int left, int right) { //When the interval is divided to one or none, it returns if (left >= right) return; //Determine a position and divide the interval recursively //It is divided into [left, key-1] and [Key + 1, right] //int key = PartSort1(a, left, right); int key = PartSort2(a, left, right); QuickSort(a, left, key - 1); QuickSort(a, key + 1, right); }

Time complexity

A whole fast platoon: O (N*logN)

### 3.2.3 excavation method

basic thought

For the pit digging method, you can choose to dig a pit at the 0 index (that is, take away the number and save it), then find a small pit to fill from the right, and then find a large pit to bury in the right from the left, so as to repeat the cycle until left and right meet, and finally put the key into the meeting point (the last pit).

code implementation

int PartSort3(int* a, int left, int right) { int midi = MidIndex(a, left, right); Swap(&a[midi], &a[left]); //Here key takes the leftmost number and lets the one on the right go first to find the small number int hole = left; int key = a[left]; while (left < right) { //First find the one smaller than key on the right and fill it in the pit on the left while (left < right && a[right] >= key) { right--; } a[hole] = a[right]; hole = right; //Find the one larger than the key on the left and exchange the pit while (left<right&&a[left]<key) { left++; } a[hole] = a[left]; hole = left; } a[left] = key;//Finally, put the key at the meeting point return left; }

Full pass sorting (recursive):

void QuickSort(int* a, int left, int right) { //When the interval is divided to one or none, it returns if (left >= right) return; //Determine a position and divide the interval recursively //It is divided into [left, key-1] and [Key + 1, right] //int key = PartSort1(a, left, right); int key = PartSort3(a, left, right); QuickSort(a, left, key - 1); QuickSort(a, key + 1, right); }

Time complexity

A whole fast platoon: O (N*logN)

## 3.3 quick sort (non recursive)

Our previous implementation of fast scheduling adopts recursive method, but recursion itself has "original sin", which lies in the following:

1. When the recursion depth is too large, the recursive program itself may be useless, but an error will be reported after compilation - stack overflow.

2. Performance problems (mentioned in some books, but now the compilation and optimization is very good, which is not a big problem).

There are two ways to change any recursive program into a non recursive program:

1. Loop (but some things are not easy to change into loop, such as traversal of binary tree, fast scheduling, etc.)

2. "Stack" simulation (this "stack" is the "stack" in the data structure, not the "stack" in the system. Generally, it is slightly more difficult to use the stack)

The fast scheduling change here is non recursive, which uses "stack" simulation.

basic thought

Non recursive here, with the help of the stack, we successively put the intervals we need to row in a single trip into the stack, take the intervals in the stack to row in a single trip, and then put the sub intervals to be processed into the stack, so as to cycle until the stack is empty.

Non recursive code implementation

void QuickSortNonR(int* a, int left, int right) { //Non recursive, we can deal with the current interval and then between partitions //First to the right, then to the left, then to the left Stack s; StackInit(&s); StackPush(&s,right); StackPush(&s,left); while (!StackEmpty(&s)) { left = StackTop(&s); StackPop(&s); right = StackTop(&s); StackPop(&s); //Process the current interval [left,right] int key = PartSort3(a, left, right); //Divide the left and right sections and stack them respectively //[left,key-1] key [key+1,right] //Enter the right section first. Only when the section has two values can it be processed if (key + 1 < right) { StackPush(&s, right); StackPush(&s, key + 1); } //Reentry left interval if (left < key - 1) { StackPush(&s, key - 1); StackPush(&s, left); } } }

Time complexity

The optimal time complexity is O(nlogn) and the worst space is O(n^2), because there is no worst case in the three values.

# 4. Merge and sort

## 4.1 recursive merge sort

basic thought

We can divide an array into two halves. For each array, when they are ordered, we can perform a merge operation. The two intervals are recursively divided into intervals. When the interval has only one value, we can merge and return to the previous layer, and let the previous layer merge and return.

code implementation

void _MergeSort(int* a, int left, int right,int* newArr) { if (left >= right) return; int mid = left + (right - left) / 2; //[left,mid][mid+1,right] _MergeSort(a, left, mid,newArr); _MergeSort(a, mid + 1, right,newArr); //It's already left and right here //Merge two intervals into one interval //Copy it to newArr, arrange it and put it back int index = left; int begin1 = left, end1 = mid; int begin2 = mid+1,end2 = right; while (begin1 <= end1 && begin2 <= end2) { if (a[begin1] < a[begin2]) { newArr[index++] = a[begin1++]; } else { newArr[index++] = a[begin2++]; } } //There must be a side here while (begin1 <= end1) { newArr[index++] = a[begin1++]; } while (begin2 <= end2) { newArr[index++] = a[begin2++]; } //Copy back to the location of element group letf -- right for (int i = left; i <= right; ++i) { a[i] = newArr[i]; } } void MergeSort(int* a, int n) { //Merge sort is an orderly re combination of left and right intervals //Therefore, ensure that the left and right intervals are orderly, and traverse to the leaves int* newArr = (int*)malloc(sizeof(int) * n); int left = 0; int right = n - 1; _MergeSort(a, left, right,newArr); }

Time complexity

O (NlogN). It can be seen that a group is divided equally every time in his recursive process. After the division, the height is about logN, and the spatial complexity is O (N)

## 4.2 iterative merging and sorting

Similar to fast scheduling, recursion will bring problems to fast scheduling, as well as merging and sorting, so try to use non recursive methods!

There are two ways to change any recursive program into a non recursive program:

1. Loop (but some things are not easy to change into loop, such as traversal of binary tree, fast scheduling, etc.)

2. "Stack" simulation (this "stack" is the "stack" in the data structure, not the "stack" in the system. Generally, it is slightly more difficult to use the stack)

Here, the non recursive implementation of merge sorting is to use "loop".

basic thought

Iterative implementation can be implemented by loops. It is easy to know that we control the iteration from the smallest subproblem, save the value of the smallest subproblem, and then provide it for later use. In fact, this is an example dynamic programming We can solve the "big BOSS" by using the solution of the sub problem.

code implementation

void MergeSortNonR(int* a, int n) { //int a[] = { 8,4,5,7,1,3,6,2,7,8 }; int* newArr=(int*)malloc(sizeof(int)*n) ; int groupNum = 1; int left; int right; //The idea of dynamic programming when we cut the smallest problem while(groupNum<n/2+1) { for (int i = 0; i < n; i += (2*groupNum)) { //Divided into two groups [begin1,end1][begin2,end2] int begin1 = i; int end1 = i + groupNum - 1; int begin2 = i + groupNum; int end2 = i + 2 * groupNum - 1; //Handle two cases. When end1 has crossed the boundary, it indicates the boundary of end1 if (end1 >= n) { end1 = n - 1; } //When end1 is out of bounds, naturally begin2 and end2 are out of bounds //The possible [begin1,end1] interval here also needs to be copied to the temporary array and then copied back to the original array if (begin2 >= n) { //Indicates that the right interval does not exist begin2 = n; end2 = n-1; } else if (begin2 < n && end2 >= n) { end2 = n - 1; } left = begin1; right = end2; //index is used to put in the temporary array newArr int index = begin1; while (begin1 <= end1 && begin2 <= end2) { if (a[begin1] < a[begin2]) { newArr[index++] = a[begin1++]; } else { newArr[index++] = a[begin2++]; } } //There must be a side here while (begin1 <= end1) { newArr[index++] = a[begin1++]; } while (begin2 <= end2) { newArr[index++] = a[begin2++]; } //Copy back to the location of element group letf -- right for (int x = left; x <= right; ++x) { a[x] = newArr[x]; } } groupNum*=2; } free(newArr); newArr = NULL; }

Time complexity

O(NlogN)

# 5. Counting and sorting

basic thought

1. Count the number of occurrences of each value in the original array

2. Sorting: traverse the Count array and write several values to the original array as many times as the value of the corresponding position appears

Of course, when the data is large, we can add one to the array after (the value - min) through relative mapping, and finally restore it back.

code implementation

void CountSort(int* a, int n) { //int a[] = { 31,24,25,16,1,0,79 }; //Go through it, find the maximum and minimum, and then open a larger array int max = a[0]; int min = a[0]; for (int i = 1; i < n; ++i) { if (max < a[i]) { max = a[i]; } if (min > a[i]) { min = a[i]; } } int size = max - min + 1; int* tmp = (int*)calloc(size,sizeof(int)); //Map the traversal of a to tmp. The length of a is n and the length of tmp is only size for (int i = 0; i < n; ++i) { tmp[a[i] - min]++;//tmp[i] stores the number of occurrences of this value } //Put it back according to the storage in tmp int index = 0; for (int i = 0; i < size; ++i) { while (tmp[i] > 0) { //This should be a subscript + mapping a[index++] = i + min; --tmp[i]; } } }

Time complexity

The time complexity of counting sorting is O (N) and the space complexity is O (max min), that is, the array we open is the range difference of this interval.

# 6. Comparison of eight orders

## 6.1 performance test and evaluation of eight sorting

// Performance comparison of test sorting void TestOP() { srand(time(0)); const int N = 10000; int* a1 = (int*)malloc(sizeof(int)*N); int* a2 = (int*)malloc(sizeof(int)*N); int* a3 = (int*)malloc(sizeof(int)*N); int* a4 = (int*)malloc(sizeof(int)*N); int* a5 = (int*)malloc(sizeof(int)*N); int* a6 = (int*)malloc(sizeof(int)*N); int* a7 = (int*)malloc(sizeof(int)*N); for (int i = 0; i < N; ++i) { a1[i] = rand(); a2[i] = a1[i]; a3[i] = a1[i]; a4[i] = a1[i]; a5[i] = a1[i]; a6[i] = a1[i]; a7[i] = a1[i]; } int begin1 = clock(); InsertSort(a1, N); int end1 = clock(); int begin2 = clock(); ShellSort(a2, N); int end2 = clock(); int begin3 = clock(); SelectSort(a3, N); int end3 = clock(); int begin4 = clock(); HeapSort(a4, N); int end4 = clock(); int begin5 = clock(); QuickSort(a4, 0, N-1); int end5 = clock(); int begin6 = clock(); MergeSort(a6, N); int end6 = clock(); printf("InsertSort:%d\n", end1 - begin1); printf("ShellSort:%d\n", end2 - begin2); printf("SelectSort:%d\n", end3 - begin3); printf("HeapSort:%d\n", end4 - begin4); printf("QuickSort:%d\n", end5 - begin5); printf("MergeSort:%d\n", end6 - begin6); free(a1); free(a2); free(a3); free(a4); free(a5); free(a6); free(a7); }

The results are as follows:

## 6.2 how to judge the stability of each sort?

Directly check whether the relative position of the same value will not change after sequencing. If it can be guaranteed, it is stable; otherwise, it is unstable.

Don't memorize, be sure to understand and analyze!

Bubble sorting: compare the two. The former is greater than the latter before exchange (ascending order). There will be no exchange order of equal values. It can ensure that the relative order of the same values will not be changed and is stable.

Simple selection sorting: in the process of exchanging positions between two numbers, one number in the array may be the same as the two numbers exchanged, so the relative order between the same numbers will be changed and unstable.

Direct insertion sorting: take out the elements from the front to the back and compare them with the previous ones. If the inserted value is smaller than the compared value, the compared value moves back; If the inserted value is larger than the compared value, it is directly inserted after the compared value, which does not change the relative order of the two same values and is stable.

Hill sorting: during pre sorting, the same data may be in different groups, which can not be controlled, so it is unstable.

Heap sorting: for example, two values of the same size, one at the "tree top" and the other in the "tree". The tree top element exchanges with the last element, which immediately affects the relative order of the same values and is unstable.

Merging and sorting: it can ensure that the relative order of the same values is unchanged and stable.

Quick sort: for example, the array has the same value as the key value, and the key will certainly move, so the relative order will change, so it is unstable.

Counting sorting: counting counts the number of occurrences of each number, but the same number is not distinguished between the first and the last, so it is unstable.

Add: what is the significance of stability?

For example, we have made an examination system. Among the candidates who hand in their papers first, their scores are in front of the array, and those who hand in their papers later, their scores are behind the array. When we rank the top few, we may meet two candidates with the same score. At this time, for the sake of fairness, the one with shorter test time should be in the front.

## 6.3 list of eight sorting time / space complexity

This is the end of the eight sorting algorithms of data structure. Thank you for reading!!! If the content is helpful to you, remember to give me three links (praise, collection and attention) - be a person with lingering fragrance.

;

}

**The results are as follows:** [External chain picture transfer...(img-46fMB6bi-1634470978201)] ## 6.2 how to judge the stability of each sort? > Directly check whether the relative position of the same value will not change after sequencing. If it can be guaranteed, it is stable; otherwise, it is unstable. > > Don't memorize, be sure to understand and analyze! > > **Bubble sorting**: Comparing the two, the exchange (ascending order) occurs only when the former is greater than the latter. There will be no exchange order of equal values, which can ensure that the relative order of the same values will not be changed and stable. > > **Simple selection sort**: In the process of exchanging positions between two numbers, one number in the array may be the same as the two numbers exchanged, which changes the relative order between the same numbers and is unstable. > > **Direct insert sort**: Take out the elements from front to back and compare them with the previous ones. If the inserted value is smaller than the compared value, the compared value moves back; If the inserted value is larger than the compared value, it is directly inserted after the compared value, which does not change the relative order of the two same values and is stable. > > **Shell Sort **: During pre sorting, the same data may be in different groups and cannot be controlled, so it is unstable. > > **Heap sort**: For example, two values of the same size, one at the "tree top" and one in the "tree", the tree top element exchanges with the last element, which immediately affects the relative order of the same values and is unstable. > > **Merge sort**: It can ensure that the relative order of the same value is unchanged and stable. > > **Quick sort**: For example, there is a following in the array key The same value, and key It will certainly move, so the relative order will change, so it is unstable. > > **Count sort**: Counting counts the number of occurrences of each number, but the same number is not distinguished between the first and the last, so it is unstable. **Add: what is the significance of stability?** > For example, we have made an examination system. Among the candidates who hand in their papers first, their scores are in front of the array, and those who hand in their papers later, their scores are behind the array. When we rank the top few, we may meet two candidates with the same score. At this time, for the sake of fairness, the one with shorter test time should be in the front. ## 6.3 list of eight sorting time / space complexity [External chain picture transfer...(img-qfgN3oVh-1634470978203)] *** **Eight sorting algorithms of data structure**This is the end of the introduction. Thank you for reading!!!**If the content is helpful to you, remember to give me three links (praise, collection and attention) - be a person with lingering fragrance.**