Description

Alice、Bob和Cynthia总是为他们之间混乱的债务而烦恼,终于有一天,他们决定坐下来一起解决这个问题。不过,鉴别钞票的真伪是一件很麻烦的事情,于是他们决定要在清还债务的时候尽可能少的交换现金。比如说,Alice欠Bob 10元,而Cynthia和他俩互不相欠。现在假设Alice只有一张50元,Bob有3张10元和10张1元,Cynthia有3张20元。一种比较直接的做法是:Alice将50元交给Bob,而Bob将他身上的钱找给Alice,这样一共就会有14张钞票被交换。但这不是最好的做法,最好的做法是:Alice把50块给Cynthia,Cynthia再把两张20给Alice,另一张20给Bob,而Bob把一张10块给C,此时只有5张钞票被交换过。没过多久他们就发现这是一个很棘手的问题,于是他们找到了精通数学的你为他们解决这个难题。
Input

输入的第一行包括三个整数:x1、x2、x3(-1,000≤x1,x2,x3≤1,000),其中 x1代表Alice欠Bob的钱(如果x1是负数,说明Bob欠了Alice的钱) x2代表Bob欠Cynthia的钱(如果x2是负数,说明Cynthia欠了Bob的钱) x3代表Cynthia欠Alice的钱(如果x3是负数,说明Alice欠了Cynthia的钱)接下来有三行,每行包括6个自然数: a100,a50,a20,a10,a5,a1 b100,b50,b20,b10,b5,b1 c100,c50,c20,c10,c5,c1 a100表示Alice拥有的100元钞票张数,b50表示Bob拥有的50元钞票张数,以此类推。另外,我们保证有a10+a5+a1≤30,b10+b5+b1≤30,c10+c5+c1≤30,而且三人总共拥有的钞票面值总额不会超过1,000。
Output

如果债务可以还清,则输出需要交换钞票的最少张数;如果不能还清,则输出“impossible”(注意单词全部小写,输出到文件时不要加引号)。
Sample Input
输入一
10 0 0
0 1 0 0 0 0
0 0 0 3 0 10
0 0 3 0 0 0

输入二
-10 -10 -10
0 0 0 0 0 0
0 0 0 0 0 0
0 0 0 0 0 0
Sample Output
输出一
5

输出二
0
HINT

对于100%的数据,x1、x2、x3 ≤ |1,000|。

囧,为了让pascal调试的时候不怎么卡,把循环队列调小了,一开始打的是没剪枝的,直接冲突了,查了半天没查出来

就是一个动态规划,dp[i,j,k]表示用到第i种钞票,A有j元钱,B有k元钱的最小交换次数

我们从小钞票开始,因为小钞票可以合成大钞票,大钞票不能当成小钞票

然后就可以加一个剪枝,如果现在的钱与最终的钱的差值不可能用i以后的钞票得到(用最小公倍数判断),那么这个状态就没有用了,所以就不要加入队列

 const
gold:array[..]of longint=(,,,,,);
h:array[..]of longint=(,,,,,);
maxq=;
var
f:array[..,..,..]of longint;
g:array[..]of longint;
s:array[..,..]of longint;
q:array[..maxq,..]of longint;
head,tail,sum:longint; procedure init;
var
x,y,z,i,j:longint;
begin
read(x,y,z);
g[]:=z-x;
g[]:=x-y;
g[]:=y-z;
fillchar(f,sizeof(f),);
for i:= to do
for j:= downto do
begin
read(s[i,j]);
inc(g[i],s[i,j]*gold[j]);
inc(sum,s[i,j]*gold[j]);
end;
head:=;
tail:=;
q[,]:=;
q[,]:=g[]-(z-x);
q[,]:=g[]-(x-y);
f[,g[]-(z-x),g[]-(x-y)]:=;
end; function min(x,y:longint):longint;
begin
if x<y then exit(x);
exit(y);
end; procedure work;
var
i,j,k:longint;
a:array[..]of longint;
begin
while head<>tail do
begin
if q[head,]= then break;
a[]:=q[head,];
a[]:=q[head,];
a[]:=sum-a[]-a[];
for i:= to do
begin
for j:= to s[i mod +,q[head,]+] do
for k:= to s[(i mod +)mod +,q[head,]+] do
begin
inc(a[i],gold[q[head,]+]*(j+k));
dec(a[i mod +],j*gold[q[head,]+]);
dec(a[(i mod +)mod +],k*gold[q[head,]+]);
if f[q[head,]+,a[],a[]]> then
if (abs(a[]-g[])mod h[q[head,]+]=) and (abs(a[]-g[])mod h[q[head,]+]=) then
begin
q[tail,]:=q[head,]+;
q[tail,]:=a[];
q[tail,]:=a[];
tail:=tail mod maxq+;
end;
f[q[head,]+,a[],a[]]:=min(f[q[head,]+,a[],a[]],f[q[head,],q[head,],q[head,]]+j+k);
dec(a[i],gold[q[head,]+]*(j+k));
inc(a[i mod +],j*gold[q[head,]+]);
inc(a[(i mod +)mod +],k*gold[q[head,]+]);
end;
for j:= to s[i,q[head,]+] do
for k:= to j do
begin
dec(a[i],gold[q[head,]+]*j);
inc(a[i mod +],(j-k)*gold[q[head,]+]);
inc(a[(i mod +)mod +],k*gold[q[head,]+]);
if f[q[head,]+,a[],a[]]> then
if (abs(a[]-g[])mod h[q[head,]+]=) and (abs(a[]-g[])mod h[q[head,]+]=) then
begin
q[tail,]:=q[head,]+;
q[tail,]:=a[];
q[tail,]:=a[];
tail:=tail mod maxq+;
end;
f[q[head,]+,a[],a[]]:=min(f[q[head,]+,a[],a[]],f[q[head,],q[head,],q[head,]]+j);
inc(a[i],gold[q[head,]+]*j);
dec(a[i mod +],(j-k)*gold[q[head,]+]);
dec(a[(i mod +)mod +],k*gold[q[head,]+]);
end;
end;
head:=head mod maxq+;
end;
if f[,g[],g[]]< then write(f[,g[],g[]])
else write('impossible');
end; begin
init;
work;
end.

