POJ 3373 Changing Digits 记忆化搜索
这道题我是看了别人的题解才做出来的。题意和题解分析见原文http://blog.csdn.net/lyy289065406/article/details/6698787
这里写一下自己对题目的理解。
1. 根据k的最大范围直接搜索n最后5位的方法是错误的,因为它并不能保证所求结果为最小。因为题目要求最后结果m要尽量小,而改变n的高位能够得到更小的值。k<n的限制条件表明解必然存在,而我们搜索的最大可修改位数应该和n的位数len相等。
2. 最终结果m必须满足:最高位非0且与n位数相同 (m若等于0不视为最高位为0),m能被k整除。在此基础上,还有两个条件,它们的优先级为:m与n对应位置上的数不相等的个数尽量少 > m尽量小。即最终得到的结果不一定是最小的,但一定是与n对应位置数字不相等个数最少的。这也决定了本题dfs的写法。
3. 面对大量大数取模运算,利用mod[i][j]预处理(10^i)*j模k的值,节省了大量时间。这样在dfs的过程中,改变某一位后的m%k的值也能计算出来了。搜索时为了求得的结果最小,先从m<n开始搜,然后再搜m>n。
前者有res = (m_modk - (mod[i][n[i]] - mod[i][j]) + k) % k;
后者有res = (m_modk + (mod[i][j] - mod[i][n[i]]) + k) % k;
4. flag数组的引入是为了剪枝。如果当搜索区间为[0, pos]且此时m模k为m_modk时,如果最多修改restnum位不能成功,则修改次数少于restnum时更不可能成功,因此就不用搜索下去了。flag[pos][m_modk]始终维护上述情况无法成功的最大restnum。
我的代码
#include<stdio.h>
#include<string.h>
#define maxn 103
#define mk 10003
int len, k, n[maxn], mod[maxn][];
int m[maxn], flag[maxn][mk];
char num[maxn];
void init_mod()//mod[i][j]表示(10^i)*j模k的值
{
for (int i = ; i <= ; i++)
mod[][i] = i % k;
for (int i = ; i < len; i++)
for (int j = ; j <= ; j++)
mod[i][j] = (mod[i-][j] * ) % k;
}
int dfs(int pos,int restnum,int m_modk)
{
if (!m_modk) return ;
if (!restnum || pos < ) return ;
if (restnum <= flag[pos][m_modk]) return ;//剪枝
for (int i = pos; i > -; i--)//搜索比n小的数,要尽可能小,则从高位开始
for (int j = ; j < n[i]; j++)
{
if (i == len - && !j) continue;
int res = (m_modk - (mod[i][n[i]] - mod[i][j]) + k) % k;
m[i] = j;
if (dfs(i - , restnum - , res))
return ;
m[i] = n[i];
}
for (int i = ; i <= pos; i++)//搜索比n大的数,要尽可能小,则从低位开始
for (int j = n[i] + ; j < ; j++)
{
int res = (m_modk + (mod[i][j] - mod[i][n[i]]) + k) % k;
m[i] = j;
if (dfs(i - , restnum - , res))
return ;
m[i] = n[i];
}
flag[pos][m_modk] = restnum;//能运行到这里说明搜索失败,更新剪枝数值
return ;
}
int main()
{
while (~scanf("%s%d", num, &k))
{
int n_modk = ;
len = strlen(num);
init_mod();
for (int i = ; i < len; i++)//将num反序存入整型数组
{
n[i] = num[len--i] - '';
m[i] = n[i];
n_modk = (n_modk + mod[i][ n[i] ]) % k;//计算n % k
}
memset(flag, , sizeof(flag));
int ok = ;
for (int i = ; i <= len; i++)//从小到大枚举可以修改的位数
if (dfs(len - , i, n_modk))
break;
for (int i = len - ; i > -; i--)
printf("%d", m[i]);
printf("\n");
}
return ;
}
POJ 3373 Changing Digits 记忆化搜索的更多相关文章
- poj 3373 Changing Digits (DFS + 记忆化剪枝+鸽巢原理思想)
http://poj.org/problem?id=3373 Changing Digits Time Limit: 3000MS Memory Limit: 65536K Total Submi ...
- POJ 1088 滑雪(记忆化搜索)
滑雪 Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 92384 Accepted: 34948 Description ...
- POJ 1088 滑雪 DFS 记忆化搜索
http://poj.org/problem?id=1088 校运会放假继续来水一发^ ^ 不过又要各种复习,功课拉下了许多 QAQ. 还有呀,就是昨天被一个学姐教育了一番,太感谢了,嘻嘻^ ^ 好了 ...
- POJ 1088 滑雪【记忆化搜索】
题意:给出一个二维矩阵,要求从其中的一点出发,并且当前点的值总是比下一点的值大,求最长路径 记忆化搜索,首先将d数组初始化为0,该点能够到达的路径长度保存在d数组中,同时把因为路径是非负的,所以如果已 ...
- poj 1088 滑雪_记忆化搜索
题意:略 直接用记忆化搜索就行了 #include<cstdio> #include<iostream> using namespace std; int n,m; int m ...
- POJ 3373 Changing Digits 好蛋疼的DP
一開始写的高位往低位递推,发现这样有些时候保证不了第四条要求.于是又開始写高位往低位的记忆化搜索,又发现传參什么的蛋疼的要死.然后又发现高位開始的记忆化搜索就是从低位往高位的递推呀,遂过之. dp[i ...
- POJ 3176-Cow Bowling(DP||记忆化搜索)
Cow Bowling Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 14210 Accepted: 9432 Desc ...
- POJ 3373 Changing Digits
题目大意: 给出一个数n,求m,使得m的长度和n相等.能被k整除.有多个数符合条件输出与n在每位数字上改变次数最小的.改变次数同样的输出大小最小的. 共同拥有两种解法:DP解法,记忆化搜索的算法. ...
- HDU 1501 & POJ 2192 Zipper(dp记忆化搜索)
题意:给定三个串,问c串是否能由a,b串任意组合在一起组成,但注意a,b串任意组合需要保证a,b原串的顺序 例如ab,cd可组成acbd,但不能组成adcb. 分析:对字符串上的dp还是不敏感啊,虽然 ...
随机推荐
- mysql-update时where条件无索引锁全表
1 5.3日数据处理需求 UPDATE md_meter set warranty_end_date = DATE_ADD(warranty_begin_date,INTERVAL 10 ...
- 查询语句为“%string_”的情况
select * from t_user where user_name like '%Joe_%'实际查询出来的语句为: 而不像预计的前两条.
- JVM——自定义类加载器
)以上两种情况在实际中的综合运用:比如你的应用需要通过网络来传输 Java 类的字节码,为了安全性,这些字节码经过了加密处理.这个时候你就需要自定义类加载器来从某个网络地址上读取加密后的字节代码,接着 ...
- Nodejs-文件流
1.什么是流? 流是程序输入输出的一个连续的字节序列. 有文件流,网络流,设备(例如鼠标,键盘,磁盘,调制解调器和打印机)的输入输出都是用流来处理的. 任何数据的最根本表现形式都是二进制. 读取文件 ...
- Python基础-week02 Python的常用数据类型
一.模块初识 import导入Py自带模块例如os,sys等及其自己编写的Py文件,导入到其他文件中,默认查找当前目录.如果不在同一目录,会报错,将该自定义py文件模块放到site-packages目 ...
- Leetcode 630.课程表III
课程表III 这里有 n 门不同的在线课程,他们按从 1 到 n 编号.每一门课程有一定的持续上课时间(课程时间)t 以及关闭时间第 d 天.一门课要持续学习 t 天直到第 d天时要完成,你将会从第 ...
- sqlserver 获取一个月有多少天
--思路:给定日期的下一个月的1号减去1天,然后取datepart(DAY,dt) declare @dt varchar(10)select @dt='2013-11-20'select datep ...
- [oldboy-django][2深入django]django一个请求的生命周期 + WSGI + 中间件
1 WSGI # WSGI(是一套协议,很多东西比如wsgiref, uwsgiref遵循这一套协议) - django系统本质 别人的socket(wsgiref或者uwsgiref) + djan ...
- 手机安装app总是显示未安装
手机安装软件总是显示未安装 查看是否开启了护眼模式或者护眼工具等干扰屏幕的软件.关掉,再安装即可
- 34条简单的SQL优化准则
转载地址:http://bbs.csdn.net/topics/260002113 我们要做到不但会写SQL,还要做到写出性能优良的SQL,以下为笔者学习.摘录.并汇总部分资料与大家分享!(1) ...