noip复赛复习15反转问题与弹性碰撞.docxVIP

  1. 1、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。。
  2. 2、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载
  3. 3、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
  4. 4、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
  5. 5、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们
  6. 6、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
  7. 7、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
noip复赛复习15反转问题与弹性碰撞

NOIP 复赛复习 15 反转问题与弹性碰撞一、反转问题算法概览:给定一个 01 串,现有翻转规则:翻转某一个位置时其后面 2 个位置也会跟着翻转,也就是每次翻转都会翻转 3 个连续的位置。要将 01 串全部翻转为 0,求最小的翻 转次数。形似这类题的问题叫做反转问题,也可以叫开关问题,对于这类题通常有以下的特 点,思考一下就可以想到。1、若某一个位置被翻转了 n 次,则其实际上被翻转了 n%2 次,因为翻转 2k 次相当与没翻转,翻转 2k+1 次相当于翻转了 1 次,因为要求最小翻转次数,所以对于某一个位置要 么只(主动)翻转一次,要么不(主动)翻转。2、分析易知翻转的顺序并不影响最终结果。(理解不了可自己举个例子在纸上模拟下)3、由此,反转问题(也叫开关问题)就转化成求反转区间的集合,常常和枚举一起使用。POJ 3276N 头牛排列成了一列,每头牛或者向前或者向后站,为了让所有的牛都面向前方,农夫约翰买了一台自动转向的机器,这个机器在购买时就必须设定一个数值 K,机器每操作一次恰好使 K 头连续的牛转向(K 头牛分别为当前的牛及其之后的牛,不影响位于它之前的牛,并且每次反转必须是 K 头牛,不可以少于 K 头)。请求出为了让所有的牛都能面向前方需要的最少的操作次数 M 和对应的最小的 K。已知:1≤N≤5000sampleinputN= 7BBFBFBB(F:面向前方,B:面向后方)sampleoutputK= 3M= 3(先反转 1~3 号的三头牛,然后再反转 3~5 号,最后反转 5~7 号)思路:定义 f[i]:区间[i,i+k-1]进行反转的话就为 1,否则为 0区间反转部分很好优化:在考虑第 i 头牛时候,如果∑i?1j=(i?K+1)f[j]和为奇数,就说明此时这个牛方向与最初相 反。由于 ∑ij=(i+1)?K+1f[j]=∑i?1j=(i?K+1)f[j]+f[i]-f[i-K+1]所以这个每一次都可以用常数算出来,时间复杂度 O(n^2)#includecstdio#includecstring#includeiostreamusing name space std;const int N=5000+10;int f[N],dir[N],n;int solve(int k){int cnt=0,sum=0;//sum 为 f 的和memset(f,0,sizeof(f));for(int i=1;i=n-k+1;i++){if((dir[i]+sum)%2){cnt++;f[i]=1;}sum+=f[i];if(i-k+1=1) sum-=f[i-k+1];}for(int i=n-k+2;i=n;i++){//检查剩下的牛有没有朝后面的情况 if((dir[i]+sum)%2) return n+1;if(i-k+1=1) sum-=f[i-k+1];}return cnt;}int main(){ while(~scanf(%d,n)){for(int i=1;i=n;i++){ char c;scanf( %c,c); if(c==B) dir[i]=1;}int ansk,ansm=n,t; for(int i=1;i=n;i++){t=solve(i);if(tansm){ansm=t;ansk=i;}}printf(%d %d\n,ansk,ansm);}}POJ 3279农夫约翰知道聪明的牛产奶多。于是为了提高牛的智商他准备了如下游戏。有一个 M×N 的格子,每个格子可以翻转正反面,它们一面是黑色,另一面是白色。黑色的格子翻转后就是 白色,白色的格子翻转过来则是黑色。游戏要做的就是把所有的格子都翻转成白色。不过因 为牛蹄很大,所以每次翻转一个格子时,与它上下左右相邻接的格子也会被翻转。因为翻格 子太麻烦了,所以牛都想通过尽可能少的次数把所有格子都翻成白色。现在给定了每个格子 的颜色,请求出用最小步数完成时每个格子翻转的次数。最小步数的解有多个时,输出字典序最小的一组。解不存在的话,则输出 IMPOSSIBLE。已知:1≤M,N≤15 sample input M= 4N= 4 每个格子的颜色如下:(0 表示白色,1 表示黑色)10 0 101 1 001 1 010 0 1 sample output 00 0 0 10 0 1 10 0 1 00 0 0思路:在上面的那道题,让最左端的奶牛反转的情况只有一种,于是直接判断的方法就可以 确定,但是这里不一样,比如,看最左上面的角,除了反转(1,1),(1,2),(2,1)都可以导致他翻装。于是不妨我们先确定最上面一行的反装方式,此时能反转(1,1)只有(2,1),所以如果已知第一行就可以知道第二行那些点需要反转。这样反复下去,只要最后一行全

文档评论(0)

skvdnd51 + 关注
实名认证
文档贡献者

该用户很懒,什么也没介绍

1亿VIP精品文档

相关文档