NOIP2015 提高组(senior) 解题报告

时间:2023-03-08 16:11:49

过了这么久才来发解题报告,蒟蒻实在惭愧 /w\

Day1 T1

【思路】

模拟

【代码】

 #include<iostream>
#include<cstring>
#include<queue>
#include<cmath>
#define FOR(a,b,c) for(int a=(b);a<=(c);a++)
using namespace std; const int maxn = ; int G[maxn][maxn];
int n,m; int main() {
cin>>n;
int r=,c=n/+;
G[r][c]=;
FOR(i,,n*n) {
if((r==)&&(c!=n)) {
r=n; c++;
G[r][c]=i;
}
else if(c==n&&r!=) {
c=; r--;
G[r][c]=i;
}
else if(r== && c==n) {
r++;
G[r][c]=i;
}
else if(r!= && c!=n) {
if(!G[r-][c+]) {
r--; c++;
G[r][c]=i;
}
else {
r++;
G[r][c]=i;
}
}
}
FOR(i,,n) {
FOR(j,,n) cout<<G[i][j]<<" ";
cout<<endl;
}
return ;
}

magic

Day1 T2

【思路】

Dfs,如果遇到已经访问的点则构成环更新ans

【代码】

 #include<cstdio>
#include<iostream>
#include<cstring>
#include<queue>
#include<algorithm>
#define FOR(a,b,c) for(int a=(b);a<=(c);a++)
using namespace std; const int maxn = +;
struct Edge{
int v,next;
}e[maxn*];
int en=-,front[maxn]; int d[maxn];
int n,ans=1e9;
int vis[maxn]; void AddEdge(int u,int v) {
en++; e[en].v=v; e[en].next=front[u]; front[u]=en;
} void dfs(int u,int fa){
for(int i=front[u];i>=;i=e[i].next) {
int v=e[i].v;
if(v!=fa)
{
if(vis[v]) {
ans=min(ans,abs(d[v]-d[u])+);
}
else {
vis[v]=;
d[v]=d[u]+;
dfs(v,u);
}
}
}
}
int read() {
char c=getchar();
while(!isdigit(c)) c=getchar();
int x=;
while(isdigit(c)) {
x=x*+c-'';
c=getchar();
}
return x;
} int main() {
memset(front,-,sizeof(front));
n=read();
int j;
FOR(i,,n) {
j=read();
AddEdge(i,j);
AddEdge(j,i);
}
FOR(i,,n) if(!vis[i]) {
vis[i]=;
dfs(i,-);
}
printf("%d",ans);
return ;
}

message

Day1 T3

【思路】

搜索+剪枝

忽略花色,统计每种码数出现次数方便出牌。

每次都先出顺子,对于手中剩下的牌我们贪心地将剩下的组合牌需要打的次数计算出来,然后更新ans以剪枝。

双王算作对牌。顺排不包括2和双王。

【代码】

 #include<cstdio>
