堆排序-升序和降序_TopK-N个数找找最大的前K个

一、堆排序

堆排序即利用堆的思想来进行排序,总共分为两个步骤:
1.建堆

升序:建大堆
降序:建小堆

image.png
2.利用堆删除思想来进行排序

方法一:把数据拷贝进堆、把堆拷贝进数据


//弊端,1.需要先有一个堆  2.时间复杂度+拷贝数据
void HeapSort(int* a, int n)
{
	HP hp;
	HeapInit(&hp);

	//建堆--向上调整建堆
	for (int i = 0; i < n; i++)		//N*logN
	{
		HeapPush(&hp,a[i]);			//把数据拷贝进堆
	}
	int i = 0;
	while (!HeapEmpty(&hp))			//N*logN
	{	
		int top = HeapTop(&hp);
		a[i++] = top;				//把堆拷贝进数据
		HeapPop(&hp);
	}

	HeapDestory(&hp);
}

int main()
{
	int a[] = { 1,5,6,7,8,9,3,4,6 };
	HeapSort(a, sizeof(a) / sizeof(a[0]));
	return 0;
}

改进思路:

升序–
建小堆? 存在可能把兄弟结点互换的情况,堆的关系可能会全乱,只能重新建堆。
建大堆,将元素第0个和最后一个互换,则最后一个就是最小元素。然后将该元素隔离开。
然后依次操作。
image.png
反之,降序–建小堆。

方法二:建堆,调整位置,再整理堆

void HeapSort(int* a, int n)
{
	//升序  建大堆--
	//降序  建小堆

	//建堆--向上调整建堆
	/*for (int i = 0; i < n; i++)	
	{
		AdjustUp(a,i);					//从根结点的孩子开始一一向上调整
	}*/

	//建堆--向下调整建堆--O(N)
	for (int i = (n - 1 - 1) / 2; i >= 0; --i)
	{	//int i = (n - 1   - 1) / 2     从倒数第一个非叶子结点倒着走,
		AdjustDown(a, n, i);			//建小堆,叶子结点不需要处理
	}
    //效率上存在差异

    //升序?降序?建堆时候注意建大堆还是小堆
    //调整顺序
	int end = n - 1;
	while (end > 0)
	{
		Swap(&a[0], &a[end]);

		//再调整,选出次小的数
		AdjustDown(a,end,0);
		--end;					//最后一个数不能看作堆里的
	}
}

int main()
{
	int a[] = { 1,5,6,7,8,9,3,4,6 };
	HeapSort(a, sizeof(a) / sizeof(a[0]));
	return 0;
}

调试观察:
image.png
第一次和第二次的调试结果
程序运行结束的结果为:
image.png符合我们的预想,降序建立小堆

二、TopK-N个数找找最大的前K个

image.png

改进思路:

1.将前K个数拷贝进新创建的空中中,建小堆;
2.后N-K个数与堆项元素一一比较,比堆项元素大,就调换他们的位置;
3.前K个数保持小堆,需要向下调整调换过位置;
4.最后的小堆的值就是前K个数。
image.png

#define _CRT_SECURE_NO_WARNINGS

#include "HeapTopK.h"

void CreateDate()//创建数据
{
	int n = 1000000;
	srand(time(0));
	const char* file = "data.txt";
	FILE* fin = fopen(file, "w");
	if (fin == NULL)
	{
		perror("fopen error");
		return;
	}

	for (size_t i = 0; i < n; i++)
	{
		int x = rand() % 10000000;
		fprintf(fin, "%d\n", x);
	}
	fclose(fin);
}

void TopK(int k)			//K=5
{
	//打开文件
	const char* file = "data.txt";
	//const char* file = "data1.txt";	//修改data1.txt中数为最大数,然后检验查看
	FILE* fount = fopen(file, "r");
	if (fount == NULL)
	{
		perror("fount fail");
		return;
	}

	//开辟k个空间,读出前k个数据放在一个数组中
	int* kminheap = (int*)malloc(sizeof(int) * k);
	if (kminheap == NULL)
	{
		perror("kminheap fail");
		return;
	}

	for (int i = 0; i < k; i++)
	{
		fscanf(fount, "%d", &kminheap[i]);
	}

	//建小堆
	for (int i = (k - 1 - 1) / 2; i >= 0; --i)
	{
		AdjustDown(kminheap, k, i);

	}

	//将fout中的元素(指针已经到了k+1个)val与kminheap中的堆顶元素(kminheap中最小的值)对比
	int val = 0;
	while (!feof(fount))
	{
		fscanf(fount, "%d", &val);//从fount中读取的值赋值给val
		if (val > kminheap[0])
		{
			kminheap[0] = val;
			AdjustDown(kminheap, k, 0);
		}
	}

	for (int i = 0; i < k; i++)
	{
		printf("%d ", kminheap[i]);
	}

}

int main()
{
	CreateDate();
	TopK(5);			//K=5
	return 0;
}

