NOI2006 网络收费 network

    技术2025-03-31  12

     

    NOI2006 网络收费

    题目大意:

    配对收费的方式

    合理的改变每一个叶子节点的颜色(付费方式)

    使改变档案费用+配对收费总和最小

     

    40%的数据中N≤4

    80%的数据中N≤7

    100%的数据中N≤10

     

     

    节点个数=2^n

    要点一:看懂系数表 做的时候就卡在这里了,囧!!

      配对收费的规则

      B较多时,  AA  收费系数为2

                      AB  收费系数为1

                      BB  收费系数为0

      其他情况反之

      设想:

      B较多时,在每一个A结点上有1个收费系数

      否则在每一个B结点上有1个收费系数

     

    要点二:由于决策对之前的费用有影响,先规定一些决策(fatherS)

      状态:opt[i,cntA,fatherS]

      表示下述子问题的最优解

      以i结点为根的子树

      管辖的叶子中,A的数目为cntA(可算B的数目)

      祖先的状态存储在fatherS

      即每一个祖先管辖叶子中,A还是B

     

    状态转移:opt[i,cntA,fatherS]

      枚举i的左孩子中A的数目cntA1

      在所有可行解中opt[left(i), cntA1, fatherS + nowS] +opt[right(i), cntA – cntA1, fatherS + nowS]取最小值

     

    边界条件: i是叶子

      根据cntA计算改变颜色需要的费用,

      根据cntAfatherS计算加在这个叶子节点上的配对费用

     

    初看时间复杂度:80, M = 2N

      状态O(M3)

      时间复杂度为O(M4)

    要点三:

      仔细分析状态opt[i,cntA,fatherS]

      假设Ak个祖先

      则cntA2N-k种可能,fatherS2k种可能

      状态有O(M2)

      而转移平摊复杂度为O(log2M)=O(N)

      证明:总的转移次数为∑K=0 to N  2^(N-K){枚举的cntA1的数目}*2^K {这一层的节点数}*M{cntAfatherS连起来的状态            数}=N*M^2

    所以时间复杂度为O(M2N):    100

     

    存储状态:对于每一层的节点滚动实现,把2^(N-K)2^K接起来同放在一维状态中,充分利用状态。

     

    这个实现应该是比较漂亮的了。

    program xqz; const maxn=1024; oo=1 shl 30; var f:array[0..1,0..maxn,0..maxn*2] of longint; w:array[0..maxn,0..maxn] of longint; c,a:array[0..maxn] of longint; i,j,k,m,n,l,r,dep,ans,s,t:longint; function max(a,b:longint):longint; begin if a>b then max:=a else max:=b; end; function min(a,b:longint):longint; begin if a<b then min:=a else min:=b; end; begin assign(input,'network.in'); reset(input); assign(output,'network.out'); rewrite(output); read(n); inc(n); m:=1 shl (n-1); for i:=1 to m do read(a[i]); for i:=1 to m do read(c[i]); for i:=1 to m do begin for j:=i+1 to m do begin read(w[i,j]); w[j,i]:=w[i,j]; end; for j:=1 to m do w[i,j]:=w[i,j-1]+w[i,j]; end; t:=0; s:=1; for i:=1 to m do for j:=0 to m*2 do begin if not((a[i]=1)xor odd(j)) then f[t,i,j]:=c[i]; l:=i; r:=j shr 1; for k:=0 to n-2 do begin if odd(j) xor odd(r) then if odd(l) then inc(f[t,i,j],w[i,(l+1)*(1 shl k)]-w[i,l*(1 shl k)]) else inc(f[t,i,j],w[i,(l-1)*(1 shl k)]-w[i,(l-2)*(1 shl k)]); l:=(l+1) shr 1; r:=r shr 1; end; end; for dep:=n-1 downto 1 do begin s:=(s+1) and 1; t:=(t+1) and 1; for i:=1 to 1 shl (dep-1) do for j:=0 to m*2 do begin if (dep=2)and(i=1)and(j=5) then k:=k; f[t,i,j]:=oo; l:=j and(1 shl (n-dep+1)-1); if l>1 shl (n-dep) then continue; r:=((j shr (n-dep+1)) shl 1+ord(l>=1 shl (n-dep-1))) shl (n-dep); for k:=max(l-1 shl (n-dep-1),0) to min(l,1 shl (n-dep-1)) do f[t,i,j]:=min(f[t,i,j],f[s,i shl 1-1,r+k]+f[s,i shl 1,r+l-k]); end; end; ans:=oo; for i:=0 to m do ans:=min(ans,f[t,1,i]); writeln(ans); close(input); close(output); end. 

     

    最新回复(0)