CSP-2020倒计时:36天

【SHOI2008】JZOJ2020年9月5日提高组 循环的债务

题目

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

Sample Output

5

Hint

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

题解

注明:以下的\(A\)表示\(Alice\),\(B\)表示\(Bob\),\(C\)表示\(Cynthia\)

发现钱的总量是不变的

思考\(DP\)

设\(f[i][j][k]\)表示考虑了前\(i\)种币值(从大到小),\(A\)有\(j\)元,\(B\)有\(k\)元

因为钱的总数不变,那么就可以算出\(C\)的钱

既然总数不变,那么就可以枚举第\(i\)种钱经过交换后\(A\)有多少,\(B\)有多少,算出\(C\)有多少,计算一下交换次数,转移即可

Code

#include<bits/stdc++.h>
using namespace std;
int x1,x2,x3,all,c1,c2,c3,z,tot,now,f[10][1005][1005],cost[7]={0,100,50,20,10,5,1},a[5][10];
int read()
{
int res=0,fh=1;char ch=getchar();
while ((ch<'0'||ch>'9')&&ch!='-') ch=getchar();
if (ch=='-') fh=-1,ch=getchar();
while (ch>='0'&&ch<='9') res=(res<<1)+(res<<3)+(ch-'0'),ch=getchar();
return res*fh;
}
int main()
{
x1=read();x2=read();x3=read();
for (int i=1;i<=3;++i)
for (int j=1;j<=6;++j)
{
a[i][j]=read();
all+=a[i][j]*cost[j];
}
for (int i=1;i<=6;++i)
c1+=a[1][i]*cost[i],c2+=a[2][i]*cost[i];
if (c1-x1+x3>all||c2-x2+x1>all||all-c1-c2-x3+x2>all)
{
printf("impossible\n");
return 0;
}
memset(f,0X3f3f3f,sizeof(f));
f[0][c1][c2]=0;
for (int i=1;i<=6;++i)
for (int j=0;j<=all;++j)
for (int k=0;j+k<=all;++k)
{
if (f[i-1][j][k]>1e9) continue;
tot=a[1][i]+a[2][i]+a[3][i];
for (int x=0;x<=tot;++x)
for (int y=0;y+x<=tot;++y)
{
z=tot-x-y;
now=(abs(a[1][i]-x)+abs(a[2][i]-y)+abs(a[3][i]-z))/2;
if (j-(a[1][i]-x)*cost[i]+k-(a[2][i]-y)*cost[i]>all) continue;
if (j-(a[1][i]-x)*cost[i]<0||k-(a[2][i]-y)*cost[i]<0) continue;
f[i][j-(a[1][i]-x)*cost[i]][k-(a[2][i]-y)*cost[i]]=min(f[i][j-(a[1][i]-x)*cost[i]][k-(a[2][i]-y)*cost[i]],f[i-1][j][k]+now);
}
}
if (f[6][c1-x1+x3][c2-x2+x1]>1e9)
{
printf("impossible\n");
return 0;
}
printf("%d\n",f[6][c1-x1+x3][c2-x2+x1]);
return 0;
}

