#!/usr/bin/env python
# -*- coding: utf-8 -*-
# @Time    : 2020/4/4 22:39
# @Author  : zhoujianwen
# @Email   : zhou_jianwen@qq.com
# @File    : topn.py
# @Describe: 

"""
    算法面试:10亿个数中取TOP-1000个数
    堆的性质:每一个节点比它的左右子节点小,
    先取前N个数,构成小顶堆,即在内存中维护一个1000数的小顶堆
    然后对文件中读取数据,和堆顶比较:
    if 比堆顶小,则丢弃
    if 比堆顶大,替换根节点,并且调整堆,保持小顶堆的性质
    所有数据处理完,得到的即是Top-N
"""


class TopN:
    # 父节点下标
    def parent(self, n):
        return int((n - 1) / 2)

    # 左节点下标
    def left(self, n):
        return 2 * n + 1

    # 右节点下标
    def right(self, n):
        return 2 * n + 2

    # 构建小顶堆,保证父节点小于左右子节点
    def buildHeap(self, n, data):
        for i in range(1, n):
            t = i
            # 调整堆,如果节点比父亲节点小,则交换
            while t != 0 and data[t] < data[self.parent(t)]:
                temp = data[t]
                data[t] = data[self.parent(t)]
                data[self.parent(t)] = temp
                t = self.parent(t)

    # 调整data[i]
    def adjust(self, i, n, data):
        # 小于堆的根节点,不调整
        if data[i] <= data[0]:
            return

        # 置换堆顶
        temp = data[i]
        data[i] = data[0]
        data[0] = temp
        # 调整堆顶
        t = 0
        while (self.left(t) < n and data[self.left(t)] < data[t]) or (
                self.right(t) < n and data[self.right(t)] < data[t]):
            if self.right(t) < n and data[self.right(t)] < data[self.left(t)]:
                # 右孩子更小,置换右孩子
                temp = data[t]
                data[t] = data[self.right(t)]
                data[self.right(t)] = temp
                t = self.right(t)
            else:
                # 否则置换左孩子
                temp = data[t]
                data[t] = data[self.left(t)]
                data[self.left(t)] = temp
                t = self.left(t)

    # 寻找topN,调整data,将topN排到最前面
    def findTopN(self, n, data):
        # 先构建n个数的小顶堆
        self.buildHeap(n, data);
        # n往后的数进行调整
        for i in range(n, len(data)):
            self.adjust(i, n, data)
        return data


# 第一组测试 12个
arr1 = [58, 26, 45, 18, 22, 39, 96, 75, 80, 65, 63, 28]
print("原数组:" + str(arr1))
topn = TopN()
result = topn.findTopN(5, arr1)
print("数组进行Top-N调整:" + str(result))

# 第二组测试 随机20个
import random

N = 10  # 从数组中取TOP-N数据排序
tempList = []
for i in range(20):  # 篇幅有限,只生成20个数据,读者可自行修改为1000
    temp = random.randint(0, 1000)
    tempList.append(temp)
print("原数组:" + str(tempList))
topn = TopN()
result = topn.findTopN(N, tempList)

temp = result[:N]
temp.sort(key=None, reverse=True)
print("数组进行Top-{0}调整:{1}".format(N,temp))  # 只显示前10个数据,后面数据省略显示

tempList.sort(key=None, reverse=True)
print("数组进行Top-{0}排序:{1}".format(N,tempList[:N]))  # 只显示前10个数据,后面数据省略显示
原数组:[58, 26, 45, 18, 22, 39, 96, 75, 80, 65, 63, 28]
数组进行Top-N调整:[63, 65, 80, 75, 96, 18, 22, 26, 39, 45, 58, 28]
原数组:[676, 931, 685, 823, 889, 428, 681, 445, 497, 940, 113, 102, 299, 224, 811, 157, 889, 41, 959, 250]
数组进行Top-10调整:[959, 940, 931, 889, 889, 823, 811, 685, 681, 676]
数组进行Top-10排序:[959, 940, 931, 889, 889, 823, 811, 685, 681, 676]
Logo

CSDN联合极客时间,共同打造面向开发者的精品内容学习社区,助力成长!

更多推荐