当前位置:   article > 正文

微软2016校园招聘9月在线笔试题解

微软2016校园招聘9月在线笔试题解

微软2016校园招聘9月在线笔试题解

题目网址列表:http://hihocoder.com/contest/mstest2015sept2/problems

题目一分析:

问题描述:在二维坐标系中,给定一个圆的圆心坐标x,y,半径r,这个三个数是浮点数,在圆内或者圆边上找一个整数点(x,y坐标都是整数的点)使该点到圆心的距离最大,如果有多个这样的点,选择x坐标最大的点,如果还有多个点存在,则选择y坐标最大的点。

解体思路:通过枚举x,求y的最大值或者最小值。在程序中需要注意y值最大最小值的求法,可以调用cmath包中的函数 floor 和ceil函数,可以看下图进行理解,其中add_y就是程序中的变量。


  1. #include <iostream>
  2. #include <stdio.h>
  3. #include <string.h>
  4. #include <string>
  5. #include <algorithm>
  6. #include <vector>
  7. #include <map>
  8. #include <list>
  9. #include <queue>
  10. #include <cmath>
  11. #define eps 1e-7
  12. using namespace std;
  13. double x,y,r,MaxDis,res_x,res_y;
  14. //计算两点之间的距离
  15. double cal_dis(int x1,int y1)
  16. {
  17. return sqrt((x-x1)*(x-x1) + (y-y1) * (y-y1));
  18. }
  19. //判断新的坐标是否更新已有的坐标
  20. void judge(int x1,int y1)
  21. {
  22. double dis = cal_dis(x1,y1);
  23. if(dis > MaxDis + eps)
  24. {
  25. MaxDis = dis;
  26. res_x = x1;
  27. res_y = y1;
  28. }
  29. else if(abs(dis - MaxDis) < eps)
  30. {
  31. if(x1 > res_x)
  32. {
  33. res_x = x1;
  34. res_y = y1;
  35. }
  36. else if(x1 == res_x && y1 > res_y)
  37. {
  38. res_y = y1;
  39. }
  40. }
  41. }
  42. int main()
  43. {
  44. int cur_x,cur_y;
  45. while(cin >> x >> y >> r)
  46. {
  47. MaxDis = 0;
  48. // 枚举x坐标
  49. for(int i = -(r + 3);i<=r + 2;i++)
  50. {
  51. // 得到当前x的坐标
  52. cur_x = int(x) + i;
  53. //判断x坐标是否超出圆外
  54. if(abs(cur_x - x) >= r + eps)
  55. continue;
  56. double add_y = sqrt(r * r - (cur_x - x) * (cur_x - x));
  57. //获得最大的y轴坐标
  58. cur_y = floor(y + add_y);
  59. judge(cur_x,cur_y);
  60. //cout << cur_x << " ==== " << cur_y << endl;
  61. //获得最小的y轴坐标
  62. cur_y = ceil(y - add_y);
  63. judge(cur_x,cur_y);
  64. //cout << cur_x << " ==== " << cur_y << endl;
  65. }
  66. cout << res_x << " " << res_y << endl;
  67. }
  68. return 0;
  69. }
