CSDN周赛59期简要题解

2023-11-18

本期题目相对比较友好,而且在比赛报名界面还提示了非编程题考察的章节——诚不欺我:

本期非编程题需要选手阅读的章节是第2章“逆向思考——从递推到递归”—2.3节“堆栈和队列:遍历的数据结构

选择和判断都考到了栈的数据结构,稍微有点基础知识找出正确答案并不难。而填空题也是最简单的二叉树的遍历问题。只不过一开始我还想回答“层序”,但填空后面紧跟着的“优先”二字,提示了我只能二选一,很显然,树的层序遍历就是广度优先遍历,填上“广度”二字即可。


编程题

两道编程题都不难,但是都有必须要被狠狠吐槽的点。

1. 坏掉的打字机

贝博士发明了一种能够自动输出数字并求和的打字机,但这台打字机出了一些故障,它除了输出数字以外还会随机地输出其他字符。艾小姐让贝博士不要着急,她写了一段程序,能够让这台打字机即使在这样的情况下也能输出正确答案。那么,你知道艾小姐的程序是怎么写的吗?

输入描述:单行文本(字符数小于100000)。

输出描述:提取该文件中所有的数字,并输出它们的和。如果结果为整数,则输出整数。如果结果为小数,则输出的小数不应该带有末尾的0(字符串中的数字在-999999到999999之间,且最多含有3位小数)。

输入样例:10.20.5.9.9.-8.22.,40.75

输出样例:57.63

字符串题,难点在于从一长串字符串中提取出所有合法的数字,再进行求和。关于合法的数字,本题的定义是:

  • 负数。很好理解,负号(-)在头部,且只有一个负号的数字。
  • 小数。比如 3.14、3.(这里小数点后面没有数字也算合法)、3.1000 (第一个槽点在此,虽然小数如题目所说未满三位,但是如果后面都是0,依然算是前面那个数的小数位)

之所以说上面的 3.1000 是个槽点,因为如果严格按照题目的要求(最多含有3位小数),那么3.1000 应该被解析成 3 和 1000 两个整数。这里故意写成 3.1000 却要解析成 3.1,唯一的作用就是恶心你。

如此,对于从一堆字符串中找到所有合法数字,最简单的办法无疑就是正则式了。按照题目字面意思,写成的正则式应该是 -?\d+\.?\d{0, 3} 。但是由于上面提到的“坑”,这个表达式找不出 3.1000,于是改写成 -?\d+\.?\d*,即不限制小数点后尾数,就能抓出所有数字了。

严格来说,这样并不符合题目的要求,比如如果真的有 3.1415 这样的数字,按照题目要求,应该是解析出 3 和 1415 才对。但所幸本题太水,而出题人好像故意留坑,从而没有考虑到这一点。

紧接着,就会遇到本题最大的槽点。上面这种方法只能通过 90%,另外 10% 的正确答案竟然是四位数!合着你告诉我所有数字“最多含有3位小数”,结果正确答案是4位数?!求求你告诉我一堆3位小数的数字是怎么通过相加得到4位小数的?虽然计算机的浮点数计算是存在误差的,但误差也不会误差到万分位这么高的位置吧?就算真的是因为误差,难道误差可以作为正确答案??

大写的无语。

当然,通过这一例的方法也很简单,保留4位小数即可,不再赘述。

2. 布尔零点计数

使一个布尔表达式的值为零的取值组合称为该表达式的一个布尔零点。

比如,布尔表达式A+B+C就只有一个零点,就是(A,B,C)的取值组合(0,0,0)。而布尔表达式A*B*C就有不止一个零点,(A,B,C)的取值组合为(0,0,1)或(1,1,0)都是它的零点。其中A、B、C都是布尔变量,而布尔变量只能取值0或1。

布尔表达式可以把布尔变量去掉而只保留运算符,称为布尔表达式的简写。如:+*(+)就是布尔表达式A*(B+C)的简写。

现在用|表示或运算,它有如下的真值表:

0|0=0
0|1=1
1|0=1
1|1=1

布尔表达式的优先级是:乘法运算优先于加法运算和或运算,但小括号可以改变优先级为“小括号中的内容优先”。

现给出只包含乘法运算和或运算的布尔表达式的简写,求表达式的零点计数。

输入描述:单行文本,表示布尔表达式的简写(字符数小于10000)。

输出描述:表达式的零点计数d(d>=0)。

输入样例:(|)*

输出样例:5

