算法----序列和的 top N
Description:
两个长度为 n 的数组 A 和 B, 各从中选出一个元素相加 A[i] + B[j], 求 top n 小的那些和。
思路 1:这样的和总共有 n^2 个, 排序,然后取前 n 个。
时间复杂度 O(n^2 lgn), 空间复杂度 O(n^2)
思路 2:用 最大堆。
     1. 首先用 n 个和建立一个最大堆, 
2. 然后循环剩余的 n^2 - n 个和,依次与堆顶元素比较(最大值), 如果小于最大值,则弹出堆顶,push 进入当前的和。
         时间复杂度 O(n^2 lgn), 空间复杂度是 O(n)
思路3: 还是采用 最小堆,并采用 多路合并 的思路。
首先,分析多路合并问题: 把 K 个有序数组合并成一个数组。
1. 将 K 个数组的首元素(最小),建立一个最小堆;
2. 弹出最小值,并插入新的数组中;将最小值所在的数组的下一个元素插入堆中。这样依次执行下去。
多路合并的代码见 http://blog.csdn.net/shoulinjun/article/details/19576585
回到我们的问题,将 N^2 个 sum 写成如下的形式:
所有 N^2 个和:(数组 A 和 B 排序后)
A[0] + B[0] <= A[0] + B[1] <= A[0] + B[2] <= ... <= A[0] + B[n-1]
A[1] + B[0] <= A[1] + B[1] <= A[1] + B[2] <= ... <= A[1] + B[n-1]
...............................
A[n-1] + B[0] <= A[n-1] + B[1] <= A[n-1] + B[2] <= ... <= A[n-1] + B[n-1]
注意: 第一步将数组 A 和 B 排序,则上述的 N 个序列都是有序的。
Bingo! 这时就可以采用与 多路合并 同样的算法:
Algorithm:
1. 将数组 A 和 B 排序;
2. 用 上述 N 个序列的首元素(最小),建立一个最小堆;
3. 弹出最小值,即为 top N;将最小值所在的数组的下一个元素插入堆中。这样执行 N 次, 得到 所有的 top N。
Complexity:
O(NlgN) + O(N)+ N*O(lgN) = O(NlgN)
Implementation:
实现细节的话,算法的第 2 步中, 最小值所在的数组的下一个元素如何得到?? 可以同时记录最小值这个和中所包含的B 中元素的下标。
下一个元素的值就是 min - B[i] + B[i+1]
//copyright @ L.J.SHOU Feb.14, 2014
#include <iostream>
#include <vector>
#include <utility>
#include <queue>
#include <algorithm>
#include <iterator>
using namespace std; //A and B have the same size K
vector<int> TopKSum(vector<int> &A, vector<int> &B)
{
vector<int> result;
priority_queue<pair<int, int>, vector<pair<int, int> >, greater<pair<int, int> > > Q; /* sort */
sort(A.begin(), A.end());
sort(B.begin(), B.end()); for(int i=0; i<A.size(); ++i)
Q.push(make_pair(A[i] + B[0], 0)); /* 0 denotes index in B */ for(int i=0; i<A.size(); ++i)
{
pair<int, int> e = Q.top(); Q.pop();
result.push_back(e.first);
/* push next element back into heap */
e.first = e.first - B[e.second] + B[e.second+1];
e.second ++;
Q.push(e);
} return result;
} int main(void)
{
vector<int> A, B, sum;
A.push_back(1);
A.push_back(2);
A.push_back(3); B.push_back(2);
B.push_back(4);
B.push_back(4); sum = TopKSum(A, B);
/* print vector */
copy(sum.begin(), sum.end(), ostream_iterator<int>(cout, " "));
cout << endl; return 0;
}
思考题:
n 个长度为 n 的数组,每个数组中选出一个元素相加, 求 top n 小的那些 sum。
复杂度: n^2 lgn
算法----序列和的 top N的更多相关文章
- 【机器学习】支持向量机(SVM)的优化算法——序列最小优化算法(SMO)概述
		SMO算法是一一种启发式算法,它的基本思路是如果所有变量的解的条件都满足最优化问题的KKT条件,那么这个最优化问题的解就得到了.因为KKT条件是该优化问题的充分必要条件. 整个SMO算法包括两个部分: ... 