1021: [SHOI2008]Debt 循环的债务 - BZOJ的更多相关文章

  1. BZOJ 1021 [SHOI2008]Debt 循环的债务

    1021: [SHOI2008]Debt 循环的债务 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 694  Solved: 356[Submit][S ...

  2. BZOJ 1021: [SHOI2008]Debt 循环的债务( dp )

    dp(i, j, k)表示考虑了前i种钱币(从小到大), Alice的钱数为j, Bob的钱数为k, 最小次数. 脑补一下可以发现, 只有A->B.C, B->A.C, C->A.B ...

  3. bzoj千题计划111:bzoj1021: [SHOI2008]Debt 循环的债务

    http://www.lydsy.com/JudgeOnline/problem.php?id=1021 如果A收到了B的1张10元,那么A绝对不会把这张10元再给C 因为这样不如B直接给C优 由此可 ...

  4. bzoj1021 [SHOI2008]Debt 循环的债务

    前天打了一场比赛,让我知道自己Dp有多弱了,伤心了一天,没刷bzoj. 昨天想了一天,虽然知道几何怎么搞,但我还是不敢写,让我知道自己几何有多弱了,伤心了一天,没刷bzoj 1021: [SHOI20 ...

  5. 【BZOJ 1021】[SHOI2008]Debt 循环的债务

    [题目链接]:http://www.lydsy.com/JudgeOnline/problem.php?id=1021 [题意] [题解] 设f[i][j][k]表示前i种面值的钱币; 第一个人当前的 ...

  6. [bzoj1021][SHOI2008]Debt 循环的债务 (动态规划)

    Description Alice. Bob和Cynthia总是为他们之间混乱的债务而烦恼,终于有一天,他们决定坐下来一起解决这个问题.不过,鉴别钞票的真伪是一件很麻烦的事情,于是他 们决定要在清还债 ...

  7. BZOJ_1021_[SHOI2008]_Debt循环的债务_(DP)

    描述 http://www.lydsy.com/JudgeOnline/problem.php?id=1021 三个人相互欠钱,给出他们每个人各种面额的钞票各有多少张,求最少需要传递多少张钞票才能把账 ...

  8. [SHOI 2008]Debt 循环的债务

    Description 题库链接 A 欠 B \(x_1\) 元, B 欠 C \(x_2\) 元, C 欠 A \(x_3\) 元.现每人手上各有若干张 100,50,20,10,5,1 钞票.问至 ...

  9. $bzoj1021-SHOI2008\ Debt$ 循环的债务 $dp$

    题面描述 \(Alice\).\(Bob\)和\(Cynthia\)总是为他们之间混乱的债务而烦恼,终于有一天,他们决定坐下来一起解决这个问题.不过,鉴别钞票的真伪是一件很麻烦的事情,于是他们决定要在 ...

随机推荐

  1. Ubuntu系统中登陆阿里云服务器的方法

    如果您购买了阿里云服务器,恰巧又在使用Ubuntu操作系统,那么恭喜你来对地方了,今天给大家分享一下如何在Ubuntu中登陆阿里云服务器: 主要使用两款软件:1.SecureCRT:2.SecureF ...

  2. 第四十五篇、UITableViewCell高度计算

    由于tableView:heightForRowAtIndexPath:方法的调用频率非常高,如果将cell高度的计算过程放在此方法中,那么效率将会非常的低,快速tableview就会出现卡顿 1.通 ...

  3. [译]JavaScript检测浏览器前缀

    原文地址: Detect Vendor Prefix with JavaScript 不管浏览器私有前缀的现状如何,我们还是要与之为伴,并且有时候还需要利用它来做一些事情.这些前缀可以用于CSS(比如 ...

  4. CSS笔记---文字两边对齐

    <style> .box{ width: 1000px; height: 500px; background-color: #aa0000; margin:0 auto; } .teste ...

  5. 关于ajax解析

    出处:http://www.cnblogs.com/huashanlin/archive/2006/10/09/524707.html 要很好地领会Ajax技术的关键是了解超文本传输协议(HTTP), ...

  6. Ubuntu将软件(Sublime Text 2为例)锁定到启动器

    Ubuntu中打开某安装好的软件,然后右击启动器(Launcher)上打开的图标就可以将该软件锁定到启动器或者从启动器解锁. 然而,有许多软件下载后直接解压就能用,不需要安装,这种情况采用上述方法锁定 ...

  7. a2x

    #include <typeinfo> template <typename T> bool a2x( T& _f , char* p) { if( !p ) retu ...

  8. NTT研发

    2.研发标准 stead---大型机 css terasolunt---反向自动生成设计书 3.开发阶段做好设计,确定需求,测试阶段只是做产品有多差或者完成了多少需求,不是用来提高产品质量的过程

  9. 多进程之间的互斥信号量实现(Linux和windows跨平台)

    多线程之间实现互斥操作方式很多种,临界区(Critical Section),互斥量(Mutex),信号量(Semaphore),事件(Event)等方式 其中临界区,互斥量,信号量算是严格意义的实现 ...

  10. GDI+绘制文本

    这是在论坛中有人提出的一个问题,原贴见:Graphics DrawString参数无效.这里给出方法,读者可以自行修改以适应自己的项目需求. 先上代码: if (!Page.IsPostBack) { ...