HeapTopK.c,HeapTopK.h的代码不赘述。
其中需要注意 srand,time 需要包含的头文件分别为<stdlib.h>,<time.h>.

  • 15
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个汇编程序,可以实现输入十个数排序升序降序输出。这个程序使用冒泡排序算法,将输入的数值按照从小到大的顺序排序,然后再按照从大到小的顺序输出。 ``` .MODEL SMALL .STACK 100H .DATA NUMS DB 10 DUP(?) ; 存储输入的十个数 MSG1 DB '请输入10个数字:$' MSG2 DB 13,10,'升序排序结果:$' MSG3 DB 13,10,'降序排序结果:$' NEWLINE DB 13,10,'$' COUNT DB 10 ; 数组长度 .CODE MAIN PROC MOV AX, @DATA MOV DS, AX ; 输出提示信息 MOV AH, 09H LEA DX, MSG1 INT 21H ; 循环读入十个数值 MOV CL, COUNT MOV SI, OFFSET NUMS READ_LOOP: MOV AH, 01H INT 21H ; 读取一个字符 CMP AL, 0DH ; 判断是否输入回车键 JE END_READ ; 如果输入回车键,结束读入 SUB AL, 30H ; 将 ASCII 码转换为数值 MOV [SI], AL ; 存储数值到数组中 INC SI LOOP READ_LOOP ; 继续读入下一个数值 ; 进行升序排序 MOV CX, COUNT DEC CX ; 最后一个元素不用比较 SORT_LOOP: MOV BX, CX ; BX 存储当循环次数 MOV SI, OFFSET NUMS ; SI 指向数组开头 INNER_LOOP: MOV DL, [SI] ; DL 存储当元素 MOV DH, [SI+1] ; DH 存储下一个元素 CMP DL, DH ; 比较两个元素 JBE SKIP_SWAP ; 如果面的元素小于等于后面的元素,不交换 XCHG DL, DH ; 否则交换两个元素 MOV [SI], DL ; 存储交换后的一个元素 MOV [SI+1], DH ; 存储交换后的后一个元素 SKIP_SWAP: INC SI ; 继续比较下一个元素 DEC BX ; 循环次数减一 JNZ INNER_LOOP ; 如果还有元素需要比较,继续循环 LOOP SORT_LOOP ; 继续进行下一轮排序 ; 输出升序排序结果 MOV AH, 09H LEA DX, MSG2 INT 21H MOV CX, COUNT MOV SI, OFFSET NUMS PRINT_LOOP: MOV DL, [SI] ADD DL, 30H ; 转换为 ASCII 码 MOV AH, 02H INT 21H ; 输出一个字符 MOV DL, ',' ; 输出逗号分隔符 INT 21H INC SI ; 继续输出下一个元素 LOOP PRINT_LOOP MOV DL, '$' MOV AH, 02H INT 21H ; 输出换行符 ; 进行降序排序 MOV CX, COUNT DEC CX ; 最后一个元素不用比较 SORT_LOOP2: MOV BX, CX ; BX 存储当循环次数 MOV SI, OFFSET NUMS ; SI 指向数组开头 INNER_LOOP2: MOV DL, [SI] ; DL 存储当元素 MOV DH, [SI+1] ; DH 存储下一个元素 CMP DL, DH ; 比较两个元素 JGE SKIP_SWAP2 ; 如果面的元素大于等于后面的元素,不交换 XCHG DL, DH ; 否则交换两个元素 MOV [SI], DL ; 存储交换后的一个元素 MOV [SI+1], DH ; 存储交换后的后一个元素 SKIP_SWAP2: INC SI ; 继续比较下一个元素 DEC BX ; 循环次数减一 JNZ INNER_LOOP2 ; 如果还有元素需要比较,继续循环 LOOP SORT_LOOP2 ; 继续进行下一轮排序 ; 输出降序排序结果 MOV AH, 09H LEA DX, MSG3 INT 21H MOV CX, COUNT MOV SI, OFFSET NUMS PRINT_LOOP2: MOV DL, [SI] ADD DL, 30H ; 转换为 ASCII 码 MOV AH, 02H INT 21H ; 输出一个字符 MOV DL, ',' ; 输出逗号分隔符 INT 21H INC SI ; 继续输出下一个元素 LOOP PRINT_LOOP2 MOV DL, '$' MOV AH, 02H INT 21H ; 输出换行符 ; 结束程序 MOV AH, 4CH INT 21H END_READ: ; 如果不足10个数值,则用0补充 SUB CL, COUNT JZ SORT_LOOP ; 如果已经输入10个数值,开始排序 ADD CL, COUNT ; 计算需要补充的0的个数 MOV SI, OFFSET NUMS ADD SI, COUNT ; 将指针移到最后一个元素的位置 FILL_LOOP: MOV DL, '0' MOV [SI], DL INC SI LOOP FILL_LOOP JMP SORT_LOOP MAIN ENDP END MAIN ``` 请注意,这个程序是用 8086 汇编语言编写的,可能需要在不同的汇编器中进行一些修改才能正确地运行。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值