1055 集体照 (25 分)

题目:1055 集体照 (25 分)

拍集体照时队形很重要,这里对给定的 N 个人 K 排的队形设计排队规则如下:

  • 每排人数为 /(向下取整),多出来的人全部站在最后一排;

  • 后排所有人的个子都不比前排任何人矮;

  • 每排中最高者站中间(中间位置为 /,其中 m 为该排人数,除法向下取整);

  • 每排其他人以中间人为轴,按身高非增序,先右后左交替入队站在中间人的两侧(例如5人身高为190、188、186、175、170,则队形为175、188、190、186、170。这里假设你面对拍照者,所以你的左边是中间人的右边);

  • 若多人身高相同,则按名字的字典序升序排列。这里保证无重名。

现给定一组拍照人,请编写程序输出他们的队形。

输入格式:

每个输入包含 1 个测试用例。每个测试用例第 1 行给出两个正整数 N(≤,总人数)和 K(≤,总排数)。随后 N 行,每行给出一个人的名字(不包含空格、长度不超过 8 个英文字母)和身高([30, 300] 区间内的整数)。

输出格式:

输出拍照的队形。即K排人名,其间以空格分隔,行末不得有多余空格。注意:假设你面对拍照者,后排的人输出在上方,前排输出在下方。

输入样例:

10 3
Tom 188
Mike 170
Eva 168
Tim 160
Joe 190
Ann 168
Bob 175
Nick 186
Amy 160
John 159

输出样例:

Bob Tom Joe Nick
Ann Mike Eva
Tim Amy John

思路:

  • 根据第二点条件可以知道,就是在排队前先将他们按身高从高到低排,同身高得按字母序。用排序加比较函数即可解决。
  • 每排中最高的人排在 m/2+1 的位置,按照第四点条件,又以完成按身高排序,那么这就不需要特列,因为最高的人第一个被排,后面的人在他左右排,最后每排中这个最高的人自然而然地就站到了 m/2+1 的位置,这也算是一种规律吧。
  • 本体代码的关键在于如何给每排排好队。我排队没有用到辅助数组,而是直接将排好队的队伍拿来排,直接排在自己的位置,这样就可以直接输出了(这样的话,就需要找到排队的规律),但也正是因为这样,代码就显得有点冗长(主要是没有改进的原因)。
    排队规律:
      1.按题目的排法,以每排最高的人排中间,剩下的左右分配,那么最高的人的左边身高是从矮到高的,而右边是从高到矮站的。这点可以利用。其次是左右排,那不就相当于是隔一个人排吗。
         2.要是是日常生活,直接报2出列,排成两排后拼接成一排就可以了。代码实现的话得判断每排人数的奇偶(除了最后一排会有特殊情况,其他排的奇偶性是一致的,所以其实只用判断一次),分奇偶的原因:奇数个人的话,隔一人取人,最后一个人就是排右边的;偶数则相反,最后一个人排左边。
      3.举个例子加以理解上述文字(纯文字看着会枯燥,还容易绕):
        5人身高为190、188、186、175、170。按我说的排队规律就是:先判断奇偶,排成一排为奇数个,那么从倒数第二个取起,为左边第一个人,隔一个取,那么排序的结果为175、188.排完后取第一个人(最高的)继续后排顺序就是190、186、170.两者合在一起就是一排排完序后的顺序。1位175,二位188,三位190,四位186,五位(最后一位)170.

代码:

 1 #include 
 2 #include 
 3 #include 
 4 #include 
 5 #include 
 6 #include 
 7 #include 
 8 #include <string>
 9 #include 
10 #include 
11 #include 
12 #include 
13 using namespace std;
14 
15 struct node{
16     string s;
17     int h;
18 }per[10005];
19 
20 bool cmp(node a, node b)
21 {
22     if(a.h == b.h)
23         return a.s < b.s;
24     else
25         return a.h > b.h;
26 }
27 
28 int main()
29 {
30     int n, k;
31     scanf("%d %d", &n, &k);
32     for(int i = 0; i < n; i++)
33     {
34         cin >> per[i].s >> per[i].h;
35     }
36     sort(per, per + n, cmp);
37     int m = n / k;    //各排的人数 
38     int fm = m + n % k;    //最后一排的人数 
39     for(int i = 0; i < k; i++)
40     {
41         if(i == 0 && n%k)  //若有人多出放最后一排特殊处理
42         {
43             for(int j = fm-1; j > 0; j-=2)    //输出 左列 
44             {
45                 cout << per[j].s;
46                 printf(" ");
47             }
48             for(int j = 0; j < fm; j+=2)    //输出 右列 
49             {
50                 cout << per[j].s;
51                 if(j != fm-2)
52                     printf(" ");
53             }
54             printf("\n");
55         }
56         else
57         {
58             if(m % 2) //如果一排人数奇数个
59             {
60                 for(int j = fm+i*m-2; j > fm+(i-1)*m; j-=2)    //输出 左列 
61                 {
62                     cout << per[j].s;
63                     printf(" ");
64                 }
65                 for(int j = fm+(i-1)*m; j < fm+i*m; j+=2)    //输出 右列 
66                 {
67                     if(j != fm+(i-1)*m)
68                         printf(" ");
69                     cout << per[j].s;
70                 }
71                 printf("\n");
72             } 
73             else  //每排偶数个
74             {
75                 for(int j = fm+i*m-1; j > fm+(i-1)*m; j-=2)    //输出 左列 
76                 {
77                     cout << per[j].s;
78                     printf(" ");
79                 }
80                 for(int j = fm+(i-1)*m; j < fm+i*m; j+=2)    //输出 右列 
81                 {
82                     if(j != fm+(i-1)*m)
83                         printf(" ");
84                     cout << per[j].s;
85                 }
86                 printf("\n");
87             }            
88         }
89     }
90     return 0;
91 }

 

总结:

  • 每排人数没分奇偶,只有测试点3、5过了。其他基本显示格式错误,没分奇偶影响格式,就是空格了吧。确实如此。
  • 分奇偶后,测试点1答案错误,2格式错误。后来想可能是全部排成一排和每排只有一个人的情况出错。将最后一排按有人多出时再特殊处理后就AC了。(也不是很清楚为啥突然就A了,等二刷)

  打代码就像写文章,不断修改才能不断加深功底,思路是框架,填充好细节才成文。真正的快乐来源于不断的尝试与失败后终于迎来的成功。

你可能感兴趣的:(1055 集体照 (25 分))