202009-3(点亮数字人生)

注意将题抽象成一个图的形式,同时注意w[MAXN]的使用来记录通过每个器件时的结果

//每通过一个器件维护依次w[MAXN],用来跟新所有器件的状态,最后得到一个总状态,然后根据相应的要求输出即可。

#include <bits/stdc++.h>
using namespace std;

const int MAXV = 505;//最多500个器件
 
struct Node{
    int v;//边的终点编号
    Node(int _v):v(_v){};//构造函数 
};

vector<Node> G[MAXV];//邻接表
int w[MAXV];//边权重
string type[MAXV];//器件类型
int inDegree[MAXV]={0};//入度
bool initV[MAXV]={false};//判断是否初始化
vector<int> test_in[10005];//最多运行10000次
vector<int> test_out[10005];

bool topologicalSort(int m,int n)
{
    int num = 0;//记录加入拓扑序列的顶点数
    queue<int> q;
    int temp_inDegree[MAXV];//存储临时入度
    memcpy(temp_inDegree,inDegree,(m+n)*sizeof(int));

    for(int i=0;i<m+n;i++)
    {
        if(temp_inDegree[i]==0) q.push(i);
    }
    while(!q.empty())
    {
        int u = q.front();
        q.pop();
        for(int i=0;i<G[u].size();i++)
        {
            int v = G[u][i].v;
            temp_inDegree[v]--;
            if(temp_inDegree[v]==0) q.push(v);
        }
        num++;
    }
    if(num == m+n) return true;
    else return false;
}

void calculate(int m,int n)
{
    queue<int> q;
    int temp_inDegree[MAXV];
    memcpy(temp_inDegree,inDegree,(m+n)*sizeof(int));//将inDegree的内容拷贝到temp_inDegree 

    for(int i=0;i<m+n;i++)
    {
        if(temp_inDegree[i]==0) q.push(i);
    }
    while(!q.empty())
    {
        int u = q.front();
        q.pop();
        for(int i=0;i<G[u].size();i++)
        {
            int v = G[u][i].v;
            temp_inDegree[v]--;

            if(!initV[v])
            {
                w[v] = w[u];
                if(type[v] == "NOT") w[v] = !w[v];
                initV[v] = true;
            }
            else
            {
                if(type[v] == "AND" || type[v] == "NAND") w[v] &= w[u];
                else if(type[v] == "OR" || type[v] == "NOR") w[v] |= w[u];
                else if(type[v] == "XOR") w[v] ^= w[u];
            }

            if(temp_inDegree[v] == 0)
            {
                if(type[v] == "NAND" || type[v] == "NOR") w[v] = !w[v];
                q.push(v);
            }
        }
    }
}

int main(){
    ios::sync_with_stdio(false),cin.tie(0),cout.tie(0);
    int q,m,n;//q表示q个问题,整个电路m个输入,器件数为n
    cin>>q;
    while(q--){
        //初始化
        for(int i=0;i<MAXV;i++){
            for(auto it = G[i].begin();it!=G[i].end();){
                it = G[i].erase(it);
            }//清空邻接表,准备处理下一个问题 
        }
        memset(inDegree, 0, sizeof(inDegree));
        fill(initV, initV + MAXV, false);
        for(int i = 0;i < MAXV;i++)
        {
            type[i].clear();
        }
        for(int i = 0;i < 10005;i++)
        {
            for(vector<int>::iterator j = test_in[i].begin();j != test_in[i].end();)
            {
                j = test_in[i].erase(j);
            }
        }
        for(int i = 0;i < 10005;i++)
        {
            for(vector<int>::iterator j = test_out[i].begin();j != test_out[i].end();)
            {
                j = test_out[i].erase(j);
            }
        }
        
        
        //初始化完成,开始准备正式处理
         
        cin>>m>>n;//输入个数,器件个数
        for(int num = m;num < n + m;num++)//从m开始进行编号 
        {
            string FUNC;//器件描述
            int k;
            cin>>FUNC;
            type[num] = FUNC;
            cin>>k;//对应几个接口相连接 
            for(int i = 0;i < k;i++)
            {
                string L;
                cin>>L;
                int startPoint = atoi(L.substr(1, L.length() - 1).c_str()) - 1;//计算起始点编号
                if(L[0] != 'I')
                {//如果是输出点,则加上输入点的偏移
                    startPoint += m;
                }
                G[startPoint].push_back(Node(num));//构造图
                inDegree[num]++;//计算入度
            }
       }
             int s;//运算次数
             cin>>s;
             for(int i = 0;i < s;i++)
            {//输入数据
            for(int j = 0;j < m;j++)
               {
                int input;
                cin>>input;
                test_in[i].push_back(input);
                }
             }
        for(int i = 0;i < s;i++)//存入需要输出的数据的位置 
           {//输出数据
            int out_num;
            cin>>out_num;
            while(out_num--)
            {
                int output;
                cin>>output;
                output = output + m - 1;
                test_out[i].push_back(output);
            }
        }
        if(topologicalSort(m, n) == false)
        {//有环
            cout<<"LOOP"<<endl;
        }
        else{ 
        for(int i = 0;i < s;i++)//对应的s次运算 
            {
                memset(w, 0, sizeof(w));
                fill(initV, initV + MAXV, false);
                for(int j = 0;j < test_in[i].size();j++)
                {//给初始输入点赋值
                    w[j] = test_in[i][j];
                }
                //计算点权
                calculate(m, n);

                for(int j = 0; j < test_out[i].size();j++)
                {
                    if(j != 0) cout<<" ";
                    cout<<w[test_out[i][j]];
                }
                cout<<endl;

            }
        }
}
    return 0;
} 

 

posted @ 2021-03-07 15:25  zmachine  阅读(116)  评论(0编辑  收藏  举报