</pre><pre name="code" class="cpp">
题目二分析:
题目描述:n个城市,每条道路连接两个城市,n-1条道路正好把n个城市连接成一颗树,求解下面sum的值:
<img src="" alt="" />
题目中有对边的权值进行改变后,再要求求解Sum,所以需要进行预处理,使对边权改变时,再次计算的时间复杂度比较低。
解题思路如下:统计每条边被上面的式子计算多少次。计算多少次,可以很容易从下图中看出来。图中绿色这条边把树分成两个子集,所以该边被计算的次数就是N(s1) * N(s2) ,其中N(s1) 表示s1集合点的个数 <span style="font-family: Arial, Helvetica, sans-serif;">N(s1) = 3 ,代码如下:</span>
<img src="" alt="" />
  1. <pre name="code" class="cpp">#include <iostream>
  2. #include <stdio.h>
  3. #include <string.h>
  4. #include <string>
  5. #include <algorithm>
  6. #include <vector>
  7. #include <map>
  8. #include <list>
  9. #include <queue>
  10. #include <cmath>
  11. using namespace std;
  12. typedef long long LL;
  13. #define MAXN 100005
  14. vector<int>next_vec[MAXN];
  15. vector<int>value_vec[MAXN];
  16. //记录每条边的长度,key是两个城市的id,id小的放前面
  17. map<pair<int,int>,int> edge_length_map;
  18. // 记录每条边在所有路径中使用的次数,key是两个城市的id,id小的放前面
  19. map<pair<int,int>,LL> num_map;
  20. int n,m;
  21. bool used[MAXN];
  22. //递归求解每条边被计算几次
  23. int dfs(int id)
  24. {
  25. used[id] = true;
  26. int acc = 1;
  27. int sub_num = 0;
  28. int sub_id;
  29. for(int i = 0;i< next_vec[id].size();i++)
  30. {
  31. sub_id = next_vec[id][i];
  32. //已经访问过不再访问
  33. if(used[sub_id] == true)
  34. continue;
  35. sub_num = dfs(sub_id);
  36. acc += sub_num;
  37. if(id < sub_id)
  38. num_map[make_pair(id,sub_id)] = (LL) sub_num * (n - sub_num);
  39. else
  40. num_map[make_pair(sub_id,id)] = (LL) sub_num * (n - sub_num);
  41. }
  42. return acc;
  43. }
  44. int main()
  45. {
  46. while(scanf("%d%d",&n,&m) == 2)
  47. {
  48. memset(used,0,sizeof(used));
  49. int u,v,k;
  50. for(int i = 0;i<n-1;i++)
  51. {
  52. scanf("%d%d%d",&u,&v,&k);
  53. if(u > v)
  54. swap(u,v);
  55. next_vec[u].push_back(v);
  56. next_vec[v].push_back(u);
  57. value_vec[u].push_back(k);
  58. value_vec[v].push_back(k);
  59. edge_length_map[make_pair(u,v)] = k;
  60. }
  61. dfs(1);
  62. LL sum = 0;
  63. map<pair<int,int>,int>::iterator ite = edge_length_map.begin();
  64. for(;ite!=edge_length_map.end();ite++)
  65. {
  66. sum += num_map[ite->first] * ite->second;
  67. }
  68. char command[10];
  69. for(int i = 0;i<m;i++)
  70. {
  71. scanf("%s",command);
  72. if(command[0] == 'Q')
  73. {
  74. cout << sum << endl;
  75. }
  76. else
  77. {
  78. scanf("%d%d%d",&u,&v,&k);
  79. if(u > v)
  80. swap(u,v);
  81. sum += (LL)(k - edge_length_map[make_pair(u,v)]) * (num_map[make_pair(u,v)]);
  82. edge_length_map[make_pair(u,v)] = k;
  83. }
  84. }
  85. for(int i = 0;i<=n;i++)
  86. {
  87. next_vec[i].clear();
  88. value_vec[i].clear();
  89. }
  90. edge_length_map.clear();
  91. num_map.clear();
  92. }
  93. return 0;
  94. }


 
题目三分析:
题目大意:给定一个序列,求该序列的子序列中有多少是<span style="color: rgb(51, 51, 51); font-family: "Helvetica Neue", Helvetica, Arial, sans-serif; font-size: 16px; line-height: 1.1;">Fibonacci数列的前缀。</span>

解题思路:很容易想到是动态规划,dp[i][j] 表示 序列的前i个数产生的子序列的数量(该子序列需要满足两个条件,1 是fibonacci数列的前缀 2子序列的最后一个数是fibonacci数列中的第j个)。理解状态的含义后,转移方程就比较好理解了,在此不详细说了,具体看程序。在实现中可以省略第一维,有点类似滚动数组。

  1. #include <iostream>
  2. #include <stdio.h>
  3. #include <string.h>
  4. #include <string>
  5. #include <algorithm>
  6. #include <vector>
  7. #include <map>
  8. #include <list>
  9. #include <queue>
  10. #include <cmath>
  11. using namespace std;
  12. typedef long long LL;
  13. #define MAXN 1000005
  14. #define mod 1000000007
  15. int a[MAXN],fib[30];
  16. LL dp[30];
  17. map<int,int> key_to_id;
  18. int main()
  19. {
  20. int n;
  21. scanf("%d",&n);
  22. for(int i = 1;i<=n;i++)
  23. scanf("%d",a+i);
  24. fib[0] =1;
  25. fib[1] = 1;
  26. int MaxNum = 2;
  27. int index = 2;
  28. while(MaxNum)
  29. {
  30. fib[MaxNum] = fib[MaxNum-1] + fib[MaxNum-2];
  31. key_to_id[fib[MaxNum]] = index++;
  32. if(fib[MaxNum] > 100005)
  33. break;
  34. MaxNum++;
  35. }
  36. memset(dp,0,sizeof(dp));
  37. LL sum = 0;
  38. int id;
  39. for(int i= 1;i<=n;i++)
  40. {
  41. if(a[i] == 1)
  42. {
  43. //当是1时,特殊处理
  44. sum = (sum + 1 + dp[0]) %mod;
  45. dp[1] += dp[0];
  46. dp[0] += 1;
  47. dp[1] %= mod;
  48. dp[0] %= mod;
  49. }
  50. else if(key_to_id.find(a[i]) != key_to_id.end())
  51. {
  52. // 是fibonacci 数列中的数时,进行如下处理
  53. id = key_to_id[a[i]];
  54. sum += dp[id -1];
  55. dp[id] += dp[id -1];
  56. dp[id] %= mod;
  57. sum %= mod;
  58. }
  59. // 如果不是fibonacci 数列中的数,不需要处理
  60. }
  61. cout << sum << endl;
  62. return 0;
  63. }


