9.16&9.19
9.16:
A: 将其变为一个序列,然后对于A[i]-=i,然后求最长不下降子序列即可
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<cctype>
using namespace std;
int read()
{
int f=1,x=0;
char ch=' ';
for(;!isdigit(ch);ch=getchar())if(ch=='-')f*=-1;
for(;isdigit(ch);ch=getchar()) x=x*10+ch-'0';
return f*x;
}
int n,son[100010][2],r[100010],a[100010],fid[100010],cnt;
void dfs(int x)
{
if(son[x][0])
dfs(son[x][0]);
fid[++cnt]=x;
if(son[x][1])
dfs(son[x][1]);
}
int f[100010],g[100010];
int work()
{
int ans=0;
memset(g,127,sizeof(g));
g[0]=-g[0];
for(int i=1;i<=n;i++)
{
int x=upper_bound(g,g+n+1,a[i])-g;
ans=max(ans,x);
g[x]=a[i];
}
return ans;
}
int main()
{
n=read();
for(int i=2;i<=n;i++)
{
int f=read();
int s=read();
s--;
son[f][s]=i;
}
for(int i=1;i<=n;i++)
r[i]=read();
dfs(1);
for(int i=1;i<=n;i++)
a[i]=r[fid[i]];
for(int i=1;i<=n;i++)
a[i]-=i;
int ans=n-work();
printf("%d\n",ans);
return 0;
}
B:dp[i][j]表示选了i个数,有j个5,最多2的个数,转移到底就好了
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cctype>
using namespace std;
int f[5][510][6010],five[510],two[510];
long long a[510];
long long read()
{
long long f=1,x=0;
char ch=' ';
for(;!isdigit(ch);ch=getchar())if(ch=='-')f*=-1;
for(;isdigit(ch);ch=getchar()) x=x*10+ch-'0';
return f*x;
}
int main()
{
int n,ans=0,res=0,k,cnt=0;
n=read();
k=read();
for(int i=1;i<=n;i++)
{
a[i]=read();
while(a[i]%5==0)
{
five[i]++;cnt++;
a[i]/=5;
}
while(a[i]%2==0)
{
two[i]++;
a[i]/=2;
}
}
memset(f,-1,sizeof(f));
f[0][0][0]=0;
f[1][0][0]=0;
for(int i=1;i<=n;i++)
for(int j=1;j<=k;j++)
for(int u=0;u<=cnt;u++)
{
if(u>=five[i]&&f[(i-1)&1][j-1][u-five[i]]!=-1)
f[i&1][j][u]=max(f[(i-1)&1][j][u],f[(i-1)&1][j-1][u-five[i]]+two[i]);
else f[i&1][j][u]=f[(i-1)&1][j][u];
}
for(int u=1;u<=cnt;u++)
{
if(u>f[n&1][k][u])
ans=f[n&1][k][u];
else ans=u;
res=max(ans,res);
}
printf("%d",res);
return 0;
}
C:二分以后是求最大团,(然后我就闲的没事退火退了4页才过。。。)
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<cctype>
#include<ctime>
using namespace std;
const int INF=1e9;
int a[101][101];
int b[5000010];
int vis[5000010];
int ans;
int read()
{
int f=1,x=0;
char ch=' ';
for(;!isdigit(ch);ch=getchar())if(ch=='-')f*=-1;
for(;isdigit(ch);ch=getchar()) x=x*10+ch-'0';
return f*x;
}
int n,k;
int fi(int x)
{
for(int i=1;i<=n;i++) vis[i]=0;
int now=0;
for(int i=1;i<=n;i++)
{
int f=1;
for(int j=1;j<i;j++)
{
if(vis[b[j]]&&a[b[i]][b[j]]>x)
{
f=0;
break;
}
}
if(f) vis[b[i]]=1;
}
for(int i=1;i<=n;i++) now+=vis[i];
return now;
}
void ra(int row)
{
double t=100.0;
while(t>1e-8)
{
int x=rand()%n+1;
int y=rand()%n+1;
if(x==y) continue;
swap(b[x],b[y]);
int now=fi(row);
int fin=ans-now;
if(ans<now) ans=now;
else if(exp((double)((double)fin/t))*RAND_MAX<=rand())
swap(b[x],b[y]);
t*=0.9812;
}
}
bool check(int x)
{
ans=0;
for(int i=1;i<=100;i++)
{
random_shuffle(b+1,b+1+n);
ra(x);
if(ans>=k) return 1;
}
return 0;
}
int main()
{
srand((unsigned)time(NULL));
n=read(); k=read();
for(int i=1;i<=n;i++)
{
for(int j=1;j<=n;j++)
{
a[i][j]=read();
}
}
for(int i=1;i<=n;i++) b[i]=i;
int l=1,r=INF;
while(l<=r)
{
int mid=(l+r)/2;
if(check(mid))
{
ans=mid;
r=mid-1;
}
else
l=mid+1;
}
printf("%d\n",ans);
return 0;
}
总结:T1被吓到然后就没写,考完后发现60-70是很好拿的
T2用贪心乱搞竟然过了85
T3当时时间只够写个20 分了,考完发现其实进一步优化能拿到更多的分数
9.19
A:输出n遍n(然后考场上zkt说我对了但我这个如果因为自己的问题出了锅就不怪他了)然后我就完美的写错了文件名
#include<cstdio>
using namespace std;
int main()
{
int t;
scanf("%d",&t);
while(t--)
{
int n;
scanf("%d",&n);
int nn=n;
while(nn--)
printf("%d",n);
printf("\n");
}
return 0;
}
B:求出最长链的中心,把它作为根,然后以子树中最长链的长度为权值取点或删点即可
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cctype>
using namespace std;
int read()
{
int f=1,x=0;
char ch=' ';
for(;!isdigit(ch);ch=getchar())if(ch=='-')f*=-1;
for(;isdigit(ch);ch=getchar()) x=x*10+ch-'0';
return f*x;
}
int maxm(int x,int y)
{
if(x>y) return x;
else return y;
}
struct node
{
int v,last;
bool flag;
}e[5000010];
int in[5000010],cnt=0;
int dep[5000010],size[5000010],len[5000010];
int f[5000010];
int check2[5000010];
int anss;
struct Node
{
int x,y;
}fin[5000010];
bool leaf[5000010];
int check[5000010];
void add(int x,int y)
{
e[++cnt].v=y;
e[cnt].flag=1;
e[cnt].last=in[x];
in[x]=cnt;
}
bool cmp(Node a,Node b)
{
return a.x>b.x;
}
void dfs(int cur,int fa)
{
dep[cur]=dep[fa]+1;
int i,t,tmp=0;
size[cur]=1;
for(i=in[cur];i;i=e[i].last)
{
t=e[i].v;
if(t!=fa)
{
f[t]=cur;
dfs(t,cur);
tmp=max(tmp,len[t]);
size[cur]+=size[t];
}
}
if(size[cur]==1)len[cur]=0;
else len[cur]=tmp+1;
return;
}
void findfs(int cur,int fa)
{
int i,t,flag=1;
for(i=in[cur];i;i=e[i].last)
{
t=e[i].v;
if(t==fa) continue;
if(!check2[t]) continue;
flag=0;
findfs(t,cur);
}
if(flag) leaf[cur]=1;
return;
}
int cou=0;
int find(int x)
{
if(x!=f[x])
{
cou++;
x=find(f[x]);
}
return x;
}
int ans=-1;
int find2(int x)
{
if(x!=f[x])
{
cou++;
if(cou==(ans-1)/2+1)
return f[x];
else
x=find2(f[x]);
}
return 1;
}
int main()
{
//freopen("ex.in","r",stdin);
int n,k;
n=read();
k=read();
for(int i=1;i<=n-1;i++)
{
int x,y;
x=read();
y=read();
add(x,y);
add(y,x);
check[x]++;
check[y]++;
}
int cnt=0;
int flag=0;
for(int i=1;i<=n;i++)
{
if(check[i]==1)
{
cnt++;
}
else
{
if(check[i]!=2)
{
flag=1;
break;
}
}
}
if(!flag&&cnt==2)
{
if((n-k)%2)
printf("%d",(n-k)/2+1);
else
printf("%d",(n-k)/2);
return 0;
}
dfs(1,0);
f[1]=1;
for(int i=1;i<=n;i++)
{
if(leaf[i])
{
find(i);
//printf("%lld\n",cou);
if(cou>ans)
{
ans=cou;
anss=i;
}
cou=0;
}
}
cou=0;
anss=find2(anss);
dfs(anss,0);
for(int i=1;i<=n;i++)
fin[i].x=len[i],fin[i].y=i;
sort(fin+1,fin+1+n,cmp);
for(int i=1;i<=k;i++)
check2[fin[i].y]=1;
findfs(anss,0);
int ffans=-1;
for(int i=1;i<=k;i++)
{
if(leaf[fin[i].y])
ffans=maxm(ffans,fin[i].x);
}
printf("%d",ffans);
return 0;
}
C:手写bitset以实现查找第k小。其实std::bitset上2分也好不过zkt说std::bitset多一个log
这个因为需要一些毒瘤的ckw的读入优化,就不放代码了
总结:T1:在题目中已经说过了。。。
T2:当时不知道为什么,就把两个并查集写串了,然后对着大样例调了2h.....
T3:考场上就只在调T2了,T3都没开。。。
浙公网安备 33010602011771号