LC 638. Shopping Offers
In LeetCode Store, there are some kinds of items to sell. Each item has a price.
However, there are some special offers, and a special offer consists of one or more different kinds of items with a sale price.
You are given the each item's price, a set of special offers, and the number we need to buy for each item. The job is to output the lowest price you have to pay for exactly certain items as given, where you could make optimal use of the special offers.
Each special offer is represented in the form of an array, the last number represents the price you need to pay for this special offer, other numbers represents how many specific items you could get if you buy this offer.
You could use any of special offers as many times as you want.
Example 1:
Input: [2,5], [[3,0,5],[1,2,10]], [3,2]
Output: 14
Explanation:
There are two kinds of items, A and B. Their prices are $2 and $5 respectively.
In special offer 1, you can pay $5 for 3A and 0B
In special offer 2, you can pay $10 for 1A and 2B.
You need to buy 3A and 2B, so you may pay $10 for 1A and 2B (special offer #2), and $4 for 2A.
Example 2:
Input: [2,3,4], [[1,1,0,4],[2,2,1,9]], [1,2,1]
Output: 11
Explanation:
The price of A is $2, and $3 for B, $4 for C.
You may pay $4 for 1A and 1B, and $9 for 2A ,2B and 1C.
You need to buy 1A ,2B and 1C, so you may pay $4 for 1A and 1B (special offer #1), and $3 for 1B, $4 for 1C.
You cannot add more items, though only $9 for 2A ,2B and 1C.
Note:
- There are at most 6 kinds of items, 100 special offers.
- For each item, you need to buy at most 6 of them.
- You are not allowed to buy more items than you want, even if that would lower the overall price.
Runtime: 53 ms, faster than 12.62% of Java online submissions for Shopping Offers.
自己的揭发,但是时间有点低,纯粹暴力搜索。
先把总的不加优惠券的价格算出来,然后每次遍历优惠券,直到不能再用优惠券。每次用优惠券都是DFS,且保存当前优化的结果,这又是BFS。
class Solution {
public int shoppingOffers(List<Integer> price, List<List<Integer>> special, List<Integer> needs) {
int maxval = 0;
int cnt = 0;
for(int x : needs){
maxval += x * price.get(cnt);
cnt++;
}
return helper(price, special, needs, maxval);
}
int helper(List<Integer> price, List<List<Integer>> special, List<Integer> needs, int maxval){
int tmpmax = maxval;
for(int i=0; i<special.size(); i++){
List<Integer> newneeds = new ArrayList<>();
for(int n : needs) newneeds.add(n);
boolean canuse = true;
int delta = maxval;
for(int j=0; j<special.get(i).size()-1; j++){
if(newneeds.get(j) < special.get(i).get(j)){
canuse = false;
break;
}else {
delta -= special.get(i).get(j) * price.get(j);
newneeds.set(j, newneeds.get(j) - special.get(i).get(j));
}
}
if(canuse) {
delta += special.get(i).get(special.get(i).size()-1);
tmpmax = Math.min(tmpmax, helper(price, special, newneeds, delta));
}
}
return tmpmax;
}
}
一个更好的解法
Runtime: 12 ms, faster than 90.29% of Java online submissions for Shopping Offers.
优化解法,
1. 遍历的时候,可以记录当前index,也就是说不需要每次从头遍历,因为只是使用优惠券顺序的关系而已。
2. 我之前会用backtracking偷懒只用一个needs,用到了set,这十分费时。直接开一个新的newneeds,每次用add,效果十分好。
class Solution {
public int shoppingOffers(List<Integer> price, List<List<Integer>> special, List<Integer> needs) {
int maxval = 0;
int cnt = 0;
for(int x : needs){
maxval += x * price.get(cnt);
cnt++;
}
return helper(price, special, needs, maxval,0);
}
int helper(List<Integer> price, List<List<Integer>> special, List<Integer> needs, int maxval, int spindex){
int tmpmax = maxval;
// StringBuilder sb = new StringBuilder();
// for(int i=0; i<needs.size(); i++){
// sb.append(needs.get(i));
// sb.append("+");
// }
// if(mp.containsKey(sb.toString())) return mp.get(sb.toString());
for(int i=spindex; i<special.size(); i++){
//List<Integer> newneeds = new ArrayList<>();
//for(int n : needs) newneeds.add(n);
//if(special.get(i).get(special.get(i).size()-1) <= maxval - tmpmax) continue;
//boolean canuse = true;
int delta = maxval;
List<Integer> newneeds = new ArrayList<>();
for(int j=0; j<special.get(i).size()-1; j++){
//if(newneeds.get(j) < special.get(i).get(j)){
if(needs.get(j) < special.get(i).get(j)){
//canuse = false;
newneeds = null;
break;
//needs.set(j, needs.get(j) - special.get(i).get(j));
}else {
delta -= special.get(i).get(j) * price.get(j);
newneeds.add(needs.get(j) - special.get(i).get(j));
//newneeds.set(j, newneeds.get(j) - special.get(i).get(j));
//needs.set(j, needs.get(j) - special.get(i).get(j));
}
}
if(newneeds != null) {
delta += special.get(i).get(special.get(i).size()-1);
tmpmax = Math.min(tmpmax, helper(price, special, newneeds, delta, i));
}
// for(int k=0; k<special.get(i).size()-1; k++){
// needs.set(k, needs.get(k) + special.get(i).get(k));
// }
}
// sb = new StringBuilder();
// for(int k=0; k<needs.size(); k++){
// sb.append(k);
// sb.append("+");
// }
// mp.put(sb.toString(), maxval);
return tmpmax;
}
}
LC 638. Shopping Offers的更多相关文章
- LeetCode 638 Shopping Offers
题目链接: LeetCode 638 Shopping Offers 题解 dynamic programing 需要用到进制转换来表示状态,或者可以直接用一个vector来保存状态. 代码 1.未优 ...
- Week 9 - 638.Shopping Offers - Medium
638.Shopping Offers - Medium In LeetCode Store, there are some kinds of items to sell. Each item has ...
- 638. Shopping Offers
In LeetCode Store, there are some kinds of items to sell. Each item has a price. However, there are ...
- 【leetcode】638. Shopping Offers
题目如下: In LeetCode Store, there are some kinds of items to sell. Each item has a price. However, ther ...
- 【LeetCode】638. Shopping Offers 解题报告(Python & C++)
作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 DFS 回溯法 日期 题目地址:https://le ...
- Leetcode之深度优先搜索&回溯专题-638. 大礼包(Shopping Offers)
Leetcode之深度优先搜索&回溯专题-638. 大礼包(Shopping Offers) 深度优先搜索的解题详细介绍,点击 在LeetCode商店中, 有许多在售的物品. 然而,也有一些大 ...
- 洛谷P2732 商店购物 Shopping Offers
P2732 商店购物 Shopping Offers 23通过 41提交 题目提供者该用户不存在 标签USACO 难度提高+/省选- 提交 讨论 题解 最新讨论 暂时没有讨论 题目背景 在商店中, ...
- poj 1170 Shopping Offers
Shopping Offers Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 4696 Accepted: 1967 D ...
- USACO 3.3 Shopping Offers
Shopping OffersIOI'95 In a certain shop, each kind of product has an integer price. For example, the ...
随机推荐
- 四、DML语言
目录 简介 主要操作 插入语句 语法 修改语句 修改单表 删除语句 DELETE TRUNCATE 两种删除总结 简介 DML语言就是数据操作语言 主要操作 插入:insert 修改:update 删 ...
- Vue介绍:vue导读3
一.全局组件 二.父组件传递信息给子组件 三.子组件传递信息给父组件 四.vue项目开发 一.全局组件 <body> <!-- 两个全局vue实例可以不用注册全局组件,就可以使用 - ...
- jedis五种数据类型的方法解释
常用命令 1)连接操作命令 quit:关闭连接(connection) auth:简单密码认证 help cmd: 查看cmd帮助,例如:help quit 2)持久化 save:将数据同步保存到磁盘 ...
- .net面试题——20190718
文章:Dapper.Net实现增删改查 autofac automap 异步 委托 依赖注入
- 两个对象的 hashCode()相同,则 equals()也一定为 true,对吗?(未完成)
两个对象的 hashCode()相同,则 equals()也一定为 true,对吗?(未完成)
- win10 出现 No AMD graphics driver is installed or the AMD driver is not functioning properly .....
原因:win10的自动更新的功能没有关闭,更新有时候会出现显卡驱动更新不及时出现的问题. 解决方法一:使用 驱动人生(或者等等....) 进行升级驱动. 解决方法二:手动升级. 1.打开设备管理器 2 ...
- 在laravel5.8中集成swoole组件----用协程实现的服务端和客户端(二)---静态文件如何部署
目前,较为成熟的技术是采用laravelS组件,注意和laravel 区别laravelS多了一个大写的S,由于laravelS默认监听5200端口,所以laravel项目要做一些调整 例如: 静态文 ...
- API接口防止参数篡改和重放攻击
{近期领导要求我对公司业务的支付类的ocr接口做研究,是否存在支付接口重放攻击,so.....} API重放攻击(Replay Attacks)又称重播攻击.回放攻击.他的原理就是把之前窃听到的数据原 ...
- Codeforces Round #586 (Div. 1 + Div. 2) B. Multiplication Table
链接: https://codeforces.com/contest/1220/problem/B 题意: Sasha grew up and went to first grade. To cele ...
- axios请求提交的form data格式 明明是JSON格式的参数却转成了字符串格式
问题:传的参数成为了字符数格式 解决:把参数的格式转换 const params = new URLSearchParams() params.append('USER_LOGIN', 'admin' ...