NOIP 2017.9.17 总结+心得

2024-03-21 13:40
文章标签 总结 17 心得 noip 2017.9

本文主要是介绍NOIP 2017.9.17 总结+心得,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这里写图片描述
世界真的很大
最近状态感觉真的很不好。。。考试连续爆炸。。。
倒不是因为想不出来,而是老是在一些细节操作上出问题
看来是时候调整一波状态了,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 总结+心得的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/832758

相关文章

Python中实现进度条的多种方法总结

《Python中实现进度条的多种方法总结》在Python编程中,进度条是一个非常有用的功能,它能让用户直观地了解任务的进度,提升用户体验,本文将介绍几种在Python中实现进度条的常用方法,并通过代码... 目录一、简单的打印方式二、使用tqdm库三、使用alive-progress库四、使用progres

Android数据库Room的实际使用过程总结

《Android数据库Room的实际使用过程总结》这篇文章主要给大家介绍了关于Android数据库Room的实际使用过程,详细介绍了如何创建实体类、数据访问对象(DAO)和数据库抽象类,需要的朋友可以... 目录前言一、Room的基本使用1.项目配置2.创建实体类(Entity)3.创建数据访问对象(DAO

Java向kettle8.0传递参数的方式总结

《Java向kettle8.0传递参数的方式总结》介绍了如何在Kettle中传递参数到转换和作业中,包括设置全局properties、使用TransMeta和JobMeta的parameterValu... 目录1.传递参数到转换中2.传递参数到作业中总结1.传递参数到转换中1.1. 通过设置Trans的

C# Task Cancellation使用总结

《C#TaskCancellation使用总结》本文主要介绍了在使用CancellationTokenSource取消任务时的行为,以及如何使用Task的ContinueWith方法来处理任务的延... 目录C# Task Cancellation总结1、调用cancellationTokenSource.

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

git使用的说明总结

Git使用说明 下载安装(下载地址) macOS: Git - Downloading macOS Windows: Git - Downloading Windows Linux/Unix: Git (git-scm.com) 创建新仓库 本地创建新仓库:创建新文件夹,进入文件夹目录,执行指令 git init ,用以创建新的git 克隆仓库 执行指令用以创建一个本地仓库的

二分最大匹配总结

HDU 2444  黑白染色 ,二分图判定 const int maxn = 208 ;vector<int> g[maxn] ;int n ;bool vis[maxn] ;int match[maxn] ;;int color[maxn] ;int setcolor(int u , int c){color[u] = c ;for(vector<int>::iter

整数Hash散列总结

方法:    step1  :线性探测  step2 散列   当 h(k)位置已经存储有元素的时候,依次探查(h(k)+i) mod S, i=1,2,3…,直到找到空的存储单元为止。其中,S为 数组长度。 HDU 1496   a*x1^2+b*x2^2+c*x3^2+d*x4^2=0 。 x在 [-100,100] 解的个数  const int MaxN = 3000

状态dp总结

zoj 3631  N 个数中选若干数和(只能选一次)<=M 的最大值 const int Max_N = 38 ;int a[1<<16] , b[1<<16] , x[Max_N] , e[Max_N] ;void GetNum(int g[] , int n , int s[] , int &m){ int i , j , t ;m = 0 ;for(i = 0 ;