本题是浙江理工大学ACM入队200题第八套中的J题

我们先来看一下这题的题面.

题面

题目描述

宁宁参加奥数班,他遇到的第一个问题是这样的:口口口+口口口=口口口,宁宁需要将1~9 九个数分别填进对应的空格内,使等式成立。现在宁宁填了一个算式,你能帮他验证是否正确么?

输入

输入为多组测试数据。

分别输入三个三位数,依次表示等式里的三个数。

输出

如果等式成立,输出:YES!,否则输出:NO!

样例输入

173 286 459

样例输出

YES!


题目分析与常见错误思路

这题很多朋友都想当然地以为只需要判断等式是否成立即可,都选择性的忽视了题目中明确写出的要求:将1~9九个数分别填进对应的空格内.

所以我们需要额外实现判断没有使用重复的数.

解决方案

如何实现捏?首先我们需要依次取出每个数的各个十进制位,这里我已经有一篇博客给出明确的方法介绍了,不清楚的可以去看看(点我看鸭),这里不多赘述了.

接下来困住大多数朋友的都是判断重复数字了,很多朋友对此毫无思路.

我们先来想想我们人脑是怎么判断有没有数字重复的呢?打个比方我们要在一组数字中找第一个数是否重复,那么当我们看到这组数字中的一个数后,我们会接着往后去看有没有第二个这个数出现,如果有就是重复了,否则就是没有.

我们是否也可以这样实现我们这里判断重复的代码捏?答案是可行的,我们可以把这三个数的每个位合起来看成一个"数组"中的元素,然后从左到右遍历这个"数组",再对于每一个遍历到的数之后的数进行一次遍历,看看其中是否有和这个数相等的数存在.不过这种思路遍历次数较多,比较接近暴力算法,时间复杂度很高(可以理解成代码运行的速度很慢),属于一种无解算法.所以我们需要寻找一个更好的算法.(这种思路的代码就不贴了,想练代码可以自己写写试试,交上去不清楚会不会时间超限,没试过)

我们前面的算法慢在哪呢?慢在我们每次都对遍历到的数再启动了一次遍历,这使得我们遍历的次数过多.而且每次遍历都是在重复遍历这些数.我们是否可以避免这些重复的操作,以此提高速度呢?答案是可行的.

那如何避免重复操作呢?我们可以对遍历到的数进行一次记录,记录什么呢?我们上面这样判断的本质其实是在看有没有哪个数字重复出现了两次及以上,所以我们就记录每个数字出现了几次即可,这样仅通过一次遍历,我们便得到了可以判断是否有重复数字的数据,大大减少了遍历的次数.这其实便是所谓的用空间换时间的技巧.

那用什么记录呢?我们目前会的数据结构非常少,基本只有数组.那能否通过数组来记录呢?当前可以!我们希望记录的形式是"数字:数字出现次数"这样形式的一个对子,并且我们可以通过数字来访问到这个数字出现的次数.这个对子和数学上的映射很像.而这个映射完全可以由数组来完成,因为数组本身的随机访问也可以看成一种映射.

我们知道,当我们给定数组中某个元素的下标,我们即可得到这个下标对应的那个元素(即随机访问),这便是一种现成的映射.那我们自然可以通过把数字作为下标,把数字出现的次数作为下标对应的元素,来充分利用数组自带的这种映射实现我们需要的功能.即,我们使用一个长度为10的数组,在其下标1到9的各个元素中存储1到9各个数字出现的次数.这样我们就可以通过各个数字来直接访问到各个数字出现的个数了.在这里,这个数组也被称作,是不是很形象鸭?当然,如果有一定数据结构基础的朋友看到我们的需求,应该会马上联想到哈希表,不过这题没必要使用,因为桶广义地来讲也是一种哈希表(哈希函数返回这个数字本身).

参考代码

下面给出了我自己做这道题时候的完整代码:

(仅作为参考,一定要自己写一下奥,作弊没意思,害人又害己)

