title: 情报传递 河南省第十届省赛
tags: [省赛]

题目描述:

抗日战争时期,在国共合作的大背景下,中共不断发展壮大,其情报工作也开始由获取警报性、保卫性信息,向获取军政战略性情报转变。各系统情报组织遵循"荫蔽精干,长期埋伏,积蓄力量,以待时机"的隐蔽战线工作方针,开展了卓有成效的情报工作。

***特科组是中共情报工作的一个杰出范例,它以点为主、系统延伸、分散辐射的力量格局,异地领导、分头派遣、单线联系的组织形式。以打入、拉出、统战联络、内线为主的工作方式,形成了一个传递信息隐蔽、效用及时、形式多样的情报网络。

***特科组的情报人员共有N人,其代号分别为0,1,……,N-1。 0号是最高领导人,特工之间有一套严格的单线联络程序,即,每个特工人员只有一个上线,他获得的情报需层层上传递到0号手里,由0号发报出去。

特工i在传递情报时,若通往到0号的通道尚未建立,则需要建立一级级单线通道;若他的上线已建立好通道,只需建立两人通道,信息发送给上线;依次类推。若特工i已建立好到0号的通道,则直接发出情报。日伪统治中心南京,既是情报来源丰富的地方,又是特工人员活动最危险的地方。因此,一旦某个特工处于不安全状态,他必须马上撤离,同时他的所有下线(处在通道上的一级级下线)也一同撤离。

已知***特科组的组织结构,你的任务是计算,当某特工i需要发送情报时,最少需要建立几个情报人员的通道;当某特工i处于不安全状态时,最少需要撤离多少人员。

输入:

第一行一个整数:  N           ( 1≤N ≤5060 )

接下来一行有N-1 个整数,  A1 A2 ……An-1  ,其中Ai是编号i的上线。

下一行一个整数:  M         表示有M个状态,( 1≤M ≤5060 )

接下来有M行 :有两种形式:  Send i    特工i处于要发送情报状态;

Danger i  特工i处于不安全状态

输出:

输出占M行 ,对于Send i,输出最少需要建立通道的情报人员数,若特工i处于通道线上,输出0;对于Danger i,输出最少需要撤离多少人员,若特工i不处于通道线上,则输出0.

样例输入:

10
0 1 2 1 3 0 0 3 2
10
Send 0
Send 3
Danger 2
Send 7
Send 5
Send 9
Danger 9
Send 4
Send 1
Send 9

样例输出:

1
3
2
1
3
1
1
1
0
1

分析:

首先根据题意可以明确的一点就是一个点的上级如果有的话只能有唯一的一个,但是一个点却可以有许多的下级,相当于是一颗树的结构。

简单的开一个一维的数组来保存每一个点的直接上线,由于每个点的下线个数不确定有多少个,所以要用一个动态的vector数组来存,然后就是需要一个bool类型的数组来标记当前的点是否有连通。

如果是Send的话,就相当于是一级一级的往上找,也就是依次访问上级,直到找到0,但是0奔上也要向上建立一个联系,因为题上的数据Send 0结果是1;

如果说DAnger的话,也就是相当于一个广搜的过程,把所有的这个点的孩子节点全部断开。

代码:

#include<iostream>
#include<vector>
#include<queue>
#include<stdio.h>
#include<sstream>
using namespace std;
int n, m;
int up[5100];///上级
bool build[5100];///当前的点有没有畅通
vector<int> down[5100];///下级

int Send(int s)
{
    int ret = 0;  /// 顺着s一直找上级,一直找到0
    while (s)
    {
        if (!build[s])
        {
            build[s] =true;
            ret++;
        }
        s = up[s];
    }
    /// 注意0本身也要向上建立通道。
    /// 因为样例的第一行 Send 0  输出为 1
    if (!build[0])
    {
        build[0] = true;
        ret++;
    }
    return ret;
}

int Danger(int s)///撤退的时候就相当于是一个广搜的过程
{
    int ret = 0;
    queue<int> q;
    q.push(s);
    while (!q.empty())
    {
        ///从队列中拉取一个节点
        s = q.front();
        q.pop();
        ///看 s 自己是否向上级建立了通道
        if (build[s])
        {
            ret++;
            build[s] =false;
        }
        ///遍历 s 所有的下级
        for (int i = 0; i < down[s].size(); ++i)
        {
            /// 看这个下级是否建立了通道
            if (build[down[s][i]])
                q.push(down[s][i]);
        }
    }
    return ret;
}

int main()
{
    int i;
    scanf("%d",&n);
    for (i = 1; i <n; ++i)
    {
        scanf("%d", &up[i]);     /// 设置 i的上级
        down[up[i]].push_back(i); /// 设置属于up[i] 的下级
    }
    char buffer[20];
    scanf("%d",&m);
    for (i = 0; i <m; ++i)
    {
        int num;
        scanf("%s", buffer);
        scanf("%d", &num);
        ///只用比较第一个字符就可以知道是哪条命令
        if (buffer[0] =='S')
            printf("%d\n", Send(num));
        else
            printf("%d\n", Danger(num));
    }
    return 0;
}
posted on 2017-05-22 08:13  渡……  阅读(177)  评论(0编辑  收藏  举报