ZOJ3675:Trim the Nails
Robert is clipping his fingernails. But the nail clipper is old and the edge of the nail clipper is potholed.
The nail clipper's edge is N millimeters wide. And we use N characters('.' or '*') to represent the potholed nail clipper. '.' represents 1 bad millimeter edge, and '*' represents 1 good millimeter edge.(eg. "*****" is a 5 millimeters nail clipper with the whole edge good. "***..." is a 6 millimeters nail clipper with half of its edge good and half of its edge bad.)
Notice Robert can turn over the clipper. Turning over a "**...*"-nail clipper will make a "*...**"-nail clipper.
One-millimeter good edge will cut down Robert's one-millimeter fingernail. But bad one will not. It will keep the one-millimeter unclipped.
Robert's fingernail is M millimeters wide. How many times at least should Robert cut his fingernail?
Input
There will be multiple test cases(about 15). Please process to the end of input.
First line contains one integer N.(1≤N≤10)
Second line contains N characters only consists of '.' and '*'.
Third line contains one integer M.(1≤M≤20)
Output
One line for each case containing only one integer which is the least number of cuts. If Robert cannot clipper his fingernail then output -1.
Sample Input
8 ****..** 4 6 *..*** 7
Sample Output
1 2
Hint
We use '-' to present the fingernail. For sample 1: fingernail: ---- nail clipper: ****..** Requires one cut. For sample 2: fingernail: ------- nail clipper: *..*** nail clipper turned over: ***..* Requires two cuts. 这个题目对于我来说还是太难了,现在对dfs仅仅是知道贴上大牛代码一份,留作复习研究
解法一:
#include <stdio.h>
#include <math.h>
#include <string.h>
#include <algorithm>
using namespace std;
int n,m,right,left,minn;
];
];
void dfs(int fig[],int s,int flag,int step)
{
];
if(step>=minn) return ;
if(flag)
{
; i<=; i++)
tem[i] = fig[i];
for(i = s,j = left; i<=m && j<=n; i++,j++)
{
if(!tem[i])
tem[i] = clp[j];
}
for(i = s; i<=m; i++)
{
if(!tem[i])
{
s = i;
break;
}
}
)
{
if(step<minn)
{
minn = step;
}
return ;
}
dfs(tem,s,,step+);
dfs(tem,s,,step+);
}
else
{
; i<=; i++)
tem[i] = fig[i];
; i++,j--)
{
if(!tem[i])
tem[i] = clp[j];
}
for(i = s; i<=m; i++)
{
if(!tem[i])
{
s = i;
break;
}
}
)
{
if(step<minn)
{
minn = step;
}
return ;
}
dfs(tem,s,,step+);
dfs(tem,s,,step+);
}
}
int main()
{
];
while(~scanf("%d",&n))
{
scanf("%s",str);
memset(clp,,sizeof(clp));
; i<n; i++)
{
if(str[i] == '*')
clp[i+] = ;
}
left = ;
; i<=n; i++)
if(clp[i])
{
left = i;
break;
}
; i--)
if(clp[i])
{
right = i;
break;
}
scanf("%d",&m);
)
{
printf("-1\n");
continue;
}
minn = ;
memset(fig,,sizeof(fig));
dfs(fig,,,);
dfs(fig,,,);
printf("%d\n",minn);
}
;
}
解法二:
首先把 (1<<m)-1 作为指甲没剪时的初态(全是1),dp[x]保存的就是对于每个状态,需要剪的最少次数。
当前状态x以二进制表示时,出现的1表示这一位置还留有指甲,0就是已剪去。而对于指甲钳,又可以将其以二进制表示,对于案例:****..**,不妨用11110011代替,1表示当前位置刀锋完好,0相反。于是对每一位分析:
| 原来指甲情况 A | 指甲钳刀锋 B | 最终指甲情况 Y |
| 0 | 0 | 0 |
| 0 | 1 | 0 |
| 1 | 0 | 1 |
| 1 | 1 | 0 |
化简一下得到每一位上的关系:Y = A & ~B
所以递推方程就是:dp[x & (~B)] = min(dp[x & (~B)], dp[x] + 1);
问题是,指甲钳并不总是和指甲最左端对齐,所以还需要对指甲钳进行移动。反应在上式,就是对B进行左右各m次的移位操作。另外,指甲钳可以反着用,于是B还需要分正反情况考虑。
#include <cstdio>
#include <cstring>
#include <iostream>
using namespace std;
#define INF 0x3f3f3f3f
<< ];
],rev[];
int main() {
int n,m,left,right;
while(~scanf("%d%s%d",&n,s,&m))
{
;i<n;i++)
rev[i]=s[n-i-];
left=; right=;
;i<n;i++)
{
left=left<<; right=right<<;
if(s[i]=='*')
left|=;
if(rev[i]=='*')
right|=;
}
;i<<<m;i++)
dp[i]=INF;
dp[(<<m)-]=;
<<m)-;i>=;i--)
{
if(dp[i]==INF)// 由i 剪到 i&(~(left<<j)的状态 若 。。 说明这还没被剪过
continue;
;j<m;j++)
{
dp[i&(~(left<<j))]=min(dp[i&(~(left<<j))],dp[i]+);
dp[i&(~(left>>j))]=min(dp[i&(~(left>>j))],dp[i]+);
dp[i&(~(right<<j))]=min(dp[i&(~(right<<j))],dp[i]+);
dp[i&(~(right>>j))]=min(dp[i&(~(right>>j))],dp[i]+);
}
}
]<INF)
printf(]);
else printf("-1\n");
}
;
}
ZOJ3675:Trim the Nails的更多相关文章
- ZOJ 3675 Trim the Nails
http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=4918 DP+状态压缩. http://www.cnblogs.com/dgsr ...
- ZOJ 3675 Trim the Nails(bfs)
Trim the Nails Time Limit: 2 Seconds Memory Limit: 65536 KB Robert is clipping his fingernails. ...
- 2014 Super Training #6 G Trim the Nails --状态压缩+BFS
原题: ZOJ 3675 http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3675 由m<=20可知,可用一个二进制数表 ...
- ZOJ Monthly, November 2012
A.ZOJ 3666 Alice and Bob 组合博弈,SG函数应用 #include<vector> #include<cstdio> #include<cstri ...
- [PHP源码阅读]trim、rtrim、ltrim函数
trim系列函数是用于去除字符串中首尾的空格或其他字符.ltrim函数只去除掉字符串首部的字符,rtrim函数只去除字符串尾部的字符. 我在github有对PHP源码更详细的注解.感兴趣的可以围观一下 ...
- php的empty(),trim(),strlen()方法
如果empty()函数的参数是非空或非零的值,则empty()返回FALSE.换句话说,"".0."0".NULL.array().var$var:以及没有任何 ...
- ORACLE中的LTRIM、RTRIM和TRIM
LTRIM.RTRIM和TRIM在ORACLE中的用法:1.LTRIM(C1,C2)其中C1和C2都可以字符串,例如C1是'Miss Liu',C2'MisL'等等.这是第一个和SQL SERVER不 ...
- mybatis : trim标签, “等于==”经验, CDATA标签 ,模糊查询CONCAT,LIKE
一.My Batis trim标签有点类似于replace效果. trim 属性, prefix:前缀覆盖并增加其内容 suffix:后缀覆盖并增加其内容 prefixOverrides:前缀判断的条 ...
- IE8下String的Trim()方法失效的解决方案
简洁方便 用jquery的trim()方法,$.trim(str)就可以了.
随机推荐
- android 主线程和子线程之间的消息传递
从主线程发送消息到子线程(准确地说应该是非UI线程) package com.zhuozhuo; import android.app.Activity; import android.os.Bun ...
- Android Contacts (android通讯录读取)-content provider
Content Provider 在数据处理中,Android通常使用Content Provider的方式.Content Provider使用Uri实例作为句柄的数据封装的,很方便地访问地进行数据 ...
- Echarts柱形图颜色设置
ECharts图为每个数据项配置颜色 (2014-11-12 15:52:53) 转载▼ 标签: 时尚 分类: 开发学习 其实给每个数据项配置很简单 只需要在series里面给data数组的每个元素设 ...
- JavaScript的事件对象_事件流
事件流事件流是描述的从页面接受事件的顺序,当几个都具有事件的元素层叠在一起的时候,那么你点击其中一个元素,并不是只有当前被点击的元素会触发事件,而层叠在你点击范围的所有元素都会触发事件.事件流包括两种 ...
- spring mvc获取request HttpServletRequest
1.最简单的方式(注解法) 2. 直接的方法,参数中添加(response类似) package spittr.web; import static org.springframework.web.b ...
- 数据存储之SQLite
SQLite是目前主流的嵌入式关系型数据库,其最主要的特点就是轻量级.跨平台,当前很多嵌入式操作系统都将其作为数据库首选.虽然SQLite是一款轻型数据库,但是其功能也绝不亚于很多大型关系数据库.学习 ...
- hiho_1051_补提交卡
题目大意 给出1到100这100个数中的某些数字(各个数字不同),这些数字形成一个个间断的连续区间,向1-100中添加M个数字,使得添加后1-100中某连续区间的长度最大,求出添加M个数字后,最长的连 ...
- vsphere vcenter 添加域管理员
选择 vcenter server 管理权限
- 如何查看连接mysql的ip地址
select SUBSTRING_INDEX(host,':',1) as ip , count(*) from information_schema.processlist group by ip;
- IoC 之 2.1 IoC基础(壹)
2.1.1 IoC是什么 Ioc-Inversion of Control,即"控制反转",不是什么技术,而是一种设计思想.在Java开发中,Ioc意味着将你设计好的对象交给容器 ...