首先直接做多重背包肯定会TLE的,

观察这个背包问题有什么特殊性呢

物品种类和重量,价值是一定的,不同的是背包的容量和物品的数量

由于当物品数量没有限制的时候,方案数是可以预处理出来的

所以我们考虑用ans=物品数量没有限制时的方案数-物品超出限制的方案数来解决

第一部分是可以用完全背包来解决的

第二问不难想到用容斥原理来解决

设f[m]为容量为m时的方案数

假如当i号物品超出数量s[i]限制(不知道其他物品有没有超),方案数f[m-w[i]*(s[i]+1)](严格超出限制

以此类推,可得出多个物品超出限制的方案数

根据容斥原理,计算一下即可

 var f:array[..] of int64;
    w:array[..] of longint;
    c:array[..,..] of longint;
    t:array[..] of longint;
    i,n,j,k,p,y,m,s:longint;
    ans:int64; begin
  for i:= to do
    read(w[i]);
  readln(n);
  for i:= to n do
  begin
    for j:= to do
      read(c[i,j]);
    readln(t[i]);
    if t[i]>m then m:=t[i];
  end;
  f[]:=;
  for i:= to do
    for j:=w[i] to do
      f[j]:=f[j]+f[j-w[i]];   for i:= to n do
  begin
    ans:=;
    for j:= to do   //用二进制表示物品是否超出限制
    begin
      s:=t[i];
      y:=;
      for k:= to do
      begin
        p:= shl (k-);
        if j and p<> then
        begin
          s:=s-w[k]*(c[i,k]+);
          if s< then break;
          inc(y);
        end;
      end;
      if s< then continue;   //注意可能不存在某几个物品都超出限制的情况
      if y mod = then ans:=ans-f[s] else ans:=ans+f[s];
    end;
    writeln(ans);
  end;
end.
04-29 00:02