本题也不难,但是读题的时候就会发现第一个槽点:题目的乘法是用 \* 表示的。很显然,这是为了防止题目描述的字符被转义(*号在MD中有加粗的作用),但问题是接收到的表达式,是没有这个反斜杠(\)的,题目也没有对此进行说明。当然,这个问题也不大,稍微有点web经验的选手应该都能看得出来。但是,规范的题目不是应该尽可能避免歧义么?

其实本题还是挺巧妙的,结合了栈的数据结构,与DP的递推思想。但是,真的不难。

首先,由于计算是有优先级的,乘法(*)优先于或运算(|),而如果出现括号的话,先计算括号中的内容——这与数学表达式是相同的。而如果说到数学表达式,就很容易想到更符合计算机思想的——后缀表达式,也叫逆波兰表达式,我记得还有一期专门说过。

后缀表达式对计算机来说是没有优先级的(或者说不用考虑优先级),从左到右,依次计算下去,这样就容易递推了对不对?当然,后缀表达式用到了栈结构,而且是两个栈。这部分属于标准内容,感兴趣的同学可以自行搜索,背模板即可。

于是,只要将题目给出的表达式转换成后缀表达式,问题就迎刃而解了。

不过,在转换的过程中会碰到一个小问题:表达式里的布尔变量被省略了。比如 **| 和 *|* 其实是不同的表达式,但由于省略了布尔变量,如果“硬”转成后缀表达式,就都变成了 **|,显然是错误的。所以在转换为后缀表达式的同时,我们还需要将省略的布尔变量“补”回来。

但是,我们其实并不用真的把布尔变量补回来,因为我们要计算的是整个表达式最终结果为 0 的组合数,所以补回布尔变量并无意义,除非你想通过暴力穷举是填充每个位置的布尔变量。——复杂度将是恐怖的 O(2^{n}) 。

这个时候就可以借鉴一下 DP 的递推思想了。我们可以维护一个数组,只保存两个数字,表示“计算到目前为止,使得前面出现过的表达式(已转成后缀表达式)计算结果为 0 和为 1 的组合分别是多少”。这句话有点长,但其实不难理解:假设只有一个布尔变量,那它要么为 0,要么为 1,用数组可以表示为 (1, 1),表示它为 0 或为 1 的组合数各为 1。当它遇到第二个布尔变量(也用(1, 1)表示),分别进行乘法(*)和或运算(|)的结果是怎样的呢?

我们记第一个布尔变量为 a,第二个布尔变量为 b,二者计算结果为 c,则

  • 当进行乘法(*)计算时:c = [a[0]*b[0]+a[0]*b[1]+a[1]*b[0], a[1]*b[1]]
  • 当进行或运算(|)计算时:c = [a[0]*b[0], a[1]*b[0]+a[1]*b[1]+a[0]*b[1]]

大家可以自行体会一下。

想到了这一点,前面“补全”表达式的答案也清晰了,只要将布尔变量的位置放上一个(1,1)即可。然后将其转成后缀表达式,再用上面介绍的计算方法依次计算即可得到最终答案。当然,最终答案是包含了使表达式结果分别为 0 和为 1 的组合个数,我们只输出为 0 的组合数就好。

然鹅。。。最终将迎来本题的最后一个大坑。

单纯上面这种方法只能通过 80% 的用例,原因是,另外两个用例的答案数字太大,会造成溢出。而题目却并没有说明对于溢出的情况如何处理(同类题目一般都会要求结果对 1e9+7 取余)。而对于 Python 这种弱数据类型的语言来说,不存在溢出,所以直接得到的大数答案,虽然是正确的,却通不过。

解决办法就是将答案对 2^{32} 取模,模拟这个溢出的过程,只取溢出后的数字作为答案。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

CSDN周赛59期简要题解 的相关文章

