解题思路:模拟

本题中,被判定为作弊有两种方式:即(a)互相回答问题判定;(b)提问被2个以上作弊者回答。整体思路是:先找到所有(a)类作弊者,再一步步找出(b)类作弊者。

使用user_to_user,users,questions三个字典,以及集合ans,它们的用途如下:

  • user_to_user:记录每个用户 id 回答过谁的问题。用于快速找出(a)类作弊者
  • questions:记录每个问题的提问者、以及回答者中已经被标记为作弊的用户数量。当被标记次数加至2时,提问者判定为(b)类作弊者
  • users:记录每个id回答过问题的序号。用于发现新的作弊者后,快速定位并标记其回答过的问题。
  • ans:记录已经被标记为作弊者的id。

模拟的过程如下:

1. 对于输入的每个问题,找出提问者 q 和若干个回答者。然后:

1.1. 对于每个回答者 x,将问题的序号 i 放入 users[x] 中

1.2. 如果 q 已经回答过 x 的问题,他们在第一轮同时判定为作弊。否则将 x 放入 user_to_user[q] 中

1.3. 初始化 questions 中的问题 i :记录下提问者 q ,被标记答者数初始化为 0

2. 第一轮被标记为作弊的 id 已经被记录在 ans 中。可以开始第二轮标记。具体做法是:对于每个作弊者 x 和其回答过的每个问题 q,问题 q 被标记一次。然后如果这是 q 第二次被标记,那么 q 的提问者将在第二轮被判定为作弊

3. 重复 2 的过程,直到没有新的作弊者被判定为作弊。

from collections import defaultdict
def solve():
    ans = set()
    user_to_user = defaultdict(set) # i提问被谁回答过
    users = defaultdict(list)       # id, questions
    questions = {}                  # id: [user, cheater_count]
    for i in range(int(input())):
        ipt = [int(x) for x in input().split()]
        q, a = ipt[0], set(ipt[2:])
        for x in a:                 # i:问题序号,q: 提问者,x:回答者
            users[x].append(i)
            if q in user_to_user[x]:
                ans |= {q, x}
            elif q != x:
                user_to_user[q].add(x)
        questions[i] = [q, 0]
    s = ans
    while s:
        ss = set()      # 下轮即将更新的作弊者
        for x in s:
            for q in users[x]:
                questions[q][1] += 1     # cheater_count 增加 1
                if questions[q][1] == 2:
                    ss.add(questions[q][0])
        ans |= ss
        s = ss
    print(len(ans))
    if ans:
        print(str(list(ans)).replace(',','')[1:-1])

while 1:
    try:
        solve()
    except EOFError:
        break