CCF-CSP-202009-3:点亮数字人生(题解)

问题描述

CCF CSP 202009-3

解题思路

拓扑排序(Topological Sort)

       使用拓扑排序确定处理电路门的顺序

大模拟

       使用与(&)、或(|)、非(!)、异或(^)等操作模拟电路门,由输入信号得出输出信号。

作用域

       这道题的输入格式很奇怪。先输入所有问题的电路门的结构,再输入所有问题的电路门的输入,最后输入所有的问题要输出的电路门编号。而不是一个个问题输入。
       多次处理同一个问题时,巧妙地利用作用域(例如,将整个问题抽象成函数,将所有所需的变量封装在一个函数内),可以避免需要清理上个问题留下的数据的问题。

变量命名与注释

       大模拟的题往往涉及很多相关的变量,变量命名应该尽量规范且贴合题意。退一步来说,最起码也得在变量定义处写好注释

AC代码

C++

Score

#include <stdio.h>
#include <vector>
#include <queue>
#include <string.h>
#include <algorithm>

using namespace std;

typedef struct Gate
{
	char GateType[5];
	bool OutputValue;

	vector<int> InputList;	   //保存外界输入的信号编号
	vector<int> GateInputList; //保存依赖于其他电路门输出的输入信号
	vector<int> OutputList;
} Gate;

bool TopLogicalSort(const vector<Gate> &GateList, vector<int> &Indgree, int GateNum, vector<int> &TopLogicalGateList)
{
	//GateNoListTemp暂存拓扑排序的部分序列
	queue<int> GateNoListTemp;

	for (int i = 1; i <= GateNum; ++i)
	{
		//入度为0的门无需依赖于其他门,放入GateNoListTemp
		if (Indgree[i] == 0)
		{
			GateNoListTemp.push(i);
		}
	}

	while (GateNoListTemp.empty() == false)
	{
		//队首的门的编号放入拓扑序列
		TopLogicalGateList.push_back(GateNoListTemp.front());

		//取出排在队首的门的编号
		Gate GateTemp = GateList[GateNoListTemp.front()];
		GateNoListTemp.pop();

		//对下一个门的入度减一
		for (int i = 0; i < GateTemp.OutputList.size(); ++i)
		{
			//若下一个门的入度为0,放入GateNoListTemp
			if (--Indgree[GateTemp.OutputList[i]] == 0)
			{
				GateNoListTemp.push(GateTemp.OutputList[i]);
			}
		}
	}

	return TopLogicalGateList.size() == GateNum;
}

void Imitate(vector<Gate> &GateList, const vector<int> &TopLogicalGateList, int SubProblemNum, const vector<vector<int>> &InputSignal, const vector<int> &OutputSignalNum, const vector<vector<int>> &OutputSignal)
{
	for (int SubProblemNo_i = 0; SubProblemNo_i < SubProblemNum; ++SubProblemNo_i)
	{
		//根据拓扑序列,使用输入得出输出
		for (int GateNo_i = 0; GateNo_i < TopLogicalGateList.size(); ++GateNo_i)
		{
			Gate Temp = GateList[TopLogicalGateList[GateNo_i]];

			//根据门的类型分别进行处理
			if (Temp.GateType[0] == 'N' && Temp.GateType[2] == 'T') //非门NOT
			{
				if (Temp.InputList.empty() == true)
					Temp.OutputValue = (!GateList[Temp.GateInputList[0]].OutputValue);
				else
					Temp.OutputValue = (!InputSignal[SubProblemNo_i][Temp.InputList[0]]);
			}
			else if (Temp.GateType[0] == 'A') //与门AND
			{
				Temp.OutputValue = true;

				for (int i = 0; i < Temp.InputList.size(); ++i)
				{
					Temp.OutputValue &= InputSignal[SubProblemNo_i][Temp.InputList[i]];
				}

				for (int i = 0; i < Temp.GateInputList.size(); ++i)
				{
					Temp.OutputValue &= GateList[Temp.GateInputList[i]].OutputValue;
				}
			}
			else if (Temp.GateType[0] == 'O') //或门OR
			{
				Temp.OutputValue = false;

				for (int i = 0; i < Temp.InputList.size(); ++i)
				{
					Temp.OutputValue |= InputSignal[SubProblemNo_i][Temp.InputList[i]];
				}

				for (int i = 0; i < Temp.GateInputList.size(); ++i)
				{
					Temp.OutputValue |= GateList[Temp.GateInputList[i]].OutputValue;
				}
			}
			else if (Temp.GateType[0] == 'X') //异或门XOR
			{
				Temp.OutputValue = false;

				for (int i = 0; i < Temp.InputList.size(); ++i)
				{
					Temp.OutputValue ^= InputSignal[SubProblemNo_i][Temp.InputList[i]];
				}

				for (int i = 0; i < Temp.GateInputList.size(); ++i)
				{
					Temp.OutputValue ^= GateList[Temp.GateInputList[i]].OutputValue;
				}
			}
			else if (Temp.GateType[0] == 'N' && Temp.GateType[1] == 'A') //与非门NAND
			{
				Temp.OutputValue = true;

				for (int i = 0; i < Temp.InputList.size(); ++i)
				{
					Temp.OutputValue &= InputSignal[SubProblemNo_i][Temp.InputList[i]];
				}

				for (int i = 0; i < Temp.GateInputList.size(); ++i)
				{
					Temp.OutputValue &= GateList[Temp.GateInputList[i]].OutputValue;
				}

				Temp.OutputValue = !Temp.OutputValue; //先算与运算,最后取反
			}
			else if (Temp.GateType[0] == 'N' && Temp.GateType[2] == 'R') //或非门NOR
			{
				Temp.OutputValue = false;

				for (int i = 0; i < Temp.InputList.size(); ++i)
				{
					Temp.OutputValue |= InputSignal[SubProblemNo_i][Temp.InputList[i]];
				}

				for (int i = 0; i < Temp.GateInputList.size(); ++i)
				{
					Temp.OutputValue |= GateList[Temp.GateInputList[i]].OutputValue;
				}

				Temp.OutputValue = !Temp.OutputValue; //先算或运算,最后取反
			}

			GateList[TopLogicalGateList[GateNo_i]].OutputValue = Temp.OutputValue;
		}

		//依次输出
		for (int OutputSignalNo_i = 0; OutputSignalNo_i < OutputSignalNum[SubProblemNo_i]; ++OutputSignalNo_i)
		{
			int OutputSignalNo = OutputSignal[SubProblemNo_i][OutputSignalNo_i];

			printf("%d ", GateList[OutputSignalNo].OutputValue);
		}
		putchar('\n');
	}
}

