寻找发帖“水王”

题目背景:Tango是微软亚洲研究院的一个试验项目。研究院的员工和实习生们都很喜欢在Tango上面交流灌水。传说,Tango有一大“水王”,他不但喜欢发贴,还会回复其他ID发的每个帖子。坊间风闻该“水王”发帖数目超过了帖子总数的一半。如果你有一个当前论坛上所有帖子(包括回帖)的列表,其中帖子作者的ID也在表中,你能快速找出这个传说中的Tango水王吗?

方案一:就是对发帖的ID 进行排序。对排完序的ID进行查找超过一半的ID号,排序的时间复杂度为O(N*logN)。在排序的ID中判断是否出现一半的时间复杂度为O(N),因为“水王”帖子的ID一定是N/2这个数字。总的时间复杂度为O(N*logN).

方案二:用快速排序中的partion()函数来进行查找。这时候不需要将ID进行排序,只需要找出index=N/2的那个数就行了。在index之前的都小于data[index],在index之后都大于data[index]。此时其时间复杂度为O(N)。最后判断该数是不是“水王”,其时间复杂度为O(N)。

方案三:根据数据的特点和规律来查找大致思想就是:因为我们不知道那个才是“水王”,所以假设每个ID都有可能是水王,这里我们先假设第一个帖子是“水王”,那么在遍历这个水王ID的时候,就要遇到一种挑战,可能自己的帖子数是会增加的,下一个ID也可能是“水王”,那么怎么办呢?那就要比较谁的帖子数多。如果下一个帖子和该帖子相同,则帖子数增加1,技能+1;如果下一个帖子数不和该帖子数不相同,则就是有别人也可能是“水王”,这时候自己的帖子数-1,两个互相抵消,看谁能挺到最后;因为如果你是“水王”,你的帖子数一定大于其他的用户,所以最后一定是“水王”胜出。因此这样遍历下来,只有水王的帖子减去对手的帖子数会是大于0的。其他任何帖子都是不能和它斗武的。

方案三解决步骤:

(1). 假设帖子的第一个ID是次数最大的,用candidate记录,nTimes代表其出现的次数。

(2). 遍历下一个ID,如果跟candidate一样,nTimes++,否则,遇到另一个对手也可能是“水王”,则nTimes--,如果nTimes == 0,下一步就要重复第一步了。

(3). 遍历结束,nTimes>0的那个candidate就是“水王”ID,他是获胜者。

代码的具体实现:

int Find(int *arr,int lenght)
{
	int Candidate;
	int nTimes,i;
	for(i=nTimes=0;i<lenght;i++)
	{
		if(0==nTimes)
		{
			Candidate=arr[i];
			nTimes=1;
		}
		else
		{
			if(Candidate==arr[i])
				nTimes++;
			else
				nTimes--;
		}
	}
	return Candidate;//Candidate一定是最后nTimes为1的数字;
}

扩展问题:随着Tango的发展,管理员发现,“超级水王”没有了。统计结果表明,有3个发帖很多的ID,他们的发帖数目都超过了帖子总数目N的1/4。你能从发帖ID列表中快速找出他们的ID吗?

虽然题目变成了三个,但是其本质还是没有变,我们依然可以延续上面的思路来解决该问题。只不过现在又三个“水王”,依次用上面的方法,我们也会很快找到这三个,因为他们的发帖数都超过了1/4。

具体实现代码如下:

int candidate[3];
void Find(int* arr, int lenght,int *candidate)
{
    int nTimes[3];
    nTimes[0]=nTimes[1]=nTimes[2]=0;  //次数初始化为0
    candidate[0]=candidate[1]=candidate[2]=0;  //代表三个水王;
    for(int i = 0; i < length; i++)
    {
        if(arr[i]==candidate[0])
        {
             nTimes[0]++;
        }
        else if(arr[i]==candidate[1])
        {
             nTimes[1]++;
        }
        else if(arr[i]==candidate[2])
        {
             nTimes[2]++;
        }
        else if(nTimes[0]==0)
        {
             nTimes[0]=1;
             candidate[0]=arr[i];
        }
        else if(nTimes[1]==0)
        {
             nTimes[1]=1;
             candidate[1]=arr[i];
        }
        else if(nTimes[2]==0)
        {
             nTimes[2]=1;
             candidate[2]=arr[i];
        }
        else
        {
             nTimes[0]--;
             nTimes[1]--;
             nTimes[2]--;
         }
    }
    return;
}  

因为只需要找出“水王”,所以思想和上面一样。

时间: 2024-11-09 09:14:28

寻找发帖“水王”的相关文章

寻找发帖水王02

题目:随着论坛的发展,管理员发现水王没有了,但是统计结果表明,有3个发帖很多的ID.据统计他们的发帖数目都超过了帖子总数目的1/4,你能从发帖列表中快速找出他们吗? 设计思路: 水王01只需要一个结果,而现在需要3个结果,上题用到的nTimes,也应改为3个计数器.现在需要3个变量来记录当前遍历过的3个不同的ID,而nTimes的3个元素分别对应当前遍历过的3个ID出现的个数.如果遍历中有某个ID不同于这3个当前ID,就判断当前3个ID是否有某个的nTimes为0,如果有,那这个新遍历的ID就取

