AGC007题解

发现自己思维能力又跟不上了...做题有点吃力...所以回归AGC刷题计划...

AGC040506都写了一部分题然后懒得补全了,所以从07开始做吧。大概是从C开始。

 

C

这也太人类智慧了吧...

我先是自己画了个柿子 咱也不知道对不对 先丢着

$\frac{1}{i-j+1} * \frac{1}{2^{j-n+1}} * ((2i-2j+1)d_1 + \frac{2i+2j-3}{2i-2j+1} x)$

就是根据每一个球到哪一个洞推的,发现是小数显然用不了 鸽了。

然后就发现了这人类智慧的题解

发现有个性质没用上 就是等差数列 我们接着转换问题

变成数轴上有2n个点 每次可以取走相邻两颗 贡献是它俩之间的距离

惊喜一刻到了:可以发现每次距离期望的变化是等差数列!

等差数列+等差数列依然是等差数列= =

所以我们直接维护首项末项和公差就可以计算了

好像...很人类智慧...

//Love and Freedom.
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#define ll long long
#define inf 20021225
#define db double
using namespace std;
int read()
{
    int s=0,f=1; char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-') f=-1; ch=getchar();}
    while(ch>='0'&&ch<='9') s=s*10+ch-'0',ch=getchar();
    return f*s;
}
int n,x,k;
int main()
{
    n=read(); x=read(); k=read();
    db bg=x,ed=x+(2.0*n-1)*k,ans=0.0;
    for(int i=(n<<1);i;i-=2)
    {
        ans+=(bg+ed)/2.0;
        bg+=(bg*2+k*5)/(1.0*i);
        ed+=(ed*2-k*5)/(1.0*i);
        if(i>2)    k=(ed-bg)/((i-2)*1.0-1.0);
    }
    printf("%.10lf\n",ans);
    return 0;
}
C

 

D

这个题...比C简单多了啊...AGC的题目顺序真是个迷...

写个柿子$f[i]=min(f[i],f[j]+max { T,2*p[i]-p[j+1] } )$

显然有单调性...分类讨论一波max的取值维护一下就行了...

//Love and Freedom.
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#define ll long long
#define inf 20021225
#define N 100100
using namespace std;
int read()
{
    int s=0,f=1; char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-') f=-1; ch=getchar();}
    while(ch>='0'&&ch<='9') s=s*10+ch-'0',ch=getchar();
    return f*s;
}
ll f[N],mn; int t,e,n,p[N];
int main()
{
    n=read(),e=read(),t=read();
    int l=0; mn=1e18;
    for(int i=1;i<=n;i++)    p[i]=read();
    for(int i=1;i<=n;i++)
    {
        while(l<=i && 2*(p[i]-p[l+1])>t)
            mn=min(mn,f[l]-2*p[l+1]), l++;
        if(l<i)    f[i]=f[l]+t;
        f[i]=min(f[i],mn+2*p[i]);
    }
    printf("%lld\n",f[n]+e);
    return 0;
}
D

 

E

神仙题.jpg

发现答案具有二分性 然后考虑怎么验证

我们用f[x][i][j]表示进x子树为i,出来是j是否可行。发现进去的方式至多只有leaf种,所以这玩意很多都没用,我们考虑把它们单拎出来。

继续考虑如果存在(i,j)(i',j')且满足(i<=i'&&j<=j')的时候,只需要保留(i,j)这样的话我们把一个f[x]的数量级降到了O(n)

考虑如何合并 我们需要对应的枚举(ls,i,j)(rs,k,l)使得(j+k+val[ls]+val[rs]<=mid)得到(x,i+val[ls],l+val[rs]) 由于上面的性质,当i单调增时,j单调降。所以我们直接双指针搞就可以了。

考虑复杂度问题 用Sx表示f[x]的个数 我们发现 Sx=2*min(Sls,Srs)(翻转对应两种)使用启发式合并,我们的复杂度就可以降到O(nlgnlgw)了。

