2018年刑侦科推理试题

hello

各位小伙伴

最近过的怎么样

有没有按时吃饭,睡觉,敲代码

一定要照顾好自己呀~



不知道大家前段时间,有没有在朋友圈中收到一份,『2018的刑侦科目推理试题』娜娜偶然在朋友圈中看到过一次~



原图奉上,没有看过的小伙伴可以仔细的看一下哈~



我看到第一题的反应是

题呢?你到给我题呀~连题都不给我~我怎么知道答案~

看到第二题我的脑袋已经无法思考一片空白了~

这尼玛是什么玩意儿,简直就是一个坑啊,然后我就放弃了。



但是最近我看到各路大神对这道题进行了解答,我才意识到,原来看似这么不正经的题其实是有很正经的答案的~


娜娜~不禁想问,这的死多少脑细胞才能做出这样的推理题~看来我那900+多集的柯南是白看了~


python语言实现



 

#encoding: utf-8
# 《2018刑侦科推理试题》非穷举的 Python 解法
# 需要先安装约束解决库 `pip3 install python-constraint`
# 使用 Python 3.5 编写

from constraint import *

problem = Problem()

# a1 - a10 表示第一题到第十题的答案变量,答案使用“1”表示“A”, “2”表示“B”,以此类推
vars = ["a1""a2""a3""a4""a5""a6""a7""a8""a9""a10"]
problem.addVariables(vars, [1234])