int main()
{
	int ProblemNum = 0;
	scanf("%d", &ProblemNum);

	//依次处理每个问题
	for (int ProblemNo_i = 0; ProblemNo_i < ProblemNum; ++ProblemNo_i)
	{
		//输入输入信号的数量和电路门的数量
		int InputSignalNum, GateNum;
		scanf("%d%d", &InputSignalNum, &GateNum);

		vector<Gate> GateList(GateNum + 1);		 //电路门数组
		vector<int> GateIndgree(GateNum + 1, 0); //电路门的入度

		//对每个门的输入信号都进行输入
		for (int GateNo_i = 1; GateNo_i <= GateNum; ++GateNo_i)
		{
			int GateSignalNum;

			//输入电路门的类型和电路门输入信号的数量
			scanf("%s %d ", GateList[GateNo_i].GateType, &GateSignalNum);

			//对输入的信号进行处理
			for (int GateSignalNo_i = 0; GateSignalNo_i < GateSignalNum; ++GateSignalNo_i)
			{
				char GateInputSignalType = getchar(); // 'I' or 'O'

				int SignalNo;
				scanf("%d", &SignalNo); //信号编号
				getchar();				//deal with the ' ' and '\n'

				if (GateInputSignalType == 'I') //外界输入的信号编号
					GateList[GateNo_i].InputList.push_back(SignalNo);
				else //依赖于其他电路门输出的输入信号
				{
					GateList[GateNo_i].GateInputList.push_back(SignalNo);
					GateList[SignalNo].OutputList.push_back(GateNo_i);
					++GateIndgree[GateNo_i];
				}
			}
		}

		int SubProblemNum;
		scanf("%d", &SubProblemNum);

		//为每个子问题的输入信号值分配存储空间
		vector<vector<int>> InputSignal(SubProblemNum);

		//输入每个子问题的数据
		for (int SubProblemNo_i = 0; SubProblemNo_i < SubProblemNum; ++SubProblemNo_i)
		{
			InputSignal[SubProblemNo_i].push_back(0);
			for (int InputSignalNo_i = 1; InputSignalNo_i <= InputSignalNum; ++InputSignalNo_i)
			{
				int SignalValue;
				scanf("%d", &SignalValue);
				InputSignal[SubProblemNo_i].push_back(SignalValue);
			}
		}

		vector<int> OutputSignalNum(SubProblemNum);
		vector<vector<int>> OutputSignal(SubProblemNum);

		//输入每个子问题需要输出的信号
		for (int SubProblemNo_i = 0; SubProblemNo_i < SubProblemNum; ++SubProblemNo_i)
		{
			scanf("%d", &OutputSignalNum[SubProblemNo_i]);

			for (int OutputSignalNo_i = 0; OutputSignalNo_i < OutputSignalNum[SubProblemNo_i]; ++OutputSignalNo_i)
			{
				int SignalValue;
				scanf("%d", &SignalValue);
				OutputSignal[SubProblemNo_i].push_back(SignalValue);
			}
		}

		vector<int> TopLogicalGateList;

		//拓扑排序:判环
		if (TopLogicalSort(GateList, GateIndgree, GateNum, TopLogicalGateList) == false)
		{
			printf("LOOP\n");
			continue;
		}

		Imitate(GateList, TopLogicalGateList, SubProblemNum, InputSignal, OutputSignalNum, OutputSignal);
	}

	return 0;
}