寻找发帖水王

理论: java实现: package 经典; public class 寻找发帖水王 { /** * @param args */ public static void main(String[] args) { // TODO 自动生成的方法存根 int[] a={1,6,4,6,5,6,6,7,6,6,6,3,6,6,13,6}; System.out.println(find(a)); } public static int find(int []a) { int result=0,ti

编程之美2.3: 寻找发帖水王

题目:传说,Tango有一大"水王",他不但喜欢发帖,还会回复其他ID发的帖子,发帖数目超过帖子总数的一半,如果你有一个当前论坛上所有帖子的列表,其中帖子作者的ID也在表中,你能快速找到这个传说中的Tango水王吗? 解题思路:由于水王的发帖数目超过一半,当每次删除两个不同ID的帖子时,水王占得帖子数目仍然大于剩下帖子的一半,重复整个过程,将ID列表中的ID总数降低,转化为更小的问题,从而得到最后水王的ID. #include <iostream> #include <

主元素 寻找发帖水王

寻找发帖“水王” 题目是这样描述的:“水王”发帖的数目超过了所有帖子的一半,有各个帖子的作者ID,求这个水王的ID 编程之美给出了两种巧妙的解法 解法一:ID排序,那么ID列表中的N/2项即为要找的ID(还要排序,时间复杂度为O(NlogN)) 解法二:通过查找,每次从列表中除去两个不一样的ID,最后就可以得出这个ID,时间复杂度O(N).写法上也有技巧,不必非要找到一个不一样的在继续下去,如果下一个一样,那么用一个变量记录这个次数,把次数+1,遇见不一样的-1.例如1,1,2,3,4 初始va

数字之魅——寻找发帖水王

寻找发帖水王这个题目给了我很大的启发,同时开阔了视野,往往在解决这类型问题的时候第一想法都是先排序再计算.而本题却给出了一个非常好的思路,时间复杂度为O(N). 它还有一个扩展问题,但是我在网上看了几篇有的写的考虑不周全,有的写的逻辑不是很清楚,这里我也根据思考和查阅,给出我自己的解法. typedef int Type; //给出Id的一个抽象 Type candidate[3] = {0}; //说明传的这个参数前要初始化. void find3(Type* ID,int N,Type* c

编程之美2.3 寻找发帖水王

这道题目由于不容易写测试用例,所以,可以把题目转换为:在一个数组中,有一个数字出现的次数超过了数组大小的一半,这和题目原意是一样的. 这道题目的思想是我们同时去掉数组中两个不一样的数字,那么,数组中原来存在的规律是不变的(仅针对这个题目). 好吧,还是先给出函数声明: /*2.3 寻找发帖水王*/ bool DutVerify(int*, int, int); int DutFindNumMoreThanHalf(int*, int); 可以看到,函数声明中多了一个:DutVerify,这个函数

第2章 数字之魅——寻找发帖“水王”

寻找发帖“水王” 问题描述 Tango是微软亚洲研究院的一个试验项目.研究院的员工和实习生们都很喜欢在Tango上面交流灌水.传说,Tango有一大"水王",他不但喜欢发贴,还会回复其他ID发的每个帖子.坊间风闻该"水王"发帖数目超过了帖子总数的一半.如果你有一个当前论坛上所有帖子(包括回帖)的列表,其中帖子作者的ID也在表中,你能快速找出这个传说中的Tango水王吗? 解法 采用Map存储每个ID和它出现的次数,之后遍历一遍Map找出其中的“水王”,时间复杂度为O

编程之美2.3笔记:寻找发帖“水王”

<编程之美>2.3: Tango是微软亚洲研究院的一个试验项目.研究院的员工和实习生们都很喜欢在Tango上面交流灌水.传说,Tango有一大“水王”,他不但喜欢发贴,还会回复其他ID发的每个帖子.坊间风闻该“水王”发帖数目超过了帖子总数的一半.如果你有一个当 前论坛上所有帖子(包括回帖)的列表,其中帖子作者的ID也在表中,你能快速找出这个传说中的Tango水王吗? 当面试的时候我们遇到这样的问题,应该怎么去思考呢?读SICP的一个很大的收获是,学会抽象.贵族娱乐城 抽象 抽象就是从问题中提取

【编程之美】寻找发帖&quot;水王&quot;

Tango是微软亚洲研究院的一个试验项目.研究院的员工和实习生们都很喜欢在Tango上面交流灌水.传说,Tango有一大"水王",他不但喜欢发帖,还会回复其他ID发的每个帖子.坊间风闻该"水王"发帖数目超过了帖子总数的一半.如果你有一个当前论坛上所有帖子(包括回帖)的列表,其中帖子作者的ID也在表中,你能快速找出这个传说中的Tango水王吗? 方法一:暴力求解 对每一个ID,遍历整个列表,统计出现的次数,如果超过一半,则说明找到了"水王".复杂度