#include<cstring>
#define FOR(a,b,c) for(int a=(b);a<=(c);a++)
using namespace std; const int N = ; int a[N],c[N];
int n,T,ans; int Qans() {
memset(c,,sizeof(c));
FOR(i,,) c[a[i]]++;
int tot=; //tot带牌
while(c[]&&c[]>) c[]--,c[]-=,tot++;
while(c[]&&c[]>) c[]--,c[]-=,tot++;
while(c[]&&c[]) c[]--,c[]--,tot++;
while(c[]&&c[]) c[]--,c[]--,tot++;
while(c[]&&c[]) c[]--,c[]--,tot++;
return tot+c[]+c[]+c[]+c[]; //带牌+三张 对子 单张
} void dfs(int now) {
if(now>=ans) return ;
int tmp=Qans();
if(now+tmp<ans) ans=now+tmp;
FOR(i,,) { //三顺子
int j=i;
while(a[j]>=) j++;
if(j-i>=) {
FOR(j2,i+,j-) {
FOR(k,i,j2) a[k]-=;
dfs(now+);
FOR(k,i,j2) a[k]+=;
}
}
}
FOR(i,,) { //双顺子
int j=i;
while(a[j]>=) j++;
if(j-i>=) {
FOR(j2,i+,j-) {
FOR(k,i,j2) a[k]-=;
dfs(now+);
FOR(k,i,j2) a[k]+=;
}
}
}
FOR(i,,) { //单顺子
int j=i;
while(a[j]>=) j++;
if(j-i>=) {
FOR(j2,i+,j-) {
FOR(k,i,j2) a[k]--;
dfs(now+);
FOR(k,i,j2) a[k]++;
}
}
}
} int main() {
//freopen("in.in","r",stdin);
//freopen("out.out","w",stdout);
scanf("%d%d",&T,&n);
while(T--) {
memset(a,,sizeof(a));
int x,y;
FOR(i,,n) {
scanf("%d%d",&x,&y);
if(x==) x=; else if(x) x--;
a[x]++;
}
ans=1e9;
dfs();
printf("%d\n",ans);
}
return ;
}

landlords

Day2 T1

【思路】

二分法

【代码】

 #include<cstdio>
#include<cstring>
#include<iostream>
#include<queue>
#include<cmath>
#define FOR(a,b,c) for(int a=(b);a<=(c);a++)
using namespace std; const int maxn = +; int d[maxn];
int n,m,len; bool can(int M) {
int last=,cur=;
FOR(i,,m) {
while(cur<=n && d[cur]-last<M) cur++;
if(d[cur]-last<M || (cur>n)) return false;
last=d[cur];
}
return true;
} int main() {
cin>>len>>n>>m;
m=n-m;
FOR(i,,n) cin>>d[i];
int L=,R=len;
while(L<R){
int M=L+(R-L+)/;
if(can(M)) L=M;
else R=M-;
}
cout<<L;
return ;
}

stone

Day2 T2

【思路】

DP+优化

设f[k][i][j]为已经有k段,A串匹配到i,B匹配到j的方案数,则有转移式:

f[k][i][j]=sigma{f[k-1][l][j-1]},A[i]==B[j]&&A[i-1]!=B[j-1]

= sigma{f[k-1][l][j-1]}+f[k][i-1][j-1],A[i]==B[j]&&A[i-1]==B[j-1]

前缀和优化时间,滚动数组优化空间。

【代码】

 #include<cstdio>
#include<cstring>
using namespace std; const int N = 1e3+;
const int M = +;
const int MOD = 1e9+; int f[][N][M],sum[][N][M],n,m,K;
char s1[N],s2[M]; int main() {
scanf("%d%d%d",&n,&m,&K);
scanf("%s",s1+),scanf("%s",s2+);
f[][][]=;
for(int i=;i<=n;i++) sum[][i][]=;
int x=;
for(int k=;k<=K;k++) {
x^=;
memset(sum[x],,sizeof(sum[x]));
memset(f[x],,sizeof(f[x]));
for(int i=;i<=n;i++)
for(int j=;j<=m;j++) {
if(s1[i]==s2[j]) {
f[x][i][j]=sum[x^][i-][j-];
if(s1[i-]==s2[j-]) f[x][i][j]=(f[x][i][j]+f[x][i-][j-])%MOD;
}
sum[x][i][j]=((sum[x][i][j]+sum[x][i-][j])%MOD+f[x][i][j])%MOD;
}
}
int ans=;
for(int i=;i<=n;i++)
ans=(ans+f[x][i][m])%MOD;
printf("%d",ans);
return ;
}

substring

Day2 T3

【思路】

       二分+LCA+差分

先求出所有查询的路长,时间为O(mlogn)。题目所求为修改后的最大查询路最小,考虑二分该最大路值ML。对于所有长度超过ML的路径求交,记录最大查询路为mx,只要我们求出这些路径的最大公共边(交)mxe,通过判断mx-mxe与ML就可调整区间。