- MapReduce算法形式五:TOP—N
		案例五:TOP—N 这个问题比较常见,一般都用于求前几个或者后几个的问题,shuffle有一个默认的排序是正序的,但如果需要逆序的并且暂时还不知道如何重写shuffle的排序规则的时候就用以下方法就行 ... 
- 经典算法(一) top k
		问题:1亿数据中,找出最大的k个数,要求使用内存不超过1m (延伸问题:1亿数据中,找出重复出现次数最多的k个,要求使用内存不超过1m 等) 分析: 1亿数字(int)占内存:100000000 * ... 
- 必考算法之 Top K 问题
		大家好,这里是<齐姐聊算法>系列之 Top K 问题. Top K 问题是面试中非常常考的算法题. 8 Leetcode 上这两题大同小异,这里以第一题为例. 题意: 给一组词,统计出现频 ... 
- Top K问题的两种解决思路
		Top K问题在数据分析中非常普遍的一个问题(在面试中也经常被问到),比如: 从20亿个数字的文本中,找出最大的前100个. 解决Top K问题有两种思路, 最直观:小顶堆(大顶堆 -> 最小1 ... 
- 标签传播算法(Label Propagation Algorithm, LPA)初探
		0. 社区划分简介 0x1:非重叠社区划分方法 在一个网络里面,每一个样本只能是属于一个社区的,那么这样的问题就称为非重叠社区划分. 在非重叠社区划分算法里面,有很多的方法: 1. 基于模块度优化的社 ... 
- 【决战西二旗】|理解Sort算法
		前言 前面两篇文章介绍了快速排序的基础知识和优化方向,今天来看一下STL中的sort算法的底层实现和代码技巧. 众所周知STL是借助于模板化来支撑数据结构和算法的通用化,通用化对于C++使用者来说已经 ... 
- K-近邻算法kNN
		K-近邻算法(k-Nearest Neighbor,简称kNN)采用测量不同特征值之间的距离方法进行分类,是一种常用的监督学习方法,其工作机制很简单:给定测试样本,基于某种距离亮度找出训练集中与其靠近 ... 
- c++模板库(简介)
		目 录 STL 简介 ......................................................................................... ... 
随机推荐
- hdu-----2491Priest John's Busiest Day(2008 北京现场赛G)
			Priest John's Busiest Day Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Jav ... 
- 理解模数转换器的噪声、ENOB和有效分辨率
			ADC的主要趋势之一是分辨率越来越高.这一趋势影响各种应用,包括工厂自动化.温度检测和数据采集.对更高分辨率的需求正促使设计者从传统的12位逐次逼近寄存器(SAR)ADC转至分辨率高达24位的Δ-ΣA ... 
- 批次更新BAPI_OBJCL_CHANGE
			FORM frm_edit_batch TABLES pt_field STRUCTURE dfies USING ps_batch TYPE ty_batch CHANGING ps_rturn T ... 
- 详解Jquery和AngularJs,Servlet中jsonp解决跨域问题(转)
			众所周知,jsonp可以解决跨域问题,下面是我在查阅资料和实际项目使用后的一些总结. Jquery中jsonp的使用 //myUrl = "http://localhost:8090/api ... 
- Mysql执行Update操作时会锁住表
			update tableA a,(select a.netbar_id,sum(a.reward_amt) reward_amt from tableB a group by a.netbar_id) ... 
- [整理]Linux压缩与解压缩命令整理。
			一.压缩文件命令 1.*.Z compress 程序压缩的档案:2.*.bz2 bzip2 程序压缩的档案:3.*.gz gzip 程序压缩的档案:4.*.tar tar 程序打包的数据,并没有压缩过 ... 
- ext afteredit
			ext afteredit详解 grid.on("afteredit",afterEidt,grid); function(obj) { obj.row;;//修改过的行从0开始 ... 
- telnet与ssh有什么不同呀
			含义: 1 使用Telnet这个用来访问远程计算机的TCP/IP协议以控制你的网络设备相当于在离开某个建筑时大喊你的用户名和口令.很快会有人进行监听,并且他们会利用你安全意识的缺乏.传统的网络服务程序 ... 
- jsm使用
			参考:http://blog.csdn.net/robinjwong/article/details/38820259 
- windows系统常见端口和木马默认使用端口
			dos命令netstat比较好用,能比较全的看到自己开放的端口及状态一般我用netstat -a端口:0服务:Reserved说明:通常用于分析操作系统.这一方法能够工作是因为在一些系统中“0”是无效 ... 