#include <stdio.h>
#include <string.h>
// 数组最好定义在main外面(以后你们会知道为啥的)
int num[10]; // 作为桶,下标所对应的元素储存着下标这个数字出现的次数,由此形成一个一一映射 int main()
{
int a, b, c;
while (scanf("%d%d%d", &a, &b, &c) != EOF)
{
if (a + b != c) // 如果不满足加法式,直接输出然后输入下一组
{
printf("NO!\n");
continue; // 跳过本组处理,开始下一组的输入
}
for (int i = 0; i < 3; i++) // 各个数字依次取出各个位
{
num[a % 10]++; // 下标为该数字的元素储存着该数字出现的次数,将该数字的出现次数加一
a /= 10;
num[b % 10]++;
b /= 10;
num[c % 10]++;
c /= 10;
}
int i; // 为了判断for是否正常退出这里我i定义在外面,也可以在单独搞一个变量来记录是否有数重复
for (i = 1; i < 9; i++)
{
if (num[i] > 1) // 如果一个数字出现了两次
{
printf("NO!\n");
break; // 直接跳出这个for循环,不用再找有没有重复了(只要一组重复就不满足了)
}
}
if (i == 9) // for正常退出时i应为9,如果不为9就是被break强行退出了
{
printf("YES!\n");
}
memset(num, 0, sizeof(num)); // 桶初始化(别忘了鸭!不然就会连着上次输入一起累加了)如果不会用memset也可以考虑直接写一个for循环把每一元素设置为0
}
return 0;
}

"正是我们每天反复做的事情,最终造就了我们,优秀不是一种行为,而是一种习惯" ---亚里士多德

这篇题解就到这里了,各位朋友如果有问题欢迎到acm成员群中提问哦!