//Love and Freedom.
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<vector>
#define ll long long
#define inf 20021225
#define pa pair<ll,ll>
#define mp make_pair
#define fs first
#define se second
#define N 131072
#define pb push_back
#define ls(x) son[x][0]
#define rs(x) son[x][1]
using namespace std;
int read()
{
    int s=0,f=1; char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-') f=-1; ch=getchar();}
    while(ch>='0'&&ch<='9') s=s*10+ch-'0',ch=getchar();
    return f*s;
}
vector<int> son[N+2]; int val[N+2];
vector<pa> f[N+2],tmp[2]; ll mid;
void dfs(int x)
{
    f[x].clear();
    if(!son[x].size()){f[x].pb(mp(0,0)); return;}
    dfs(ls(x)); dfs(rs(x));
    for(int p=0;p<2;p++)
    {
        tmp[p].clear(); int a=son[x][p],b=son[x][p^1]; int r=0;
        for(int i=0;i<f[a].size();i++)
        {
            while(r+1<f[b].size()&&f[b][r+1].fs+f[a][i].se+val[a]+val[b]<=mid)    r++;
            if(r<f[b].size() && f[b][r].fs+f[a][i].se+val[a]+val[b]<=mid)    tmp[p].pb(mp(f[a][i].fs+val[a],f[b][r].se+val[b]));
        }
    }
    int i=0,j=0; ll cur=1e18;
    while(i<tmp[0].size()&&j<tmp[1].size())
    {
        if(tmp[0][i]<tmp[1][j])
        {
            if(tmp[0][i].se<cur)
                f[x].pb(tmp[0][i]),cur=tmp[0][i].se;
            i++;
        }
        else
        {
            if(tmp[1][j].se<cur)
                f[x].pb(tmp[1][j]),cur=tmp[1][j].se;
            j++;
        }
    }
    while(i<tmp[0].size())
    {
        if(tmp[0][i].se<cur)
            f[x].pb(tmp[0][i]),cur=tmp[0][i].se;
        i++;
    }
    while(j<tmp[1].size())
    {    
        if(tmp[1][j].se<cur)
            f[x].pb(tmp[1][j]),cur=tmp[1][j].se;
        j++;
    }
}
bool check()
{
    dfs(1);
    if(f[1].size())    return 1;
    return 0;
}
int main()
{
    int n=read(); ll l=0,r=0;
    for(int i=2;i<=n;i++)
    {
        int fa=read(); val[i]=read();
        son[fa].pb(i); r+=val[i];
    }
    ll ans=r;
    while(l<=r)
    {
        mid=l+r>>1;
        if(check())    ans=mid,r=mid-1;
        else    l=mid+1;
    }
    printf("%lld\n",ans);
    return 0;
}
E
 

F

为什么他们什么都能想到啊QAQ

建议去看editorial...

大概就是维护拐点信息,感性理解还是可以的...大概自己做是没戏了...

//Love and Freedom.
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<queue>
#define ll long long
#define inf 20021225
#define N 1001000
using namespace std;
int read()
{
    int s=0,f=1; char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-') f=-1; ch=getchar();}
    while(ch>='0'&&ch<='9') s=s*10+ch-'0',ch=getchar();
    return f*s;
}
char s[N],t[N]; queue<int> q;
int main()
{
    int n=read(),ans=0;
    scanf("%s",s+1);
    scanf("%s",t+1);
    if(strcmp(s+1,t+1)==0)    return printf("0\n"),0;
    int pt=n,ps=n;
    while(pt)
    {
        while(pt && t[pt]==t[pt-1]) pt--;
        while(ps && (ps>pt||s[ps]!=t[pt]))    ps--;
        if(!ps)    return printf("-1\n"),0;
        while(!q.empty() && q.front()-q.size()>=pt)    q.pop();
        if(ps!=pt)    q.push(ps);
        ans=max(ans,(int)q.size()+1);
        pt--;
    }
    printf("%d\n",ans);
    return 0;
}
F

 

posted @ 2019-09-19 15:25  寒雨微凝  阅读(237)  评论(0编辑  收藏  举报