1045. 快速排序

著名的快速排序算法里有一个经典的划分过程:我们通常采用某种方法取一个元素作为主元,通过交换,把比主元小的元素放到它的左边,比主元大的元素放到它的右边。 给定划分后的N个互不相同的正整数的排列,请问有多少个元素可能是划分前选取的主元?

例如给定N = 5, 排列是1、3、2、4、5。则:

  • 1的左边没有元素,右边的元素都比它大,所以它可能是主元;
  • 尽管3的左边元素都比它小,但是它右边的2它小,所以它不能是主元;
  • 尽管2的右边元素都比它大,但其左边的3比它大,所以它不能是主元;
  • 类似原因,4和5都可能是主元。

    因此,有3个元素可能是主元。

    输入格式:

    输入在第1行中给出一个正整数N(<= 105); 第2行是空格分隔的N个不同的正整数,每个数不超过109

    输出格式:

    在第1行中输出有可能是主元的元素个数;在第2行中按递增顺序输出这些元素,其间以1个空格分隔,行末不得有多余空格。

    输入样例:
    5
    1 3 2 4 5
    
    输出样例:
    3
    1 4 5
    

    算法:

    1. 算法开始。
    2. 获得N。
    3. 如果i>=N,则跳到第七步。
    4. 得到一个数字,如果这个数字比max大,那么它把值赋给max,并将其插到array数组的末尾。用top代表主元数列的末尾。
    5. 如果这个数字比max小,那么用二分查找法搜索恰好比此数小的数字,覆盖top,去除array数组里所有比它大的数字。
    6. 回到第三步。
    7. 输出array里元素的数量和所有元素。
    8. 算法结束。
    时间复杂度O(nlogn),空间复杂度O(n)。注意:如果没有可能的主元元素,那么要输出零,并且要输出两个换行符号,否则会发生格式错误。此代码运行于VS2015,修改scanf_s即可在PAT上AC。
    #include <stdio.h>
    #include <stdlib.h>
    int main(void) {
    	int* array;
    	int top = 0, max = 0, n, i, begin, end, middle, temp;
    	scanf_s("%d", &n);
    	array = (int*)calloc(n, sizeof(int));
    	for (i = 0; i < n; i++) {
    		scanf_s("%d", &temp);
    		if (temp > max) {
    			array[top++] = temp;
    			max = temp;
    		}
    		else {
    			begin = 0;
    			end = top - 1;
    			while (begin < end) {
    				middle = (begin + end) / 2;
    				if (array[middle] == temp) {
    					top = middle;
    					break;
    				}
    				else if (array[middle]>temp) {
    					end = middle - 1;
    				}
    				else if (array[middle] < temp) {
    					begin = middle + 1;
    				}
    			}
    			if (begin >= end) {
    				if (end >= 0 && array[end] < temp) {
    					top = end + 1;
    				}
    				else {
    					top = begin;
    				}
    			}
    		}
    	}
    	if (top) {
    		printf("%d\n", top);
    		printf("%d", array[0]);
    		for (i = 1; i < top; i++) {
    			printf(" %d", array[i]);
    		}
    	}
    	else {
    		printf("%d\n\n", top);
    	}
    	free(array);
    	return 0;
    }



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值