#第 2 题
def a2_func(a2, a5):
   return (a2 == 1 and a5 == 3or (a2 == 2 and a5 == 4or (a2 == 3 or a5 == 1or (a2 == 4 or a5 == 2)
problem.addConstraint(a2_func, ["a2""a5"])

#第 3 题
def a3_func(a1, a2, a3, a4, a5, a6, a7, a8, a9, a10):
   return (a3 == 1 and a6 == a2 == a4 != a3) or (a3 == 2 and a3 == a2 == a4 != a6) \
       or (a3 == 3 and a3 == a6 == a4 != a2) or (a3 == 4 and a3 == a6 == a2 != a4)
problem.addConstraint(a3_func, vars)


#第 4 题
def a4_func(a1, a2, a3, a4, a5, a6, a7, a8, a9, a10):
   return (a4 == 1 and a1 == a5) or (a4 == 2 and a2 == a7) or (a4 == 3 and a1 == a9) or (a4 == 4 and a6 == a10)
problem.addConstraint(a4_func, vars)

#第 5 题
def a5_func(a1, a2, a3, a4, a5, a6, a7, a8, a9, a10):
   return (a5 == a8 == 1or (a5 == a4 == 2or (a5 == a9 == 3or (a5 == a7 == 4)
problem.addConstraint(a5_func, vars)

#第 6 题
def a6_func(a1, a2, a3, a4, a5, a6, a7, a8, a9, a10):
   return (a6 == 1 and a2 == a4 == a8) or (a6 == 2 and a1 == a6 == a8) \
          or (a6 == 3 and a3 == a10 == a8) or (a6 == 4 and a5 == a9 == a8)
problem.addConstraint(a6_func, vars)

#第 7 题
def a7_func(a1, a2, a3, a4, a5, a6, a7, a8, a9, a10):
   all_answers = [a1, a2, a3, a4, a5, a6, a7, a8, a9, a10]
   counter = [0000]
   for a in all_answers: 
       counter[a - 1] += 1
   imin = counter.index(min(counter)) + 1
   n = len(set(counter))
   # 这题为“此10题种被选中的选项最少的为,ABCD选择,说明 ABCD 的数量个不相同”
   return n == 4 and ((a7 == 1 and imin == 3or (a7 == 2 and imin == 2or (a7 == 3 and imin == 1or (a7 == 4 and imin == 4))
problem.addConstraint(a7_func, vars)

#第 8 题
def a8_func(a8, a7, a5, a2, a10, a1):
   adj = lambda x: abs(a1 - x) != 1
   return (a8 == 1 and adj(a7)) or (a8 == 2 and adj(a5)) or (a8 == 3 and adj(a2)) or (a8 == 4 and adj(a10))
problem.addConstraint(a8_func, ["a8""a7""a5""a2""a10""a1"])

#第 9 题
def a9_func(a1, a2, a5, a6, a9, a10):
   cond1 = a1 == a6
   cond = lambda x: cond1 != (x == a5)
   return (a9 == 1 and cond(a6)) or (a9 == 2 and cond(a10)) or (a9 == 3 and cond(a2)) or (a9 == 4 and cond(a9))
problem.addConstraint(a9_func, ["a1""a2""a5""a6""a9""a10"])

#第 10 题
def a10_func(a1, a2, a3, a4, a5, a6, a7, a8, a9, a10):
   all_answers = [a1, a2, a3, a4, a5, a6, a7, a8, a9, a10]
   counter = [0000]
   for a in all_answers: 
       counter[a - 1] += 1
   v = abs(max(counter) - min(counter))
   return (a10 == 1 and v == 3or (a10 == 2 and v == 2or (a10 == 3 and v == 4or (a10 == 4 and v == 1)
problem.addConstraint(a10_func, vars)


# 约束设置完毕,开始求解
solutions = problem.getSolutions()
print("可能的解决数量:{0}".format(len(solutions)))

chars = ['A''B''C''D']

for si in range(0, len(solutions)):
   print("\n---------- 第 {0} 组结果 ----------".format(si + 1))
   s = solutions[si]
   for i in range(111):
       key = "a" + str(i)
       answer = chars[s[key] - 1]
       print("第 {0} 题答案:{1}".format(i, answer))


运行结果:



c++语言实现


C语言实现:



一上来就是两发,下面代码是精简后的代码。


 

#include <cstdio>
#define JUDGE(i, JA, JB, JC, JD) \
   if (c[i] == 0 && (JA)) continue; \
   if (c[i] == 1 && (JB)) continue; \
   if (c[i] == 2 && (JC)) continue; \
   if (c[i] == 3 && (JD)) continue
void main({
   int c[10], n[4], dn, sn, dc, sc; 
   for (int k = 0; k < (1 << 20); ++k) {
       n[0] = n[1] = n[2] = n[3] = 0;
       for (int i = 0; i < 10; ++i) {
           c[i] = ((k >> 2*i) & 3);
           ++n[c[i]];
       }
       dn = 0, sn = 10;
       for (int i = 0; i < 4; ++i) {
           if (dn < n[i]) {dn = n[i]; dc = i;}
           if (sn > n[i]) {sn = n[i]; sc = i;}
       }
       JUDGE(1, c[4] != 2, c[4] != 3, c[4] != 0, c[4] != 1);
       JUDGE(2, c[2] == c[5] || c[2] == c[1] || c[2] == c[3], 
                c[5] == c[2] || c[5] == c[1] || c[5] == c[3],
                c[1] == c[2] || c[1] == c[5] || c[1] == c[3], 
                c[3] == c[2] || c[3] == c[1] || c[3] == c[5]);
       JUDGE(3, c[0] != c[4], c[1] != c[6], c[0] != c[8], c[5] != c[9]);
       JUDGE(4, c[4] != c[7], c[4] != c[3], c[4] != c[8], c[4] != c[6]);
       JUDGE(5, c[7] != c[1] || c[7] != c[3], c[7] != c[0] || c[7] != c[5], 
                c[7] != c[2] || c[7] != c[9], c[7] != c[4] || c[7] != c[8]);
       JUDGE(6, sc != 2, sc != 1, sc != 0, sc != 3);
       JUDGE(7, c[0] - c[6] == 1 || c[0] - c[6] == -1
                c[0] - c[4] == 1 || c[0] - c[4] == -1
                c[0] - c[1] == 1 || c[0] - c[1] == -1
                c[0] - c[9] == 1 || c[0] - c[9] == -1);
       if (c[0] == c[5]) {
           JUDGE(8, c[4] == c[5], c[4] == c[9], c[4] == c[1], c[4] == c[8]);
       } else {
           JUDGE(8, c[4] != c[5], c[4] != c[9], c[4] != c[1], c[4] != c[8]);
       }
       JUDGE(9, dn - sn != 3, dn - sn != 2, dn - sn != 4, dn - sn != 1);
       printf("1:%c, 2:%c, 3:%c, 4:%c, 5:%c, 6:%c, 7:%c, 8:%c, 9:%c, 10:%c",
           'A'+c[0], 'A'+c[1], 'A'+c[2], 'A'+c[3], 'A'+c[4], 
           'A'+c[5], 'A'+c[6], 'A'+c[7], 'A'+c[8], 'A'+c[9]);
   }
}


Java语言实现

 

public static void answer(){
       // 用 1234 分别对应 ABCD, 计算方便
       int[] answers = {1,2,3,4};
       // 群举答案
       for (int q1 : answers) {
           for (int q2 : answers) {
               for (int q3 : answers) {
                   for (int q4 : answers) {
                       for (int q5 : answers) {
                           for (int q6 : answers) {
                               for (int q7 : answers) {
                                   for (int q8 : answers) {
                                       for (int q9 : answers) {
                                           for (int q10 : answers) {
                                               int[] questions = new int[10];
                                               questions[0] = q1;
                                               questions[1] = q2;
                                               questions[2] = q3;
                                               questions[3] = q4;
                                               questions[4] = q5;
                                               questions[5] = q6;
                                               questions[6] = q7;
                                               questions[7] = q8;
                                               questions[8] = q9;
                                               questions[9] = q10;
                                               if (isEnd(questions)){
                                                   // 遍历输出符合条件的答案
                                                   for (int i = 0 ; i < 10; i++){
                                                       System.out.println((i+1) + ":" + questions[i]);
                                                   }
                                               }
                                           }
                                       }
                                   }
                               }
                           }
                       }
                   }
               }
           }
       }
   }
   /**
    * 判断每个答案是否符合题意
    * 为了方便 questions 数组中从 0 开始,
    * 题目比数组角标多 1(不要问为什么, 奏是这么开)
    * 比如 question[0] 的值表示第 1 题答案
    **/

   static boolean isEnd(int[] questions){
       // 第二题, 第 5 题的答案是
       switch (questions[4]){
           case 1:
               // 如果第 5 题答案是 A, 判断第 2 题答案是不是 C 不是返回 false, 是继续
               if(questions[1] != 3)
                   return false;
               break;
           case 2:
               // 原理同上
               if(questions[1] != 4)
                   return false;
               break;
           case 3:
               // 原理同上
               if(questions[1] != 1)
                   return false;
               break;
           case 4:
               // 原理同上
               if(questions[1] != 2)
                   return false;
               break;
       }
       // 第 3 题, 以下选项中哪一题的答案与其他三项不同
       switch (questions[2]){
           case 1:
               if(!(questions[2]!=questions[5] && questions[5]==questions[1] && questions[1] == questions[3]))
                   return false;
               break;
           case 2:
               if(!(questions[5]!=questions[2] && questions[2]==questions[1] && questions[1] == questions[3]))
                   return false;
               break;
           case 3:
               if(!(questions[1]!=questions[5] && questions[2]==questions[5] && questions[5] == questions[3]))
                   return false;
               break;
           case 4:
               if(!(questions[3]!=questions[5] && questions[5]==questions[1] && questions[1] == questions[2]))
                   return false;
               break;
       }
       // 第 4 题, 以下选项中那两题的答案相同
       switch (questions[3]){
           case 1:{
               // 判断第 1 题与第 5 题答案是否相同
               if (questions[0] != questions[4]){
                   return  false;
               }
               break;
           }
           case 2:{
               // 原理同上
               if (questions[1] != questions[6]){
                   return  false;
               }
               break;
           }
           case 3:{
               // 原理同上
               if (questions[0] != questions[8]){
                   return  false;
               }
               break;
           }
           case 4:{
               // 原理同上
               if (questions[5] != questions[9]){
                   return  false;
               }
               break;
           }
       }
       // 第 5 题, 以下选项中哪一题的答案与本题相同
       switch (questions[4]){
           case 1:
               // 判断第 8 题答案是否是 A
               if (questions[7] != 1)
                   return  false;
               break;
           case 2:
               // 原理同上
               if(questions[3] != 2)
                   return  false;
               break;
           case 3:
               // 原理同上
               if (questions[8] != 3)
                   return false;
           case 4:
               // 原理同上
               if (questions[6] != 4)
                   return  false;
               break;
       }
       // 第 6 题, 以下选项中哪两题的答案与第 8 题相同
       switch (questions[5]){
           case 1:
               // 判断第 14 题答案是否与第 8 题答案相同
               if(questions[1] != questions[7] || questions[4] != questions[7])
                   return  false;
               break;
           case 2:
               // 原理同上
               if(questions[0] != questions[7] || questions[5] != questions[7])
                   return  false;
               break;
           case 3:
               // 原理同上
               if(questions[2] != questions[7] || questions[9] != questions[7])
                   return  false;
               break;
           case 4:
               // 原理同上
               if(questions[4] != questions[7] || questions[8] != questions[7])
                   return  false;
               break;
       }
       // 由于第 710 题问题是同类型的, 所以一块计算 start
       int[] check10 = new int[5];
       // 把每个题的答案 (1234) 作为新数组下表, value++ 计算出现次数
       for (int i=0;i < questions.length;i++){
           check10[questions[i]]++;
       }
       // 出现最少与最多选项的次数初始化为 A 的次数
       int low = check10[1];
       int longer = check10[1];
       // 出现最少的选项, 初始化为 A
       int lowA = 1;
       // 最少与最多次数的选项相关计算
       for (int i=1;i<5;i++) {
           if(check10[i] >0 && check10[i] < low){
               low = check10[i];
               lowA = i;
           }
           if (check10[i] > longer){
               longer = check10[i];
           }
       }
       // 第 7 题, 在此十道题中, 被选中次数最少的选项字母为
       switch (questions[6]){
           case 1:
               // 判断才出现最少的字母是否为 C
               if (lowA != 3)
                   return  false;
               break;
           case 2:
               // 原理同上
               if (lowA != 2)
                   return  false;
               break;
           case 3:
               // 原理同上
               if (lowA != 1)
                   return  false;
               break;
           case 4:
               // 原理同上
               if (lowA != 4)
                   return  false;
               break;
       }
       // 第 10 题, 在此 10 道题中, ABCD 四个字母出现次数最多与最少者的差为
       // 最多次数与最少次数的差值
       int t = longer-low;
       switch (questions[9]){
           case 1:
               // 判断差值是否为 3
               if (t != 3)
                   return  false;
               break;
           case 2:
               // 原理同上
               if (t != 2)
                   return  false;
               break;
           case 3:
               // 原理同上
               if (t != 4)
                   return  false;
               break;
           case 4:
               // 原理同上
               if (t != 1)
                   return  false;
               break;
       }
       // 第 710 题校验 end
       // 第 8 题, 以下选项中哪一题的答案与第 1 题的答案在字母中不相邻
       switch (questions[7]) {
           case 1:
               // 判断第 7 题与第一题答案差值绝对是是否为 1
               if (Math.abs(questions[6] - questions[0]) == 1)
                   return false;
               break;
           case 2:
               // 原理同上
               if (Math.abs(questions[4] - questions[0]) == 1)
                   return false;
               break;
           case 3:
               // 原理同上
               if (Math.abs(questions[1] - questions[0]) == 1)
                   return false;
               break;
           case 4:
               // 原理同上
               if (Math.abs(questions[9] - questions[0]) == 1)
                   return false;
               break;
       }
       // 第 9 题, 已知第 1 题与第 6 题的答案相同与第 X 题与第 5 题的答案相同的真假性相反, 那么 X 为
       // 判断第 1 题与第 6 题的答案是否相同
       boolean isOne = questions[0]==questions[5]?true:false;
       switch (questions[8]){
           case 1:
               if(isOne){
                   // 第 1 题与第 6 题相同, 第 6 题与第 5 题答案相同返回 false;
                   if (questions[5] == questions[4])
                       return false;
               }else {
                   // 第 1 题与第 6 题不相同, 第 6 题与第 5 题答案不相同返回 false;
                   if (questions[5] != questions[4])
                       return false;
               }
               break;
           case 2:
               // 原理同上
               if(isOne){
                   if (questions[9] == questions[4])
                       return false;
               }else {
                   if (questions[9] != questions[4])
                       return false;
               }
               break;
           case 3:
               // 原理同上
               if(isOne){
                   if (questions[1] == questions[4])
                       return false;
               }else {
                   if (questions[1] != questions[4])
                       return false;
               }
               break;
           case 4:
               // 原理同上
               if(isOne){
                   if (questions[8] == questions[4])
                       return false;
               }else {
                   if (questions[8] != questions[4])
                       return false;
               }
               break;
       }
       return true;
   }
   public static void main(String[] args) {
       answer();
   }



看到上面的答案之后,娜娜躲在角落了瑟瑟发抖,大牛,不就是膝盖吗?我给还不行吗?


然后我看到了下面这条信息,这一是道杭州学军中学的推理社的招新题~招新题~



然后娜娜开始研究怎么把这道题解出来~



此时此刻娜娜

娜娜的心情就和外面的天气一样

当我以为这题没有答案的时候

有人做出了答案

当我以为做答案的都是大牛的时候

有人说这是入门级的题

当我想着别人能做我也能做到的时候

发现自己竟然做不出来



各位小伙伴

开往幼儿园的班车就要发车了

没做出来的小伙伴请上车

和娜娜一起回去学习


不要担心

我一定会回来的~~~

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值