我先谔谔一波 /kk 我谔谔

看题第一眼:欸这不就是按题意递归嘛,,直接搞不就好了

3 min 后,重新看题

然后自己手玩了几个样例,噢,递归太多了,铁定会 T 啊......

然后,作为一个从没写过记忆化搜索的蒟蒻,苦苦思索了十几分钟。。。。。。

然后忽然想到可以用一个数组记录每一次的状态,然后如果递归到重复的数字直接返回记录的值,这就叫记忆化搜索!

我竟然就这样把记忆化搜索给 yy 出来了......

然后就是无尽的采坑。。。


首先,就是开了一个叫 \(\mathtt{mem}\) 的三维数组,然后每次递归就查询对应的位置,如果有值就返回,否则再按题目的意思一个一个去把 \(\mathtt{mem}\) 数组赋值。写出来大概就是这个样子:

  1. long long w(long long a,long long b,long long c)
  2. {
  3. if(mem[a][b][c]!=0) return mem[a][b][c];
  4. if(a<=0||b<=0||c<=0) return 1;
  5. else if(a>20||b>20||c>20) mem[a][b][c]=w(20,20,20);
  6. else if(a<b&&b<c) mem[a][b][c]=w(a,b,c-1)+w(a,b-1,c-1)-w(a,b-1,c);
  7. else mem[a][b][c]=w(a-1,b,c)+w(a-1,b-1,c)+w(a-1,b,c-1)-w(a-1,b-1,c-1);
  8. }

然而又 WA 又 T 的,心酸。。。

然后突然发现数据范围: \([−9223372036854775808,9223372036854775807]\)

那递归开头就去判断如果下标是一个很大的负数咋办???岂不是会出现奇奇怪怪的错误???

所以,可以将函数里的第一行和第二行调换一下,这样就先去判断是否是负数了

  1. long long w(long long a,long long b,long long c)
  2. {
  3. if(a<=0||b<=0||c<=0) return 1;
  4. if(mem[a][b][c]!=0) return mem[a][b][c];
  5. else if(a>20||b>20||c>20) mem[a][b][c]=w(20,20,20);
  6. else if(a<b&&b<c) mem[a][b][c]=w(a,b,c-1)+w(a,b-1,c-1)-w(a,b-1,c);
  7. else mem[a][b][c]=w(a-1,b,c)+w(a-1,b-1,c)+w(a-1,b,c-1)-w(a-1,b-1,c-1);
  8. }

然而又 WA 了!!!咋回事啊???

于是经过长达十几分钟的手玩数据,发现最后不会返回答案!!!所以在函数最后返回这个值就好了......

  1. long long w(long long a,long long b,long long c)
  2. {
  3. if(a<=0||b<=0||c<=0) return 1;
  4. else if(mem[a][b][c]!=0) return mem[a][b][c];
  5. else if(a>20||b>20||c>20) mem[a][b][c]=w(20,20,20);
  6. else if(a<b&&b<c) mem[a][b][c]=w(a,b,c-1)+w(a,b-1,c-1)-w(a,b-1,c);
  7. else mem[a][b][c]=w(a-1,b,c)+w(a-1,b-1,c)+w(a-1,b,c-1)-w(a-1,b-1,c-1);
  8. return mem[a][b][c];
  9. }

这下呢???20分!!!

然后又是十几分钟的手玩数据,突然发现 \(\mathtt{mem}\) 数组的值每次不会被清空。换而言之,在以后我们每次都在使用上次记忆的值。

既然如此,每次把 \(\mathtt{mem}\) 数组清零不久好了???

这个时候主函数就被改成了这样

  1. int main()
  2. {
  3. int a,b,c;
  4. while(scanf("%d%d%d",&a,&b,&c)&&(a!=-1)&&(b!=-1)&&(c!=-1))
  5. printf("w(%d, %d, %d) = %d\n",a,b,c,w(a,b,c));
  6. return 0;
  7. }

还是不行!!!为啥???不难发现,若 \(a,b,c\) 的值过大,记忆数组存不下!!!

如何解决呢?考虑当 \(>20\) 时,并不会对 \(a,b,c\) 本身进行操作,换而言之,\(a=114514\) 和 \(a=21\) 本质是一样的!!!

  1. int main()
  2. {
  3. int a,b,c;
  4. while(scanf("%d%d%d",&a,&b,&c)&&(a!=-1)&&(b!=-1)&&(c!=-1))
  5. {
  6. memset(mem,0,sizeof(0));
  7. printf("w(%d, %d, %d) = ",a,b,c);
  8. //注意,下面的 a,b,c 要被修改,所以得先行输出
  9. if(a>20) a=21;
  10. if(b>20) b=21;
  11. if(c>20) c=21;
  12. printf("%d\n",w(a,b,c));
  13. }
  14. return 0;
  15. }

还是过不了!!!我要被搞疯了

最后还是把错误 yy 了出来, while 里的判断条件会使得如果 \(a,b,c\) 有一个是负数(不是 \(-1\) )就会结束......

