题目
为了保障安静的阅读环境,有些公共图书馆对儿童入馆做出了限制。例如“12 岁以下儿童禁止入馆,除非有 18 岁以上(包括 18 岁)的成人陪同”。现在有两位小/大朋友跑来问你,他们能不能进去?请你写个程序自动给他们一个回复。
输入格式:
输入在一行中给出 4 个整数:
禁入年龄线 陪同年龄线 询问者1的年龄 询问者2的年龄
这里的禁入年龄线
是指严格小于该年龄的儿童禁止入馆;陪同年龄线
是指大于等于该年龄的人士可以陪同儿童入馆。默认两个询问者的编号依次分别为 1
和 2
;年龄和年龄线都是 [1, 200] 区间内的整数,并且保证 陪同年龄线
严格大于 禁入年龄线
。
输出格式:
在一行中输出对两位询问者的回答,如果可以进就输出 年龄-Y
,否则输出 年龄-N
,中间空 1 格,行首尾不得有多余空格。
在第二行根据两个询问者的情况输出一句话:
- 如果两个人必须一起进,则输出
qing X zhao gu hao Y
,其中X
是陪同人的编号,Y
是小孩子的编号; - 如果两个人都可以进但不是必须一起的,则输出
huan ying ru guan
; - 如果两个人都进不去,则输出
zhang da zai lai ba
; - 如果一个人能进一个不能,则输出
X: huan ying ru guan
,其中X
是可以入馆的那个人的编号。
输入样例 1:
12 18 18 8
输出样例 1:
18-Y 8-Y
qing 1 zhao gu hao 2
输入样例 2:
12 18 10 15
输出样例 2:
10-N 15-Y
2: huan ying ru guan
解析
可以列表分析,假设 A 0 B 0
表示 A 为状态 0,B 为状态 0。
状态 0:< 禁入年龄线
状态 1:禁入年龄线 ~ 陪同年龄线
状态 2:>= 陪同年龄线
用 Excel 列成表格,然后翻译成对应的代码即可
代码
Python 版本
min_entry_age, min_parent_age, age1, age2 = [int(x) for x in input().split(' ')]
if age1 < min_entry_age:
state1 = 0
elif age1 < min_parent_age:
state1 = 1
elif age1 >= min_parent_age:
state1 = 2
if age2 < min_entry_age:
state2 = 0
elif age2 < min_parent_age:
state2 = 1
elif age2 >= min_parent_age:
state2 = 2
match (state1, state2):
case (0, 0):
print(f'{age1}-N {age2}-N')
print('zhang da zai lai ba')
case (0, 1):
print(f'{age1}-N {age2}-Y')
print('2: huan ying ru guan')
case (0, 2):
print(f'{age1}-Y {age2}-Y')
print('qing 2 zhao gu hao 1')
case (1, 0):
print(f'{age1}-Y {age2}-N')
print('1: huan ying ru guan')
case (1, 1):
print(f'{age1}-Y {age2}-Y')
print('huan ying ru guan')
case (1, 2):
print(f'{age1}-Y {age2}-Y')
print('huan ying ru guan')
case (2, 0):
print(f'{age1}-Y {age2}-Y')
print('qing 1 zhao gu hao 2')
case (2, 1):
print(f'{age1}-Y {age2}-Y')
print('huan ying ru guan')
case (2, 2):
print(f'{age1}-Y {age2}-Y')
print('huan ying ru guan')
C 版本
#include <stdio.h>
int main() {
int min_entry_age, min_parent_age, age1, age2;
scanf("%d %d %d %d", &min_entry_age, &min_parent_age, &age1, &age2);
int state1, state2;
if (age1 < min_entry_age)
state1 = 0;
else if (age1 < min_parent_age)
state1 = 1;
else if (age1 >= min_parent_age)
state1 = 2;
if (age2 < min_entry_age)
state2 = 0;
else if (age2 < min_parent_age)
state2 = 1;
else if (age2 >= min_parent_age)
state2 = 2;
if (state1 == 0 && state2 == 0) {
printf("%d-N %d-N\n", age1, age2);
printf("zhang da zai lai ba\n");
} else if (state1 == 0 && state2 == 1) {
printf("%d-N %d-Y\n", age1, age2);
printf("2: huan ying ru guan\n");
} else if (state1 == 0 && state2 == 2) {
printf("%d-Y %d-Y\n", age1, age2);
printf("qing 2 zhao gu hao 1\n");
} else if (state1 == 1 && state2 == 0) {
printf("%d-Y %d-N\n", age1, age2);
printf("1: huan ying ru guan\n");
} else if (state1 == 1 && state2 == 1) {
printf("%d-Y %d-Y\n", age1, age2);
printf("huan ying ru guan\n");
} else if (state1 == 1 && state2 == 2) {
printf("%d-Y %d-Y\n", age1, age2);
printf("huan ying ru guan\n");
} else if (state1 == 2 && state2 == 0) {
printf("%d-Y %d-Y\n", age1, age2);
printf("qing 1 zhao gu hao 2\n");
} else if (state1 == 2 && state2 == 1) {
printf("%d-Y %d-Y\n", age1, age2);
printf("huan ying ru guan\n");
} else if (state1 == 2 && state2 == 2) {
printf("%d-Y %d-Y\n", age1, age2);
printf("huan ying ru guan\n");
}
return 0;
}
补充
C 语言的 switch 由于没法像 Python match 那样可以一下匹配两个数,只能写一大堆啰嗦的 if elseif。
可以考虑合并两个 state 为一个数,然后再用 switch 语句来判断。
怎么合并呢?
我们的 state 可以取 0、1、2,对应二进制 00、01、10,只需要用两个 bit 就可表示。两个 state 总共就是 4 bit,而 C 语言里最小的整数 char 也有 8 bit,完全放得下。
所以做法是,用一个 char 类型变量,总共 8 bit,从低往高(从右往左)第 1、2 位是 state1,第 3、4 位是 state2,剩下的不用,保持为 0。
比如 state1=0,state2=2,那么 state=
00 00 10(state2) 00(state1)
也就是 8。
这是修改后的代码,相同的部分就不重复写了
char state = 0b0000;
if (age1 < min_entry_age)
state |= 0b0001; // 1
else if (age1 < min_parent_age)
state |= 0b0010; // 2
else
state |= 0b0011; // 3
if (age2 < min_entry_age)
state |= 0b0100;
else if (age2 < min_parent_age)
state |= 0b1000;
else
state |= 0b1100;
// switch 部分
switch (state) {
case 0b0000:
printf("%d-N %d-N\n", age1, age2);
printf("zhang da zai lai ba\n");
break;
case 0b0001:
//剩下的按照上面的方法算出来就可以了
//...
}
这种利用位来标记信息的方式很常见,比如标志变量 int flag
,不同位的含义各不相同,为 1 表示启用,为 0 表示禁用。
这么做的好处是可以把很多参数/变量合并为一个。