python

       这一份只有50分,找不到错。

class Gate:
    def __init__(self):
        self.type = ""
        self.in_degree = 0
        self.input_sig = []
        self.input_gate = []
        self.output_gate = []


def topological_sort(gate_num, gate_list):
    temp_queue = []
    topological_order = []

    for i in range(gate_num):
        if gate_list[i].in_degree == 0:
            temp_queue.append(i)

    while temp_queue:
        temp = temp_queue.pop(0)
        topological_order.append(temp)

        output_gate = gate_list[temp].output_gate

        for i in output_gate:
            gate_list[i].in_degree -= 1

            if gate_list[i].in_degree == 0:
                temp_queue.append(i)

    return topological_order


def imitation(topological_order, gate_list, input_signal, output_signal):
    for gate_No in topological_order:
        temp_gate = gate_list[gate_No]
        gate_type = temp_gate.type
        input_sig = temp_gate.input_sig
        input_gate = temp_gate.input_gate
        result = 1

        if gate_type == "NOT":
            if not input_gate:
                result = 1 ^ input_signal[input_sig[0]]
            else:
                result = 1 ^ output_signal[input_gate[0]]
        elif gate_type == "AND":
            for sig_i in input_sig:
                result &= input_signal[sig_i]
            for sig_i in input_gate:
                result &= output_signal[sig_i]
        elif gate_type == "OR":
            result = 0
            for sig_i in input_sig:
                result |= input_signal[sig_i]
            for sig_i in input_gate:
                result |= output_signal[sig_i]
        elif gate_type == "XOR":
            result = 0
            for sig_i in input_sig:
                result ^= input_signal[sig_i]
            for sig_i in input_gate:
                result ^= output_signal[sig_i]
        elif gate_type == "NAND":
            for sig_i in input_sig:
                result &= input_signal[sig_i]
            for sig_i in input_gate:
                result &= output_signal[sig_i]
            result = 1 ^ result
        else:  # gate_type == "NOR":
            result = 0
            for sig_i in input_sig:
                result |= input_signal[sig_i]
            for sig_i in input_gate:
                result |= output_signal[sig_i]
            result = 1 ^ result
        output_signal[gate_No] = result


if __name__ == "__main__":
    question_num = int(input())

    for question_i in range(question_num):
        line_info = input().split()

        input_sig_num = int(line_info[0])

        gate_num = int(line_info[1])
        if not gate_num:  # test5
            continue

        gate_list = [Gate() for i in range(gate_num)]

        for gate_i in range(gate_num):
            line_info = input().split()
            gate_type = line_info[0]
            gate_list[gate_i].type = gate_type

            gate_input_num = int(line_info[1])

            for gate_input_sig_i in range(gate_input_num):
                input_sig_type = line_info[2 + gate_input_sig_i][0]
                input_sig = int(line_info[2 + gate_input_sig_i][1:]) - 1

                if input_sig_type == "I":
                    gate_list[gate_i].input_sig.append(input_sig)
                else:
                    gate_list[gate_i].input_gate.append(input_sig)
                    gate_list[gate_i].in_degree += 1
                    gate_list[input_sig].output_gate.append(gate_i)

        topological_order = topological_sort(gate_num, gate_list)

        imitation_times = int(input())

        input_signal = []
        output_signal = []
        if input_sig_num:
            for imitation_i in range(imitation_times):
                input_sig_i = input().split()
                input_sig_i = [int(i) for i in input_sig_i]
                input_signal.append(input_sig_i)
        else:
            input_signal = [[] * imitation_times]

        for imitation_i in range(imitation_times):
            output_sig_i = input().split()
            output_sig_i = [int(i) for i in output_sig_i]
            output_sig_i.pop(0)

            output_signal.append(output_sig_i)

        if len(topological_order) != gate_num:
            print("LOOP")
            continue

        for imitation_i in range(imitation_times):
            input_sig_i = input_signal[imitation_i]
            output_sig_i = output_signal[imitation_i]

            if output_sig_i == [] or output_sig_i[0] == 0:
                continue

            output_signal_i = [0] * gate_num

            imitation(topological_order, gate_list, input_sig_i, output_signal_i)

            ans = []
            for output_sig in output_sig_i:
                ans.append(str(output_signal_i[output_sig - 1]))
            print(" ".join(ans))
  • 2
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值