题意:有一个公司,每天有员工进出,$a[i]>0$时表示$a[i]$这个员工进入公司,$a[i]<0$时表示$-a[i]$这个员工出公司,公司对进出办公室有一些严格的规定

  • 员工每天最多只能进入一次办公室
  • 如果那天他没有进办公室的话,他显然不能离开
  • 每天开始和结束时,办公室都是空的(员工不能呆在晚上),办公室也可能在一天中的任何时候都是空的

现在给你序列$a$($a[i] \neq 0$),问你是否能够把数组$a$分为几个相邻的子数组,使得每一个子数组员工的进出情况符合要求并输出每一个子数组的长度,或者输出$-1$表示不可能

思路:用一个数组$mk$标记某个人状态,$mk[i]=0$表示$i$号员工在这一天内还没有操作,$mk[i]=1$表示$i$号员工在这一天内已经进入办公室,$mk[i]=-1$表示$i$号员工在这一天内已经进入办公室并且出去了,$num$记录此时办公室的人数,如果$num=0$时则可以进入新的一天,同时需要用一个数组$p$记录这一天已经出去的员工编号,在进入新的一天时,则需要将这些员工的$mk[i]$变为$0$,中间出现不合法的情况直接跳出循环,输出$-1$

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
#include <iostream>
#include <algorithm>
#include <cstdio>

using namespace std;

const int N = 100010;
const int M = 1000010;

int n, a[N], mk[M];
int num, cnt, p[N];
int res[N], pos;

int main()
{
scanf("%d", &n);
for (int i = 1; i <= n; i++) scanf("%d", &a[i]);
int flag = 1; res[++pos] = 0;
for (int i = 1; i <= n; i++) {
if (a[i] > 0) {
if (0 == mk[a[i]]) {
mk[a[i]] = 1, num++;
}
else {
flag = 0; break;
}
}
else {
if (1 == mk[-a[i]]) {
mk[-a[i]] = -1, num--;
p[++cnt] = -a[i];
if (0 == num) {
for (int j = 1; j <= cnt; j++) mk[p[j]] = 0;
cnt = 0, res[++pos] = i;
}
}
else {
flag = 0; break;
}
}
}
if (0 != num) flag = 0;
if (!flag) printf("-1\n");
else {
printf("%d\n", pos - 1);
for (int i = 2; i <= pos; i++) {
printf("%d", res[i] - res[i - 1]);
printf(i == pos ? "\n" : " ");
}
}
return 0;
}