【SHOI2008】JZOJ2020年9月5日提高组 循环的债务的更多相关文章

  1. JZOJ2020年8月11日提高组T4 景点中心

    JZOJ2020年8月11日提高组T4 景点中心 题目 Description 话说宁波市的中小学生在镇海中学参加计算机程序设计比赛,比赛之余,他们在镇海中学的各个景点参观.镇海中学共有n个景点,每个 ...

  2. JZOJ2020年8月11日提高组T3 页

    JZOJ2020年8月11日提高组T3 页 题目 Description 战神阿瑞斯听说2008年在中华大地上,将举行一届规模盛大的奥林匹克运动会,心中顿觉异常兴奋,他想让天马在广阔的天空上,举行一场 ...

  3. JZOJ2020年8月11日提高组T2 宝石

    JZOJ2020年8月11日提高组T2 宝石 题目 Description 见上帝动了恻隐之心,天后也想显示一下慈悲之怀,随即从口袋中取出一块魔术方巾,让身边的美神维纳斯拿到后堂的屏风上去试试,屏风是 ...

  4. JZOJ2020年8月11日提高组T1 密码

    JZOJ2020年8月11日提高组T1 密码 题目 Description 在浩浩茫茫的苍穹深处,住着上帝和他的神仆们,他们闲谈着下界的凡人俗事,对人世间表现的聪明智慧,大加赞赏.今天他们正在观赏大地 ...

  5. JZOJ2020年8月11日提高组反思

    JZOJ2020年8月11日提高组反思 T1 看到题 啊这?! 我看错了吗??? 我理解错题了吗?? 好吧没有-- 高精度模板题,不用多说 T2 看到这种矩阵的问题 以为是前缀和搞事情 结果呢 扫描线 ...

  6. JZOJ2020年9月5日提高组反思

    JZOJ2020年9月5日提高组反思 T1 考试的时候没有头绪,就打了个暴力,愉快的拿到了10分的\(impossible\) 正解是\(DP\),设\(f[i][j][k]\)表示地\(i\)种币值 ...

  7. 【GDOI2014模拟】JZOJ2020年8月14日提高组 服务器

    [GDOI2014模拟]JZOJ2020年8月14日提高组 服务器 题目 Time and Memory Limits Description 我们需要将一个文件复制到n个服务器上,这些服务器的编号为 ...

  8. JZOJ2020年8月14日提高组反思

    JZOJ2020年8月14日提高组反思 T1 看到题 一脸:我是谁,我在哪,我要干啥 看到字符串凉一半 还有查询修改 想到线段树但不会建模 暴力安排 T2 一开始觉得:水题 然后啪啪打脸 空间小,数据 ...

  9. 【GDKOI2014】JZOJ2020年8月13日提高组T4 内存分配

    [GDKOI2014]JZOJ2020年8月13日提高组T4 内存分配 题目 Description Input Output 输出m行,每行一个整数,代表输入中每次程序变化后系统所需要的空闲内存单位 ...

随机推荐

  1. [Luogu P2831] 愤怒的小鸟 (状压DP)

    题面: 传送门:https://www.luogu.org/problemnew/show/P2831 Solution 首先,我们可以先康一康题目的数据范围:n<=18,应该是状压或者是搜索. ...

  2. php 使用 phpword 操作 word 读取 word

    思路 1. 加载word文件.2. 循环判断加载出来的数据.( 数据下面有很多个节点 )( 节点是按照数据的类型分类的 例如 无样式的文本是RunText,换行是TextBreak,表格是table. ...

  3. c#中简单工厂模式

    运算类 public class yunsuan { public static operation create(string operate) { operation oper = null; s ...

  4. c#写入文件流

    using (FileStream wir=new FileStream(@"C:\Documents and Settings\Administrator\桌面\1.txt",F ...

  5. C++ 基础 2:C++ 对 C 语言的拓展

    1 引用 1.1 定义及编程实践 引用,是某个已存在变量的另一个名字. 一旦把引用初始化为某个变量,就可以使用该引用名称或变量名称来指向变量. 注意: 引用没有定义,是一种关系型声明.声明它和原有某一 ...

  6. JS小案例:循环间隔重复变色

    在A.B.C三个区块中,有且仅有一个红色,要求红色每隔一秒即进入下一个区块,变色过程不断循环往复. 参考代码: <!DOCTYPE html> <html lang="zh ...

  7. jm8.6编解码器概述

    自己在学习h264的路上,欢迎讨论交流. 前段时间研究JM出品的h264编码器,代码实在看不下去,因此换了个角度来研究诸多算法--逆向方式(解码),本系列文章记录一些遇到的东西和思考. 1. JM介绍 ...

  8. 响应式关系数据库处理R2DBC

    目录 简介 R2DBC介绍 项目依赖 创建ConnectionFactory 创建Entity Bean 初始化数据库 获取所有的用户 Prepare Statement 事务处理 WebFlux使用 ...

  9. win7-64位 jdk安装

    1.jdk安装 jdk安装主要是进行jdk以及jre安装,注意jre需要安装到一个空文件夹内即可. 官网地址:http://www.oracle.com/technetwork/java/javase ...

  10. hive简单的项目实战

    解压user.zip [root@hadoop1 test]# unzip user.zip -d /test/bigdatacase/dataset Archive: user.zip inflat ...