题目四分析:

题目大意:给定两个NXN的矩阵,对其中一个矩阵进行顺时针旋转操作(旋转操作有四种,90度,180度,270度,360度顺时针旋转),当N为偶数时,把矩阵分成相同大小的四块,再进行旋转操作,一直递归到N为奇数时,问第一个矩阵能否通过旋转与第二个矩阵完全相同。

解题思路:编写一个rotate函数实现矩阵90度旋转(其他都可以由它得到),第二个就是递归的实现,记得递归时的状态还原

  1. #include <iostream>
  2. #include <stdio.h>
  3. #include <string.h>
  4. #include <string>
  5. #include <algorithm>
  6. #include <vector>
  7. #include <map>
  8. #include <list>
  9. #include <queue>
  10. #include <cmath>
  11. using namespace std;
  12. typedef long long LL;
  13. #define MAXN 105
  14. #define mod 1000000007
  15. int matri_a[MAXN][MAXN],matri_b[MAXN][MAXN];
  16. // 对二维数组中的某一正方形小块顺时针旋转90度
  17. void Rotate(int x1,int y1,int x2,int y2)
  18. {
  19. int len = x2 - x1 + 1;
  20. int **a = new int*[len];
  21. for(int i = 0 ;i< len;i++)
  22. a[i] = new int[len];
  23. for(int i = 0;i<len;i++)
  24. {
  25. for(int j = 0;j<len;j++)
  26. {
  27. a[j][len - 1 - i] = matri_a[x1 + i][y1 + j];
  28. }
  29. }
  30. for(int i = 0;i<len;i++)
  31. {
  32. for(int j = 0;j<len;j++)
  33. {
  34. matri_a[x1 + i][y1 + j] = a[i][j];
  35. }
  36. }
  37. delete a;
  38. }
  39. //判断两个矩阵对应小方块的元素是否都相等
  40. bool Is_same(int x1,int y1,int x2,int y2)
  41. {
  42. int len = x2 - x1 + 1;
  43. for(int i = 0;i<len;i++)
  44. {
  45. for(int j = 0;j<len;j++)
  46. {
  47. if(matri_a[x1 + i][y1 + j] != matri_b[x1 + i][y1 + j])
  48. return false;
  49. }
  50. }
  51. return true;
  52. }
  53. bool dfs(int x1,int y1,int x2,int y2)
  54. {
  55. int len = x2 - x1 + 1;
  56. int i = 0;
  57. bool ret = false;
  58. for(;i<4;i++)
  59. {
  60. Rotate(x1,y1,x2,y2);
  61. //直接旋转看能否与B矩阵保持一致,该方法是一种剪枝策略
  62. ret |= Is_same(x1,y1,x2,y2);
  63. // 如果直接旋转能保持一致,则不需要求解子问题,如果不加剪枝,可能会超时
  64. if(ret)
  65. break;
  66. if(len %2 == 0)
  67. {
  68. int tmp = len /2 -1;
  69. // 求解子问题,如果子问题都能解,则该问题也能解
  70. ret |= (dfs(x1,y1,x1 + tmp ,y1 + tmp) && dfs(x1 ,y1 + tmp + 1,x1 + tmp,y2) &&
  71. dfs(x1 + tmp + 1,y1,x2,y1 + tmp) && dfs(x1 + tmp +1 ,y1 + tmp +1,x2,y2));
  72. }
  73. if(ret)
  74. break;
  75. }
  76. // 状态还原,但是程序中不加如下几句代码也能够AC,是这个问题具有特殊性,不还原也是可以的,但是对于一般问题,还是按要求写比较好
  77. while(i++<3)
  78. {
  79. Rotate(x1,y1,x2,y2);
  80. }
  81. return ret;
  82. }
  83. int main()
  84. {
  85. int t,n;
  86. scanf("%d",&t);
  87. while(t--)
  88. {
  89. scanf("%d",&n);
  90. for(int i = 0 ;i<n;i++)
  91. for(int j = 0 ;j<n;j++)
  92. scanf("%d",matri_a[i] + j);
  93. for(int i = 0 ;i<n;i++)
  94. for(int j = 0 ;j<n;j++)
  95. scanf("%d",matri_b[i] + j);
  96. bool is_ok = dfs(0,0,n-1,n-1);
  97. if(is_ok)
  98. cout << "Yes" << endl;
  99. else
  100. cout << "No" << endl;
  101. }
  102. return 0;
  103. }



声明:本文内容由网友自发贡献,转载请注明出处:【wpsshop】
推荐阅读
相关标签
  

闽ICP备14008679号