1.题目描述:点击打开链接
2.解题思路:本题是一道模拟题。要求找出一个队列,使得队列中的每个人的握手次数符合输入的情况,如果不存在输出无解。其中每三个人可以在任何时刻组成一个队伍开始比赛,后面的人不能再与他们握手。通过手动操作与观察可以发现,如果把队列中的人的握手次数均取除以3后的余数。那么正好形成0 1 2 0 1 2....的循环序列。所以可以通过这个来模拟该过程。事先用map保存握手次数i下对应的列表。接下来用一个变量j控制所到达的握手次数,如果发现j≥3且此时的队列为空,那么就缩小3看是否存在队列,否则继续减小。如果当j<3,队列仍为空且总人数不为零,说明无解,因为这个位置已经找不到人来填补了。
3.代码:
#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
#include<algorithm>
#include<string>
#include<sstream>
#include<set>
#include<vector>
#include<stack>
#include<map>
#include<queue>
#include<deque>
#include<cstdlib>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<ctime>
#include<functional>
using namespace std;
typedef long long ll;
map<int, vector<int> >base;
int n;
#define N 200000+10
int a[N];
int main()
{
//freopen("t.txt", "r", stdin);
while (~scanf("%d", &n))
{
base.clear();
memset(a, 0, sizeof(a));
int maxd = 0;
for (int i = 0; i < n; i++)
{
int x; cin >> x;
base[x].push_back(i + 1);
}
int cur = 0;
int j = 0, cnt = n;
while (cnt>0)
{
if (j<3&&base[j].empty()){ puts("Impossible"); goto x1; }//已经不能再减小了,但队列仍旧没人,而总人数还不为零,说明无解
int p = base[j].back(); base[j].pop_back(); cnt--;
a[cur++] = p; j++;
while(j >= 3 && base[j].empty())j -= 3;//每次都减小3,看队列中是否有人
}
puts("Possible");
for (int i = 0; i < cur; i++)
printf("%d%c", a[i], i == cur - 1 ? '\n' : ' ');
x1:;
}
return 0;
}