zoukankan      html  css  js  c++  java
  • 【JZOJ4816】【NOIP2016提高A组五校联考4】label

    题目描述

    这里写图片描述

    输入

    这里写图片描述

    输出

    这里写图片描述

    样例输入

    3
    2 2 0
    1 2
    3 3 2
    1 3
    1 2
    3 3 1
    1 2
    2 3

    样例输出

    4
    2
    12

    数据范围

    这里写图片描述

    样例解释

    这里写图片描述

    解法

    设f[i][j]为在第i个点填了j的合法方案。
    f[i][j]=(f[son(i)][l])(l[1,jk][j+k,m])
    时间复杂度为O(nm)
    观察到f值呈对称状,且中间一段的值是相同的。
    利用这个性质优化动态规划。

    代码

    #include<iostream>
    #include<stdio.h>
    #include<math.h>
    #include<string.h>
    #include<algorithm>
    #define ll long long
    #define ln(x,y) ll(log(x)/log(y))
    #define sqr(x) ((x)*(x))
    using namespace std;
    const char* fin="label.in";
    const char* fout="label.out";
    const ll inf=0x7fffffff;
    const ll maxn=107,maxm=11007,mo=1000000007;
    ll t,n,m,delta,i,j,k,tot,ans,tmd;
    ll fi[maxn],la[maxn*2],ne[maxn*2];
    ll f[maxn][maxm],g[maxn][maxm],h[maxn][maxm];
    void add_line(ll a,ll b){
        tot++;
        ne[tot]=fi[a];
        la[tot]=b;
        fi[a]=tot;
    }
    void dfs(ll v,ll from){
        ll i,j,k,tmp,pre=-1,tmb=0,o;
        for (i=1;i<maxm;i++) f[v][i]=1;
        for (k=fi[v];k;k=ne[k])
            if (la[k]!=from){
                dfs(la[k],v);
                for (i=1;i<=tmd;i++){
                    tmp=0;
                    if (delta==0){
                        tmp=(tmp+g[la[k]][i]+h[la[k]][i]+mo-f[la[k]][i])%mo;
                    }else{
                        j=i-delta;
                        if (j>0) tmp=(tmp+g[la[k]][j])%mo;
                        j=i+delta;
                        if (j<=m) tmp=(tmp+h[la[k]][j])%mo;
                    }
                    f[v][i]=(f[v][i]*tmp)%mo;
                }
            }
        for (i=2,j=tmd;i<=j;i++) if (f[v][i]==f[v][i-1]) {
            pre=i-2;
            break;
        }else tmb=(tmb+f[v][i-1])%mo;
        if (pre==-1) pre=tmd-1;//,tmb=(tmb+f[v][pre])%mo;
        for (i=1;i<=pre;i++) g[v][i]=(g[v][i-1]+f[v][i])%mo;
        for (j=min(m-pre+1,maxm);i<j;i++) g[v][i]=(g[v][i-1]+f[v][pre+1])%mo;
        for (j=min(m,maxm-1);i<=j;i++) g[v][i]=(g[v][i-1]+f[v][m-i+1])%mo;
        h[v][1]=(tmb*2+(m-2*pre+mo)%mo*f[v][pre+1])%mo;
        for (i=2;i<=pre+1;i++) h[v][i]=(h[v][i-1]-f[v][i-1]+mo)%mo;
        for (j=min(m-pre+1,maxm-1);i<=j && i<=m;i++) h[v][i]=(h[v][i-1]+mo-f[v][pre+1])%mo;
        for (j=min(m,maxm-1);i<=j;i++) h[v][i]=(h[v][i-1]+mo-f[v][m-i+2])%mo;
    }
    int main(){
        freopen(fin,"r",stdin);
        freopen(fout,"w",stdout);
        scanf("%d",&t);
        for (;t;t--){
            scanf("%d%d%d",&n,&m,&delta);
            tot=0;
            tmd=min((m+1)/2,maxm-delta-5);
            memset(fi,0,sizeof(fi));
            memset(h,0,sizeof(h));
            memset(g,0,sizeof(g));
            for (i=1;i<n;i++){
                scanf("%d%d",&j,&k);
                add_line(j,k);
                add_line(k,j);
            }
            dfs(1,0);
            ans=h[1][1];
            printf("%lld
    ",ans);
        }
        return 0;
    }

    启发

    观察转移方程的性质,譬如对称之类的,可以优化动态规划。

  • 相关阅读:
    近期总结和学习计划
    Processes and Threads
    button按钮触发点击事件后出现自动跳转问题
    Thinkphp 整合Ueditor 笔记
    laravel Excel数据导出
    Laravel 图片上传
    php排序笔记归并排序
    php排序笔记冒泡选择插入希尔堆排序
    XML lesson 1
    Java Web JSP语法:
  • 原文地址:https://www.cnblogs.com/hiweibolu/p/6714873.html
Copyright © 2011-2022 走看看