「浙江理工大学ACM入队200题系列」问题 J: 零基础学C/C++83——宁宁的奥数路的更多相关文章

  1. 「浙江理工大学ACM入队200题系列」问题 L: 零基础学C/C++85——完美数

    本题是浙江理工大学ACM入队200题第八套中的L题 我们先来看一下这题的题面. 题面 题目描述 任何一个自然数的约数中都有1和它本身,我们把小于它本身的因数叫做这个自然数的真约数. 如6的所有真约数是 ...

  2. 「浙江理工大学ACM入队200题系列」问题 K: 零基础学C/C++84——奇偶ASCII值判断

    本题是浙江理工大学ACM入队200题第八套中的K题 我们先来看一下这题的题面. 题面 题目描述 任意输入一个字符,判断其ASCII是否是奇数,若是,输出YES,否则,输出NO; 例如,字符A的ASCI ...

  3. 「浙江理工大学ACM入队200题系列」问题 E: 零基础学C/C++78——求奇数的乘积

    本题是浙江理工大学ACM入队200题第八套中的E题 我们先来看一下这题的题面. 题面 输入 输入数据包含多个测试实例,每个测试实例占一行,每行的第一个数为n,表示本组数据一共有n个,接着是n个整数,你 ...

  4. 「浙江理工大学ACM入队200题系列」问题 L: 零基础学C/C++52——计算数列和2/1,3/2,5/3,8/5......

    本题是浙江理工大学ACM入队200题第五套中的L题 我们先来看一下这题的题面. 题面 题目描述 有一分数序列:2/1,3/2,5/3,8/5,13/8,21/13,-- 计算这个数列的前n项和.注意: ...

  5. 「浙江理工大学ACM入队200题系列」问题 F: 零基础学C/C++39——求方程的解

    本题是浙江理工大学ACM入队200题第四套中的F题 我们先来看一下这题的题面. 由于是比较靠前的题目,这里插一句.各位新ACMer朋友们,请一定要养成仔细耐心看题的习惯,尤其是要利用好输入和输出样例. ...

  6. 「浙江理工大学ACM入队200题系列」问题 A: 零基础学C/C++34—— 3个数比较大小(冒泡排序与选择排序算法)

    本题是浙江理工大学ACM入队200题第四套中的A题,同时给出了冒泡排序和选择排序算法 我们先来看一下这题的题面. 由于是比较靠前的题目,这里插一句.各位新ACMer朋友们,请一定要养成仔细耐心看题的习 ...

  7. 「浙江理工大学ACM入队200题系列」问题 H: 零基础学C/C++18——三位数反转

    本题是浙江理工大学ACM入队200题第二套中的H题 我们先来看一下这题的题面. 由于是比较靠前的题目,这里插一句.各位新ACMer朋友们,请一定要养成仔细耐心看题的习惯,尤其是要利用好输入和输出样例. ...

  8. 「浙江理工大学ACM入队200题系列」问题 B: 零基础学C/C++12——求平均值

    本题是浙江理工大学ACM入队200题第二套中的B题 我们先来看一下这题的题面. 由于是比较靠前的题目,这里插一句.各位新ACMer朋友们,请一定要养成仔细耐心看题的习惯,尤其是要利用好输入和输出样例. ...

  9. [Python] 文科生零基础学编程系列二——数据类型、变量、常量的基础概念

    上一篇:[Python] 文科生零基础学编程系列--对象.集合.属性.方法的基本定义 下一篇: (仍先以最简单的Excel的VBA为例,语法与Python不同,但概念和逻辑需要理解透彻) p.p1 { ...

随机推荐

  1. 拥挤的奶牛题解---队列优化DP---DD(XYX)​​​​​​​的博客

    拥挤的奶牛 时间限制: 1 Sec  内存限制: 128 MB 题目描述 FJ的n头奶牛(1<=n<=50000)在被放养在一维的牧场.第i头奶牛站在位置x(i),并且x(i)处有一个高度 ...

  2. 7个技巧让你写出干净的 TSX 代码

    原文链接:https://dev.to/ruppysuppy/7-tips-for-clean-react-typescript-code-you-must-know-2da2 "干净的代码 ...

  3. .NET 7 SDK 对容器的内置支持

    大家好,我是等天黑. .NET 又双叒叕出新功能了, 就在前几天,.NET 团队发布博客,宣称在 .NET SDK 中内置了对 .NET 应用容器化的支持. 小试牛刀 在正式介绍它之前,先通过一个简单 ...

  4. CF-1684C - Column Swapping

    Problem - 1684C - Codeforces 题意: 现在有一个n*m的棋盘,每个棋子有一个值,你可以交换两列棋盘的棋子位置,使得每一行的棋子从左到右为非递减. 题解: 只需要判断一行不满 ...

  5. Go 语言入门 1-管道的特性及实现原理

    入坑 go 也快一年了,从今天开始会定期分享一下 Go 语言学习过程中的一些基础知识. go 语言中的管道, 主要是用于协程之间的通信, 比 UNIX 的管道更加轻量和易用. 我们先看一下管道的数据结 ...

  6. Merge Into 语法支持

    KINGBASE 兼容Oracle 语法,实现了merge into 的功能.以下以例子的形式,介绍merge into语法的使用.以下例子在V8R6 ,且 database_mode=oracle ...

  7. KingbaseES blob 类型数据导入导出

    KingbaseES兼容了oracle的blob数据类型.通常是用来保存二进制形式的大数据,也可以用来保存其他类型的数据. 下面来验证一下各种数据存储在数据库中形式. 建表 create table ...

  8. KingbaseES sys_prewarm 扩展

    Oracle 在查询数据 可以通过cache hint 所访问的数据cache 到数据库buffer,对于KingbaseES,如何将数据加载到cache 了?sys_prewarm 扩展插件可以实现 ...

  9. Self-Attention:初步理解

    Self-Attention 的基本结构与计算 Attention(注意力)实际上就是权重的另一种应用的称呼,其具体结构与初始输入的 content \(\vec{x_{1}}, \vec{x_{2} ...

  10. haodoop企业优化

    MapReduce 跑的慢的原因 MapReduce程序效率的瓶颈在于两点 计算机性能 CPU,内存,磁盘健康,网络 I/O操作优化 数据倾斜 Map和Reduce数设置不合理 Map运行时间太长,导 ...