腾讯2014软件开发笔试题目

news/2024/7/21 4:15:28 标签: C++, 面试笔试, 腾讯

腾讯2014软件开发笔试题目

                                                                    -----9月21日,腾讯2014软件开发校招-简答题-广州

简答题:

1、请设计一个排队系统,能够让每个进入队伍的用户都能看到自己在 中所处的位置和变化。队伍可能随时有人加入和退出,当有人退出影响到用户的位置排名时需要即时反馈到用户。

2、A、B两个整数集合,设计一个算法求他们的交集,尽可能的高效。

(博主能力有限,不是所有题目都会求解,第1题不是我的擅长,这里贴出来让大家知道腾讯的考题。我的重点放在第2题上面!)

 

第2题  题解(个人见解,仅供参考!)

思路1:排序法

  对集合A和集合B进行排序(升序,用快排,平均复杂度O(N*logN)),设置两个指针p和q,同时指向集合A和集合B的最小值,不相等的话移动*p和*q中较小值的指针,相等的话同时移动指针p和q,并且记下相等的数字,为交集的元素之一,依次操作,直到其中一个集合没有元素可比较为止。

  优点:操作简单,容易实现。

  缺点:使用的排序算法不当,会耗费大量的时间,比如对排好序的集合使用快排, 时间复杂度是O(N2)

  这种算法是大家都能比较快速想到的办法,绝大多数时间放在了对集合的排序上,快排的平均复杂度是O(N*logN),对排好序的集合做查找操作,时间复杂度为O(N),当然这种算法肯定比遍历要快多了。

code:

 

[cpp]  view plain copy
  1. #include <stdio.h>  
  2. #include <stdlib.h>  
  3. #define M 8  
  4. #define N 5  
  5. int cmp(const void *a, const void *b)  
  6. {  
  7.     int *x = (int *)a;  
  8.     int *y = (int *)b;  
  9.     return (*x) - (*y);  
  10. }  
  11.   
  12. int main(void)  
  13. {  
  14.     int A[] = {-1, 2 ,39 ,10, 6, 11, 188, 10};  
  15.     int B[] = {39 ,8 , 10, 6, -1};  
  16.     //对数组A和数组B进行快排  
  17.     qsort(A, M, sizeof(int), cmp);  
  18.     qsort(B, N, sizeof(int), cmp);  
  19.     //FindIntersection(A, B);  
  20.     int i = 0, j = 0;  
  21.     int cnt = 0;  
  22.     int result[M > N ? M : N];//保存集合的结果  
  23.     //设置i、j索引,分别指向数组A和B,相等则同时移动,不相等则移动较小值的索引  
  24.     while(i < M && j < N)  
  25.     {  
  26.         if(A[i] == B[j])  
  27.         {  
  28.             result[cnt] = A[i];  
  29.             i++;  
  30.             j++;  
  31.             cnt++;  
  32.         }  
  33.         else if(A[i] < B[j])  
  34.         {  
  35.             i++;  
  36.         }  
  37.         else  
  38.         {  
  39.             j++;  
  40.         }  
  41.     }  
  42.     for(i = 0; i < cnt; i++)  
  43.     {  
  44.         printf("%4d", result[i]);  
  45.     }  
  46.     return 0;  
  47. }  

 

 

思路2:索引法 

以空间换时间,把集合(感谢网友的指正,集合里面的元素是不重复的!)中的元素作为数组下表的索引。来看例子:      

A= {1 ,12, 13, 25},那Asub[1] = 3,Asub[12] = 1 ,Asub[13] = 1 ,Asub[25] = 1 ;

B={1, 2,  3, 15 ,}那Bsub[1] = 1; Bsub[2] = 1; Bsub[3] = 1; Bsub[15] = 1;

  对元素少的集合扫一遍,发现Asub[1] = 3 和Bsub[1] = 1有相同的索引1,并且重复度为1,所以交集肯定包括{1, 1}; Bsub[2] = 1而Asub[2] = 0,表示无交集,依次类推,可以得到集合A和B的交集。

  假设集合中存在负数,可以把集合分成正整数和负整数(加个负号变正整数)两部分,解法同上!

  优点:速度快,时间复杂度O(N)

  缺点:空间消耗大,以空间换取时间

  这是我看到题目第一个想到的算法,再来想到排序法,而集合压缩是有感而发的,索引法的缺点是空间消耗多,原因是可能索引值太大,要申请很多的不必要的空间,这个缺点也是有克服的方法的,就是采用哈希查找,找到一个比较合适的哈希函数,把索引的值减小了,从而减少消耗的内存空间。比如哈希函数为f(x) = (x + MOD) % MOD 除留余数法,MOD为常数),还有平方取中法、折叠法等方法,然而,无论哈希函数设计有多么精细,都会产生冲突现象,也就是2个关键字处理函数的结果映射在了同一位置上,因此,有一些方法可以避免冲突。这里没有仔细钻研,只提供一些思路,有兴趣的朋友可以继续研究。