如何求交? 差分。所谓差分就是将一个对区间的操作变为对区间端点的操作。将查分推广到树上。每个结点带个cnt,对于路径(u,v),cnt[u]++,cnt[v]++,cnt[lca(u,v)]-=2,在树上统计cnt[x]=sigma{cnt[son]},这样只要满足cnt[x]==1的边就在这条路上,满足cnt[x]==tot的边就在路径的交上。

总的时间为O(mlogn+(m+n)logL)

【代码】

 #include<cstdio>
#include<vector>
#include<cstring>
#include<iostream>
#include<algorithm>
#define FOR(a,b,c) for(int a=(b);a<=(c);a++)
using namespace std; const int N = *1e5+;
struct Edge{ int v,w; }; int n,m;
int u[N],v[N],w[N],lca[N],dist[N],val[N];
vector<Edge> g[N]; void read(int& x) {
char c=getchar(); int f=; x=;
while(!isdigit(c)) {if(c=='-')f=-; c=getchar();}
while(isdigit(c)) x=x*+c-'',c=getchar();
x*=f;
} int siz[N],fa[N],son[N],top[N],dep[N],dis[N];
void dfs1(int u) {
siz[u]=; son[u]=;
for(int i=;i<g[u].size();i++) {
int v=g[u][i].v;
if(v!=fa[u]) {
fa[v]=u; dep[v]=dep[u]+;
dis[v]=dis[u]+g[u][i].w;
dfs1(v);
siz[u]+=siz[v];
if(siz[v]>siz[son[u]]) son[u]=v;
}
}
}
void dfs2(int u,int tp) {
top[u]=tp;
if(son[u]) dfs2(son[u],tp);
for(int i=;i<g[u].size();i++) {
int v=g[u][i].v;
if(v!=fa[u]&&v!=son[u]) dfs2(v,v);
}
}
int LCA(int u,int v) {
while(top[u]!=top[v]) {
if(dep[top[u]]<dep[top[v]]) swap(u,v);
u=fa[top[u]];
}
return dep[u]<dep[v]? u:v;
} int tot,mx,mxe,cnt[N];
int find_mxe(int u) {
for(int i=;i<g[u].size();i++) {
int v=g[u][i].v;
if(v!=fa[u]) cnt[u]+=find_mxe(v);
}
if(cnt[u]==tot) mxe=max(mxe,val[u]);
int tmp=cnt[u]; cnt[u]=;
return tmp;
}
bool can(int ML) {
tot=; mx=mxe=;
FOR(i,,m) if(dist[i]>ML){
tot++; mx=max(mx,dist[i]);
cnt[u[i]]++,cnt[v[i]]++,cnt[lca[i]]-=;
}
find_mxe();
return mx-mxe<=ML;
} int main() {
freopen("transport.in","r",stdin);
freopen("transport.out","w",stdout);
read(n),read(m);
FOR(i,,n-) {
read(u[i]),read(v[i]),read(w[i]);
g[u[i]].push_back((Edge){v[i],w[i]});
g[v[i]].push_back((Edge){u[i],w[i]});
}
dfs1(),dfs2(,);
FOR(i,,n-) {
if(dep[u[i]]<dep[v[i]]) swap(u[i],v[i]);
val[u[i]]=w[i];
}
int x,y,L=,R=,M;
FOR(i,,m) {
read(x),read(y);
dist[i]=dis[x]+dis[y]-*dis[lca[i]=LCA(x,y)];
R=max(R,dist[i]); u[i]=x,v[i]=y;
}
R++;
while(L<R) {
M=(L+R)>>;
if(can(M)) R=M; else L=M+;
}
printf("%d",L);
return ;
}

transport

NOIP2015 所涉及到的知识有:模拟,二分法,搜索及优化,DP及优化,LCA

PS:Day1 T2 和 Day2 T3在部分OJ会栈溢出

题目/数据传送门

相关文章