P3092 [USACO13NOV] No Change G 题解
题解
思路
看到 \(1\le k\le16\),我们想到状压DP。
以每枚硬币是否被使用为状态,对其进行枚举。
令 \(dp_i\) 表示状态 \(i\) 下最多能支付到第 \(dp_i\) 件商品,令 \(f_{i,j}\) 表示从第 \(i+1\) 个位置开始,第 \(j\) 枚硬币可以支付 \((i+1,f_{i,j}]\) 这一区间的商品。
于是可以知道:每个状态下最多能支付到的商品位置,是从它的前继状态最多能支付到的那件商品的下一件商品开始,使用新增加的那枚硬币,所能支付到的最后一件商品的位置。
结合转移方程理解这句话:
\]
写成代码形式:
if(i&(1<<j)==0) dp[i]=max(dp[i],f[dp[i^(1<<j)][j];
像这样。答案就是所有 \(dp_i\ge N\) 的状态中剩余钱数的最小值。
时间复杂度 \(O(NK\log N+2^KK)\),可以解决。
实现
首先需要预处理 \(f\) 数组。笔者使用二分,更优的做法是双指针,留给读者自行思考(绝对不是因为我不会)。
在预处理过程中需要频繁查询区间和,因此可以预处理 \(c\) 数组的前缀和。
代码
#include <cstdio>
#define N 100005
int max(int x,int y) {return x>y?x:y;}
int k,n;
int a[20],c[N];
int f[N][20]; //(i,f[i][k]]
int dp[N];
int cnt(int x) //没用的硬币求和
{
int ret=0;
for(int i=0;i<k;i++)
if(!(x&(1<<i))) ret+=a[i];
return ret;
}
int main()
{
scanf("%d%d",&k,&n);
for(int i=0;i<k;i++) scanf("%d",&a[i]);
for(int i=1;i<=n;i++)
scanf("%d",&c[i]),c[i]+=c[i-1];
for(int i=0;i<=n;i++)
for(int j=0;j<k;j++)
{
int lb=i,rb=n;
while(lb<rb)
{
int mid=lb+rb+1>>1;
if(c[mid]-c[i]<=a[j]) lb=mid;
else rb=mid-1;
}
f[i][j]=lb;
}
int maxc=-1;
for(int i=0;i<(1<<k);i++)
{
for(int j=0;j<k;j++)
if(i&(1<<j))
dp[i]=max(dp[i],f[dp[i^(1<<j)]][j]);
if(dp[i]==n)
maxc=max(maxc,cnt(i));
}
printf("%d",maxc);
}
\]
P3092 [USACO13NOV] No Change G 题解的更多相关文章
- 洛谷P3092 [USACO13NOV]没有找零No Change
P3092 [USACO13NOV]没有找零No Change 题目描述 Farmer John is at the market to purchase supplies for his farm. ...
- 状压 DP:[USACO06NOV] Corn Fields,[USACO13NOV] No Change
[USACO06NOV] Corn Fields (试题来源:Link ) 题目描述 Farmer John has purchased a lush new rectangular pasture ...
- 洛谷P3104 Counting Friends G 题解
题目 [USACO14MAR]Counting Friends G 题解 这道题我们可以将 \((n+1)\) 个边依次去掉,然后分别判断去掉后是否能满足.注意到一点, \(n\) 个奶牛的朋友之和必 ...
- 洛谷P2115 Sabotage G 题解
题目 [USACO14MAR]Sabotage G 题解 本蒟蒻又来了,这道题可以用二分答案来解决.我们可以设答案最小平均产奶量为 \(x \ (x \in[1,10000])\) .然后二分搜索 \ ...
- P3092 [USACO13NOV]没有找零No Change
题目描述 Farmer John is at the market to purchase supplies for his farm. He has in his pocket K coins (1 ...
- 洛谷 P3092 [USACO13NOV]没有找零No Change
题目描述 Farmer John is at the market to purchase supplies for his farm. He has in his pocket K coins (1 ...
- luogu P3092 [USACO13NOV]没有找零No Change
题目描述 Farmer John is at the market to purchase supplies for his farm. He has in his pocket K coins (1 ...
- P3092 [USACO13NOV]没有找零No Change 状压dp
这个题有点意思,其实不是特别难,但是不太好想...中间用二分找最大的可买长度就行了. 题干: 题目描述 Farmer John <= K <= ), each with value .., ...
- Luogu P3092 [USACO13NOV]没有找零No Change【状压/二分】By cellur925
题目传送门 可能是我退役/NOIP前做的最后一道状压... 题目大意:给你\(k\)个硬币,FJ想按顺序买\(n\)个物品,但是不能找零,问你最后最多剩下多少钱. 注意到\(k<=16\),提示 ...
- P2882 Face The Right Way G 题解
题目 Farmer John has arranged his N \((1 ≤ N ≤ 5,000)\) cows in a row and many of them are facing forw ...
随机推荐
- pycharm之debugger使用
1.未打断点运程序,输出全部结果 2.打断点后,点击debug,代码执行到断点前停止(断点所在行不执行) 3.step over,是在单步执行时,在函数内遇到子函数时不会进入子函数内单步执行,而是将子 ...
- NOIP2024加赛8
NOIP2024加赛8 T1 flandre 第 4 个样例没给全,说明这可以直接猜结论 首先我们假设选定了 $ x $ 个数,那么我们肯定是把他们从小到大排好序依次放,这样才能使整体效果最大.然后我 ...
- redis 使用lua 生成流水号
在实际的业务场景中,我们会用到流水号. 之前的流水号做法是,使用redis的全局锁.然后对数据库进行更新,数据库更新 这个也会有一些问题,比如对于同一个流水号,多个线程去更新,由于事务比较长,那么就会 ...
- uniapp 画布
1.前言 uniapp中的canvas与HTML中的canvas用法并不同,他的使用文档请参考微信小程序画布 2.基本使用 1.准备canvas容器,并为其设置canvas-id和宽高(为了兼容H5, ...
- 下列哪个选项是对MTU的正确计算方式?
A. IP数据包头部 + TCP数据报头部 + 数据 B. MAC头 + IP头 + TCP头 + 数据 C. MAC头 + IP头 + TCP头 + 数据 + FCS D. 前同步码 ...
- 企业IT基础资源管理的“帮帮团”上线啦——源启云原生基础设施管理平台
为助力企业提升基础资源一体化管理和交付效率,以更先进的基础设施管理方式来满足现代企业业务持续扩展和复杂化的需要,中电金信运用基础设施即代码(Infrastructure as Code,简称IaC)技 ...
- CreateElement vs DOM Parser
Benchmark: CreateElement vs DOM Parser - MeasureThat.net 可见 createElement 效率并不逊于 HTML.
- Java Function 接口用法
Java 8 中 Function 接口的介绍 Java 8 中提供了一个函数式接口 Function,这个接口表示对一个参数做一些操作然后返回操作之后的值.这个接口的有一个抽象方法 apply,这个 ...
- python 根据中文表头标题抓取动态(表格)文档数据
思路 如图左侧表头标题,要获得右侧数据.网页数据提取成汉字,表格数据间会有空格,用split()分隔成list.用index()查找某个汉字表头位置,输出list下一个位置既是要得到值 text2 = ...
- 在不同操作系统上安装 PostgreSQL
title: 在不同操作系统上安装 PostgreSQL date: 2024/12/26 updated: 2024/12/26 author: cmdragon excerpt: PostgreS ...