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 ...
随机推荐
- STM32L1xx——ADC(中断/DMA)样例代码
此代码欲实现的功能是:使用中断或者DMA的方式采集滑动变阻器采集到的电压值,使用单ADC单通道采样! (由于不是直接需要电压,所以转换函数我就没列出来,可根据自身需要去网上查到转换的函数.) 代码结构 ...
- python基础:数据类型阶段总结
name =“ alex”1.移除name变量对应的值两边的空格,并输出处理结果 res=name.strip(’ ‘) print(res) 2.判断neme变量对应的值是 ...
- 团队项目-Beta版本发布
这个作业属于哪个课程 课程链接 这个作业要求在哪里 作业要求链接 团队名称 众志陈成 这个作业的目标 通过团队协作了解软件开发的大致流程,并在这个过程中体会调整与优化程序的方法,为以后真实的软件开发奠 ...
- 关于Java的Object.clone()方法与对象的深浅拷贝
文章同步更新在个人博客:关于Java的Object.clone()方法与对象的深浅拷贝 引言 在某些场景中,我们需要获取到一个对象的拷贝用于某些处理.这时候就可以用到Java中的Object.clon ...
- jQuery.fn.extend() 函数详解
jQuery.fn.extend()函数用于为jQuery扩展一个或多个实例属性和方法(主要用于扩展方法). jQuery.fn是jQuery的原型对象,其extend()方法用于为jQuery的原型 ...
- CSS基础学习 16.CSS过渡
- redis.conf 文件解释
# Redis示例配置文件 # 注意单位问题:当需要设置内存大小的时候,可以使用类似1k.5GB.4M这样的常见格式: # # 1k => 1000 bytes # 1kb => 1024 ...
- 浅谈script中的defer与async
一直没有在script上使用过别的属性,今天看基础代码才发现这个,搜索.记录下,以便记忆. 大家都知道,js加载会造成阻塞,阻碍页面的继续渲染. defer: js与页面同步加载,不阻碍页面渲染,会在 ...
- Linq 分组查询
根据部门分组 ,然后存储部门下所有员工 public class Custom { public string dname { get; set; } public List<Employees ...
- JavaScript属性名和属性值
㈠属性名 var obj = new Object(); 向对象中添加属性 ⑴属性名: - 对象的属性名不强制要求遵守标识符的规范 什么乱七八糟的名 ...