NOIP 2017.9.17 总结+心得
来源:互联网 发布:requirejs 后盾网源码 编辑:程序博客网 时间:2024/06/05 03:59
世界真的很大
最近状态感觉真的很不好。。。考试连续爆炸。。。
倒不是因为想不出来,而是老是在一些细节操作上出问题
看来是时候调整一波状态了,NOIP说实话也的却不远了
看题先:
1.Mushroom的序列
【问题描述】
Mushroom手中有n个数排成一排,现在Mushroom想取一个连续的子序列,使得这个子序列满足:最多只改变一个数,使得这个连续的子序列是严格上升子序列,Mushroom想知道这个序列的最长长度是多少。
【输入格式】
第一行一个整数n,表示有n个数。
第二行为n个数。
【输出格式】
一个数,为最长长度。
【输入样例】
6
7 2 3 1 5 6
【输出样例】
5
【样例解释】
选择第2个数到第6个数,把1改变成4即可。
【数据范围】
对于30%的数据,n<=10
对于60%的数据,n<=1000
对于100%的数据,n<=100000
老实说这道题其实很简单。。
一开始把题目看错了,话说既然是连续的为什么要叫子序列?不是应该叫子串吧?这让我不得不开始质疑出题人的语文水平。。
一开始我以为就是一个nlogn的LIS再加1,和n取一个min
发现题读错了才开始想正解
也算是秒想吧,DP预处理一边从每个点开始往后的最长连续上升子串,每个点往前的最长连续上升子串
枚举分界点
特判只取一边然后加1的情况,和n取一个min
然后我就把我一开始读错题的代码交上去了。。。
完整代码:
#include<stdio.h>#include<algorithm>using namespace std;const int INF=0x3f3f3f3f;int f[100010],a[100010],g[100010],ans=0,n;int main(){ freopen("seq.in","r",stdin); freopen("seq.out","w",stdout); scanf("%d",&n); for(int i=1;i<=n;i++) scanf("%d",&a[i]); for(int i=1;i<=n;i++) { if(a[i]>a[i-1]) f[i]=f[i-1]+1; else f[i]=1; ans=max(ans,f[i]); } for(int i=n;i>=1;i--) { if(a[i]<a[i+1]) g[i]=g[i+1]+1; else g[i]=1; ans=max(ans,g[i]); } ans=min(ans+1,n); for(int i=1;i<=n;i++) if(a[i+1]-a[i-1]>=2) ans=max(ans,f[i-1]+g[i+1]+1); printf("%d\n",ans); return 0;}
2.Mushroom的区间
【题目描述】
Mushroom有一行数,初始时全部是0。现在Mushroom有m个区间[L,R],他希望用以下操作得到新的序列。
从m个给定区间中选择一个区间[s,t],把区间中的数对应元素全部翻转。(0变1,1变0)
请告诉Mushroom他能得到多少区间。(模10^9+7)
【输入格式】
第一行包含两个整数n,m。表示n个数和m个区间。
接下来m行是所表示的区间。
【输出格式】
一个整数,表示能得到的区间数。
【样例输入】
3 3
1 1
2 2
3 3
【样例输出】
8
【数据范围】
对于30%的数据,n,m<=20
对于60%的数据,n,m<=100
对于100%的数据,n,m<=100000
【样例解释】
每个位置都可以单个修改,所以有8种可能。
这道题其实一看就知道应该是那种代码量少,思维难度高的题
当时就是直接开始yy
考虑m个区间互不重叠的情况,答案直接就是2^m
然后假设只有两个区间,如果互不重叠,答案就是4,如果两个区间有重叠部分,在草稿纸上一画,好像也是4
然后我灵光一现,大胆推测,答案就是2^m,只不过重复的区间不算数,然后就unique区间去个重,2^cnt
报0
后来考完看题解,才知道我这个思路其实没有错,但是漏了一种情况
如果有两个区间首尾相接且不重叠,那么答案就是4了,如果现在再来一个大区间,正好把两区间最左和最右覆盖完,那么这个大区间其实是没有用的
也就是说每个区间算不算数,只需要看他左右两个端点有没有通过之前的区间连在一起
然后我们就可以用并查集来维护。
为了区分左和右把左边搞成L-1
完整代码:
#include<stdio.h>#include<algorithm>using namespace std;const int mod=1e9+7;struct seq{ int x,y;}a[100010];bool cmp(const seq &a,const seq &b){ if(a.x==b.x) return a.y<b.y; return a.x<b.x;}int n,m,ans=1,fa[200010];int getfather(int x){ if(fa[x]==x) return x; return fa[x]=getfather(fa[x]);}int main(){ freopen("seg.in","r",stdin); freopen("seg.out","w",stdout); scanf("%d%d",&n,&m); for(int i=0;i<=n;i++) fa[i]=i; for(int i=1;i<=m;i++) scanf("%d%d",&a[i].x,&a[i].y); sort(a+1,a+m+1,cmp); for(int i=1;i<=m;i++) { int x=getfather(a[i].x-1),y=getfather(a[i].y); if(x!=y) ans=(ans*2)%mod,fa[x]=y; } printf("%d\n",ans);}
3.来自风平浪静的明天
【题目描述】
冬眠了五年,光终于从梦中醒来。
千咲、要,大家都在。
隐约记得“昨天”的海船祭,爱花意外成为贡女,沉入海底。
海面冰封,却有丝丝暖流在冰面之下涌动。
此时,爱花沉睡在祭海女神的墓地。她的胞衣在一点点脱落,化作一簇簇暖流,夹杂着她的感情,向海面上涌去。
爱花,你在哪里?
五年之后,纺已经成为海洋学研究科的大学生。
在纺的帮助下,光得知了海面下海流的情况。
纺告诉光,暖流一旦产生,就会不断地向四周扩散,直到遇到海中的岩石。
红腹海牛,快告诉光,爱花在哪里。
纺帮你绘制了一张海流情况图,长度为N,宽度为M。
海很大,一边有沙滩,一边一望无际,但长度和宽度都不会超过300。沙滩是金黄色的,所以用Y表示。海是蓝色的,所以用B表示。暖流很暖和,所以用H表示
海中有大大小小的石头。石头很危险,所以用X表示
光相信自己一定能找到爱花(爱花的位置只有一种可能)
【输入格式】
第一行包括两个整数N,M。
接下来N行,每行M个字符。
【输出格式】
仅一行,表示爱花的位置(如果你无能为力,请输出 -1 ,只要你尽力,光不会责怪你)
【样例输入】
5 5
YYYHB
YYHHH
YHHXB
BBHBB
BBBBB
【样例输出】
2 3
【数据范围】
对于30%的数据,n,m<=10
对于70%的数据,n,m<=100
对于100%的数据,n,m<=300
【样例解释】
在(2,3)出现第一个H后,经过3s后,出现样例输入的地图。
P.S. Mushroom拜托他GF出的这题= =
题面巨长,巨难懂
最后一句看得我心痛
这道题暴力写得好的话能A
直接去枚举哪一个是原点,然后暴力模拟海流扩散的过程
考试的时候写挂了
特判实在是有点多
记录一个海流扩散的时间,看有没有海流扩散的范围超过时间的
时间就通过看一个点周围有没有B没有扩散出去
完整代码:
#include<stdio.h>#include<cstring>#include<queue>#include<map>using namespace std;const int INF=0x3f3f3f3f;queue <pair<int,int> > state ;int dis[310][310],vis[310][310],fx[4]={0,0,1,-1},fy[4]={1,-1,0,0};int n,m,anx,any,tim;char mp[310][310];bool bfs(int X,int Y){ while(!state.empty()) state.pop(); memset(vis,0,sizeof(vis)); memset(dis,0,sizeof(dis)); tim=INF,vis[X][Y]=1; state.push(make_pair(X,Y)); while(!state.empty()) { int x0=state.front().first,y0=state.front().second; int flag=1,flag1=0,flag2=4,flag3=0; state.pop(); for(int i=0;i<4;i++) { int x1=x0+fx[i],y1=y0+fy[i]; if(x1>n || x1<1 || y1>m || y1<1) continue ; if(!vis[x1][y1] && mp[x1][y1]=='H') { flag=0; dis[x1][y1]=dis[x0][y0]+1; if(dis[x1][y1]>tim) return false ; state.push(make_pair(x1,y1)),vis[x1][y1]=1; } if((mp[x1][y1]=='X' || mp[x1][y1]=='Y')) flag1++; if( mp[x1][y1]=='H') flag3++; } if(x0==1) flag2--; if(x0==n && n!=1) flag2--; if(y0==1) flag2--; if(y0==m && m!=1) flag2--; if(flag3+flag1!=flag2 &&tim==INF) tim=dis[x0][y0]; } return true ;}int main(){ freopen("calm.in","r",stdin); freopen("calm.out","w",stdout); scanf("%d%d",&n,&m); for(int i=1;i<=n;i++) scanf("%s",mp[i]+1); for(int i=1;i<=n;i++) for(int j=1;j<=m;j++) if(mp[i][j]=='H') if(bfs(i,j)) { printf("%d %d\n",i,j); return 0; } printf("-1\n"); return 0;}
4.心得+总结
这次考试呢,题虽然简单但是发挥的却不尽人意
感觉状态还不是很好
NOIP临近了,一边刷题一边复习一边调整状态吧
以后做简单题一定要仔细仔细再仔细
交题的时候要检查检查再检查
嗯,就是这样
- NOIP 2017.9.17 总结+心得
- NOIP模拟2017.9.19 总结+心得
- NOIP 2017.10.3 总结+心得
- NOIP 2017.10.4 总结+心得
- NOIP 2017.10.20 总结+心得
- NOIP 2017.10.23 总结+心得
- NOIP 2017.10.24 总结+心得
- NOIP 2017.10.27 总结+心得
- NOIP模拟赛2017.9.11 考试心得+总结
- 2017.9.17 noip模拟赛 总结
- 2017.9.17 胡策题 【题解 + 总结】【NOIP模拟】
- noip总结
- NOIP总结
- NOIP总结
- NOIP总结
- NOIp总结
- 论蒟蒻是怎样作死的(noip模拟赛心得&易错总结)
- 2017.9.20 noip模拟赛 总结
- LintCode算法题解——奇偶分割数组、二进制中1个数、反转整数、加一、排序数组转换为高度最小的二叉搜索树、二进制求和
- React学习
- leetcode 151. Reverse Words in a String 字符串分割逆序
- [Flink--计算引擎]--第三代计算引擎--Flink
- jsp 将表格以Excel形式导出方法之一(不用导包)
- NOIP 2017.9.17 总结+心得
- lucene.net查询条件
- linux 常用指令
- 吐槽CSDN--想钱想疯了--推荐文章里面广告博文去不掉
- 音视频倍速原理及实现
- 面试OR笔试42——划分数组
- 解决哈希(HASH)冲突的主要方法
- 392. Is Subsequence
- 图片分析