code:(我的代码仅适用与正整数部分,未处理负数)

 

[cpp]  view plain copy
  1. /* 
  2.     Tencent: A、B两个整数集合,设计一个算法求他们的交集,尽可能的高效 
  3. */  
  4. #include <stdio.h>  
  5. #include <stdlib.h>  
  6. #include <string.h>  
  7. #define M 6  
  8. #define N 5  
  9. int Mymin(int a, int b)  
  10. {  
  11.     return a < b ? a : b;  
  12. }  
  13. int main(void)  
  14. {  
  15.     int A[] = {1, 10, 12, 23, 5, 45};  
  16.     int B[] = {1, 10, 12, 123, 52};  
  17.   
  18.     //find MaxNumber in A  
  19.     int ifindA = 0;  
  20.     int MaxInA = A[0];  
  21.     for(ifindA = 0; ifindA < M; ifindA++)  
  22.     {  
  23.         MaxInA = MaxInA > A[ifindA] ? MaxInA : A[ifindA];  
  24.     }  
  25.     //find MaxNumber in B  
  26.     int ifindB = 0;  
  27.     int MaxInB = 0;  
  28.     for(ifindB = 0; ifindB < M; ifindB++)  
  29.     {  
  30.         MaxInB = MaxInB > A[ifindB] ? MaxInB : A[ifindB];  
  31.     }  
  32.   
  33.     int *AsubPositive = (int *)malloc(sizeof(int) * (MaxInA + 1));  
  34.     int *BsubPositive = (int *)malloc(sizeof(int) * (MaxInB + 1));  
  35.     memset(AsubPositive, 0, sizeof(int) * (MaxInA + 1));  
  36.     memset(BsubPositive, 0, sizeof(int) * (MaxInB + 1));  
  37.   
  38.   
  39.     //COPY Positive and Negative numbers of A  
  40.     int i = 0;  
  41.     for(i = 0; i < M; i++)  
  42.     {  
  43.         AsubPositive[A[i]]++;  
  44.     }  
  45.     //COPY Positive and Negative numbers of B  
  46.     int j = 0;  
  47.     for(j = 0; j < N; j++)  
  48.     {  
  49.         BsubPositive[B[j]]++;  
  50.     }  
  51.   
  52.     int  k = 0;  
  53.     int icount = 0;  
  54.     //扫描AsubNegative和BsubPositive  
  55.     printf("the Intersection of A and B is : { ");  
  56.     for(k = 0; k < M; k++)  
  57.     {  
  58.         //有交集输出该数  
  59.         icount = Mymin(AsubPositive[A[k]], BsubPositive[A[k]]);  
  60.         if(icount == 1)  
  61.         {  
  62.             printf("%-3d",A[k]);  
  63.         }  
  64.         A[k] = 0;  
  65.     }  
  66.     printf(" }");  
  67.     return 0;  
  68. }  

思路3:集合压缩

 

  对于一个集合来说,我们很容易就可以得到集合的最大值和最小值,假设集合A的最大值和最小值分别为MaxInA,MinInA;假设集合B的最大值和最小值分别为MaxInB,MinInB;那么集合A的所有元素一定在闭区间【MinInA, MaxInA】里面,集合B的所有元素一定在闭区间【MinInB, MaxInB】里面,从这两个集合里面我们可以作如下判断:(集合A和集合B都在链表中!此算法使用链表结构,操作起来比数组更方便)

  1. 若MinInA == MinInB或者MaxInA == MaxInB,那么MinInA 或者MaxInA (相等的那个数)就一定在交集里面,存入交集(可以用数组存),删除链表中相应的结点;若不想等则跳到第3步;

  2. 重新找到集合A和B中的最大值和最小值MinInA 、MaxInA 、MinInB、MaxInB;跳回第1步;

  3. 更新区间(交集的区间),区间的更新如下:区间下界为Lower = max(MinInA, MinInB),上届为Upper = min(MaxInA , MaxInB),那么剩下的交集一定在闭区间【Lower ,Upper】里面,按照这个区间来剔除掉集合A和集合B中不符合条件的元素,剔除结束后,若其中一个集合为空,跳到第4步,否则返回第2步;

  4. 程序结束,退出!

  这种适用于集合里面数值比较散乱,最大值最小值差值比较大的情况!算法的思想在于不断减小搜索的范围,时间的消耗主要在查找集合的最大值和最小值上,我们来看一个例子,集合A= {1, 3, 10, 100, 123, 0, 6} ,B = {3, 2, 10, 23, -1},

  集合A的闭区间【0, 123】,集合B的区间【-1,23】,交集的闭区间就为【0,23】,按照这个区间,剔除集合A中的{ 100, 123},剔除集合B的{-1},集合A={1, 3, 10, 0, 6}集合B={3, 2, 10, 23},没有相等的,继续缩小范围,为【2,10】,这时MaxInA == MaxInB,满足条件,把10存入交集数组中,剔除两个集合的结点;集合变为A= {3,6}集合B={3},满足MinInA == MinInB或者MaxInA == MaxInB,把3存入交集数组中,集合B为空,结束!如图:

  对于第三个方法,我只是把算法的思想做了一下总结,并没有编写代码运行调试并与其他算法做比较!比较过的朋友,欢迎告知三种算法的优劣性!