随机推荐

  • 2020北京邮电大学计算机学院复试经验分享

    初试组内第4 复试组内第1 综合第2 已成功上岸 最近大家问我复试的比较多 趁还热乎 在这里给大家分享一下吧 仅供参考 然后初试经验贴在这里 不要因为初试成绩不好就放弃复试或者不认真对待 复试是干嘛的就是用来翻盘的 都坚持了一年了 也不差这
  • C# 获取计算机信息

    文章目录 一 本机信息 1 本机名 2 获得本机MAC地址 3 获得计算机名 4 显示器分辨率 5 主显示器分辨率 6 系统路径 二 操作系统信息 1 操作系统类型 2 获得操作系统位数 3 获得操作系统版本 三 处理器信息 1 处理器个数
  • Sublime Text3设置文本的自动换行

    1 点击Preferences Settings 然后出现以下页面 2 点击保存即可 如果想要修改其他属性 可以直接在Default里面找就可以
  • Java正则表达式验证电话号码

    在注册会员时 经常需要输入电话号码 电话号码是指手机号码或者固定电话 如果输入的内容不合法 则会向用户输出提示 本实例模拟实现电话号码的验证功能 接收用户在控制台输入的电话号码 然后进行判断 并将结果输出 在这里使用 Java正则表达式 一
  • linux改变文件所属用户和组

    1 改变文件所属用户 chown 用户名 文件名 2 改变文件所属组 chgrp 用户名 文件名
  • 狂神说-Mybatis笔记(总)

    环境 JDK1 8 MySQL 8 0 23 maven 3 6 1 IDEA2020 3 框架 需要配置文件 官方中文文档 https mybatis org mybatis 3 zh index html 一 简介 1 什么是Mybat
  • 通俗易懂权限管理模块设计-Java

    最近一直在做CMS系统 发现一些内容其实都是重复出现的 例如权限管理模块 权限管理模块就是为了管理用户是否有权利访问某个权限 如果不能则拒绝访问 其实Java中已经有很成熟的权限管理框架 例如 Shiro Spring Security等
  • 怎么让人物脚贴地 模型_Unity利用FinalIK实现角色脚掌贴着地面行走工具

    using System Collections Generic using UnityEditor using UnityEngine public class FootBones public GameObject Bone1 publ
  • 设计原则学习之里氏替换原则

    以下内容均来自抖音号 it楠老师教java 的设计模式课程 1 原理概述 子类对象 objectofsubtype derivedclass 能够替换程序 program 中父类对象 objectofbase parentclass 出现的
  • C语言程序的undefined,c语言中undefined reference to ""怎么解决

    2 gcc c test c gcc c main c 得到两个 o 文件 一个是 main o 一个是 test o 然后我们链接 o 得到可执行程序 3 gcc o main main o这时 你会发现 报错了 4 main o In
  • python安装pandas失败问题

    开始使用pip install pandas报错 后来将pip语句更换为 pip default time 100 install pandas 成功安装pandas
  • 朋友拿下字节27K的offer,实名羡慕了....

    最近有朋友去字节面试 面试前后进行了20天左右 包含4轮电话面试 1轮笔试 1轮主管视频面试 1轮hr视频面试 据他所说 80 的人都会栽在第一轮面试 要不是他面试前做足准备 估计都坚持不完后面几轮面试 其实 第一轮的电话面试除了一些常规的
  • IBM也下场LLM了,自对齐、高效率的单峰驼Dromedary来了

    近期IBM Research发布了dromedary 并指出这个模型通过一种称为自对齐 SELF ALIGN 的新方法 结合了原则驱动 principle driven 的推理和LLM的生成能力 用于AI代理的自我对齐 使人类的监督最少化
  • 链式调用demo

    所谓链式调用就是调用完一个函数后还能再继续调用其它函数 这样大大减少了代码量 尤其是项目比较大的时候 逻辑集中清晰明了 且易于查看和修改 react hooks处理hooks原理用到了链式调用 fiber memorizedStaste h
  • 费马小定理题

    费马小定理 假如p是质数 且gcd a p 1 那么 A题HDU 4704 首先是挡板法 隔板法 然后用即可 高中数学范围不多叙述 然后得到答案是 这题读入数据大 就算快速幂也肯定TLE 所以用费马小定理 把数据规模降到int 范围内 时间
  • Re: Programming in C with Bluetooth Sockets

    Re Programming in C with Bluetooth Sockets This is the function run by the bluetooth device that will recieve the data c
  • 若依框架注册功能

    后台 逻辑峰的博客 CSDN博客 若依框架登录注册 前台
  • 响应式开发

    响应式开发是指一个网站能够兼容多个终端 不同屏幕分辨率的终端上网页展示方式是不一样的 实现原理 根据用户的行为以及设备的不同 实现页面的不同展示效果 具体的开发过程 1 设置视口标签 width 视口的宽度 device width 设备的
  • Java使用S7协议连接西门子PLC1200、1500

    Java使用S7协议连接西门子PLC1200 1500 1 引入s7包 2 测试代码 可参考使用 1 引入s7包 使用 https github com s7connector s7connector
  • CSDN周赛59期简要题解

    本期题目相对比较友好 而且在比赛报名界面还提示了非编程题考察的章节 诚不欺我 本期非编程题需要选手阅读的章节是第2章 逆向思考 从递推到递归 2 3节 堆栈和队列 遍历的数据结构 选择和判断都考到了栈的数据结构 稍微有点基础知识找出正确答案