最终 \(\mathtt{AC}\) 代码......

  1. #include <iostream>
  2. #include <stdio.h>
  3. #include <string.h>
  4. using namespace std;
  5. long long mem[101][101][101]={};
  6. long long w(long long a,long long b,long long c)
  7. {
  8. if(a<=0||b<=0||c<=0) return 1;
  9. else if(mem[a][b][c]!=0) return mem[a][b][c];
  10. else if(a>20||b>20||c>20) mem[a][b][c]=w(20,20,20);
  11. else if(a<b&&b<c) mem[a][b][c]=w(a,b,c-1)+w(a,b-1,c-1)-w(a,b-1,c);
  12. else mem[a][b][c]=w(a-1,b,c)+w(a-1,b-1,c)+w(a-1,b,c-1)-w(a-1,b-1,c-1);
  13. return mem[a][b][c];
  14. }
  15. int main()
  16. {
  17. long long a,b,c;
  18. while(scanf("%lld%lld%lld",&a,&b,&c)==3)
  19. {
  20. if((a==-1)&&(b==-1)&&(c==-1)) break;
  21. memset(mem,0,sizeof(0));
  22. printf("w(%lld, %lld, %lld) = ",a,b,c);
  23. if(a>20) a=21;
  24. if(b>20) b=21;
  25. if(c>20) c=21;
  26. printf("%lld\n",w(a,b,c));
  27. }
  28. return 0;
  29. }

所以说我仍然谔谔

【做题笔记】洛谷P1464 Function的更多相关文章

  1. (水题)洛谷 - P1464 - Function

    https://www.luogu.org/problemnew/show/P1464 #include<bits/stdc++.h> using namespace std; #defi ...

  2. 洛谷P1464 Function  HDU P1579 Function Run Fun

    洛谷P1464 Function HDU P1579 Function Run Fun 题目描述 对于一个递归函数w(a,b,c) 如果a≤0 or b≤0 or c≤0就返回值11. 如果a> ...

  3. 洛谷 P1464 Function【记忆化搜索】

    题目链接 题目描述 对于一个递归函数w(a,b,c) 如果a<=0 or b<=0 or c<=0就返回值1. 如果a>20 or b>20 or c>20就返回w ...

  4. 洛谷 P1464 Function【动态规划(递推)/记忆化搜索(递归)】

    题目描述 对于一个递归函数w(a,b,c) 如果a<=0 or b<=0 or c<=0就返回值1. 如果a>20 or b>20 or c>20就返回w(20,2 ...

  5. 洛谷 P1464 Function

    题目描述 对于一个递归函数w(a,b,c) 如果a<=0 or b<=0 or c<=0就返回值1. 如果a>20 or b>20 or c>20就返回w(20,2 ...

  6. 洛谷P1464 Function

    对于一个递归函数w(a,b,c)w(a,b,c) 如果a \le 0a≤0 or b \le 0b≤0 or c \le 0c≤0就返回值11. 如果a>20a>20 or b>20 ...

  7. 洛谷 P1464 Function(简单记忆化)

    嗯... 让一切从水开始吧... 水过初赛,但愿复赛能够接着水过... 这道题不记忆化会tle,所以用空间换时间,将每次的答案(只有20*20*20个)存下来,如果之前已经求过,就不需要重复求了... ...

  8. java实现 洛谷 P1464 Function

    import java.util.HashMap; import java.util.Iterator; import java.util.LinkedList; import java.util.S ...

  9. C语言程序设计做题笔记之C语言基础知识(下)

    C 语言是一种功能强大.简洁的计算机语言,通过它可以编写程序,指挥计算机完成指定的任务.我们可以利用C语言创建程序(即一组指令),并让计算机依指令行 事.并且C是相当灵活的,用于执行计算机程序能完成的 ...

随机推荐

  1. P1462 通往奥格瑞玛的道路【二分+Dij】

    P1462 通往奥格瑞玛的道路 提交 29.89k 通过 6.88k 时间限制 1.00s 内存限制 125.00MB 题目提供者gconeice 难度提高+/省选- 历史分数100 提交记录 查看题 ...

  2. 09 : 构造方法 & 代码块

    构造方法 概念 构造方法是一种特殊的方法,它是一个与类同名的方法 对象的创建就是通过构造方法来完成. 其功能主要是完成对象的创建或者对象的初始化 当类实例化new一个对象时会自动调用构造方法 构造方法 ...

  3. webpack打包后不能调用,改用uglifyjs打包压缩

    背景: 项目基于原生js,没用到任何脚手架和框架,但也需要打包压缩. 项目的js中声明了一些全局变量 供其他js调用. 这时候如果用webpack打包,基于webpack特性,会嵌套一层大函数,会将j ...

  4. Silver Cow Party POJ - 3268

    #include<iostream> #include<queue> #include<cstring> using namespace std; +,INF=0x ...

  5. D - How Many Answers Are Wrong HDU - 3038【带权并查集】

    How Many Answers Are Wrong Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Ja ...

  6. babel 的简单使用

    之前在项目中使用.balelrc文件,但是一直不知道具体怎么使用,就知道可以将es6语法转码为es5语法. 今天就简单的做个例子,也算是记录一下困扰了好久的问题. 转码步骤: 首先在项目的目录中安装B ...

  7. 微信小程序配置合法域名和业务域名

    在微信小程序的开发过程中,当需要请求第三方网站数据时,都是直接调用wx.request接口的: xxxx:function(){ wx.request({ url: 'xxxxxxxxxx', dat ...

  8. 最长公共子串(LCS) lg SP1811

    后缀自动机的一大用处就是求最长公共子串了 这道题的话题意就是给你两个字符串,求最长公共子串 做法的话是先使用一个字符串建立SAM,然后让另一个串在上面进行匹配 匹配的策略是优先匹配当前节点的下一个字符 ...

  9. redis分布式锁在springboot中的实现

    理论知识   redis分布式锁的实现方案请参考文章 如何优雅地用redis实现分布式锁 本案例简介   以秒杀活动为例子,在多线程高并发的情况下需要保证秒杀业务的线程安全性,确保秒杀记录与所扣库存数 ...

  10. SpringBoot学习- 6、MetaData描述无法tip显示问题

    SpringBoot学习足迹 (一)先说说现象 1.在application.properties可以定义属性并在MetaData中增加description 如 生成additional-sprin ...