分析
本来以为这题跟1058题目都差不多的,看来是我想多了Orz
异或是怎么想到的,tql
又是抄柳神答案的一题0.0
柳神分析:错误是指错选或者漏选。用异或运算来判断一个选项和正确选项是否匹配,如果是匹配的,那么异或的结果应当是0;如果不匹配,那么这个选项就是存在错选或者漏选的情况~例如:设a为00001,b为00010,c为00100,d为01000,e为10000,如果给定的正确答案是ac,即10001,那么如果给出的选项也是10001,异或的结果就是0;如果给出的选项是a或者ab,即00001或00011,异或之后不为0,就是错选和漏选的~通过异或操作的结果,再用与运算就可以把错选和漏选的选项找出来~fullscore表示一道题满分的分值;trueopt表示正确的选项,存储的是正确选项二进制的值,二进制由hash给出分别是1,2,4,8,16;cnt是错误的次数,maxcnt是最大错误次数~
题解
#include <iostream>
#include <vector>
#include <cmath>
using namespace std;
int main() {
int n, m, optnum, truenum, temp, maxcnt = 0;
int hash[] = {1, 2, 4, 8, 16}, opt[1010][110] = {0};
char c;
scanf("%d %d", &n, &m);
vector<int> fullscore(m), trueopt(m);
vector<vector<int>> cnt(m, vector<int>(5));
for (int i = 0; i < m; i++) {
scanf("%d %d %d", &fullscore[i], &optnum, &truenum);
for (int j = 0; j < truenum; j++) {
scanf(" %c", &c);
trueopt[i] += hash[c-'a'];
}
}
for (int i = 0; i < n; i++) {
double grade = 0;
for (int j = 0; j < m; j++) {
getchar();
scanf("(%d", &temp);
for (int k = 0; k < temp; k++) {
scanf(" %c)", &c);
opt[i][j] += hash[c-'a'];
}
int el = opt[i][j] ^ trueopt[j];
if (el) {
if ((opt[i][j] | trueopt[j]) == trueopt[j]) {
grade += fullscore[j] * 1.0 / 2;
}
if (el) {
for (int k = 0; k < 5; k++)
if (el & hash[k]) cnt[j][k]++;
}
} else {
grade += fullscore[j];
}
}
printf("%.1f\n", grade);
}
for (int i = 0; i < m; i++)
for (int j = 0; j < 5; j++)
maxcnt = maxcnt > cnt[i][j] ? maxcnt : cnt[i][j];
if (maxcnt == 0) {
printf("Too simple\n");
} else {
for (int i = 0; i < m; i++) {
for (int j = 0; j < cnt[i].size(); j++) {
if (maxcnt == cnt[i][j])
printf("%d %d-%c\n", maxcnt, i+1, 'a'+j);
}
}
}
return 0;
}