题目部分摘取自july CSDN网站:http://blog.csdn.net/v_july_v/article/details/11921021 

注:1.本文版权归作者和CSDN所有,未经允许不得转载,侵权必究!

2.本博客与博客园上的博客为同一博客主:http://www.cnblogs.com/bestDavid/

后记:  

  只要是算法,就无法同时解决时间复杂度和空间复杂度这一矛盾,我们只能具体问题具体分析,根据实际情况选取最合适的算法,尽量保持程序高效的执行效率!我的写代码能力和算法能力只能算初学者级别,所以在贴出的代码中可能有许多漏洞,朋友们若是有什么建议,请多多给与我更多的指教!在这里发表一下自己的看法,多谢支持!




http://www.niftyadmin.cn/n/951308.html

相关文章

java算时差,java计算时间差及比较时间

比如&#xff1a;现在是2004-03-26 13&#xff1a;31&#xff1a;40过去是&#xff1a;2004-01-02 11&#xff1a;30&#xff1a;24我现在要获得两个日期差&#xff0c;差的形式为&#xff1a;XX天XX小时XX分XX秒方法一&#xff1a;DateFormat df new SimpleDateFormat("…

科大讯飞2014届实习生招聘笔试题

说明&#xff1a;考试时间为60分钟。 题目是笔者刚考完回忆起来的&#xff0c;答案也只是笔者的一些见解&#xff0c;有不对的地方望大家指教。 1. 已知二叉树的前序遍历为ABCDEFGHIJ&#xff0c;中序遍历为CBEDAHGIJF&#xff0c;请画出其二叉树结构。 2.求一个整数数组的最大…

api.php wechat,微信Api的集成

如何集成?需要2样东西: IocBy配置和一个properties文件IocBy配置, 当然就是MainModule类了IocBy(args{"*js", "ioc/","*anno", "net.wendal.nutzbook","*weixin", // 仅需要添加这一行,引用的是org.nutz.plugins.weixin.We…

阿里巴巴2014校招笔试题-2013年9月14日

不得不吐槽&#xff0c;阿里真是太混乱了&#xff0c;北京的笔试在考场等了两个半小时&#xff0c;考卷都没运到考场&#xff0c;阿里巴巴集团校园招聘 回应说&#xff1a;“北京的同学们&#xff0c;简单解释下&#xff0c;为了试卷的保密&#xff0c;印刷的时间都比较晚&…

php session 加密,cookie和session的加密?

我现在写的前后和后台登陆大量使用了cookie和session&#xff0c;但是我直接就放进去了&#xff0c;读取也直接就读取出来了&#xff0c;最近听老师讲课提到cookie和session加密&#xff0c;我不知道现在正统做法是怎么的&#xff1f;是cookie和session一概都加密&#xff1f;还…

2014阿里巴巴笔试题

题目&#xff1a;两棵二叉树T1和T2&#xff0c;T1的节点数是百万量级&#xff0c;T2的节点数一千以内&#xff0c;请给出判断T2是否T1子树的可行算法。 分析&#xff1a; 首先想到的是递归&#xff0c;但是T1的数量级太大&#xff0c;递归会导致栈溢出&#xff0c;于是以非递归…

php变量名语法,PHP语法小结之基础和变量

本系列文章&#xff0c;我们将简单的为大家总结一下PHP之中语法知识&#xff0c;第一篇&#xff0c;我们先来介绍基础和变量&#xff0c;希望大家能够喜欢。最近有个H5项目的需求&#xff0c;需要服务端&#xff0c;考察过后决定用PHP实现一个HTTP服务端&#xff0c;于是开始重…