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:

  1. There are at most 6 kinds of items, 100 special offers.
  2. For each item, you need to buy at most 6 of them.
  3. 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的更多相关文章

  1. LeetCode 638 Shopping Offers

    题目链接: LeetCode 638 Shopping Offers 题解 dynamic programing 需要用到进制转换来表示状态,或者可以直接用一个vector来保存状态. 代码 1.未优 ...

  2. Week 9 - 638.Shopping Offers - Medium

    638.Shopping Offers - Medium In LeetCode Store, there are some kinds of items to sell. Each item has ...

  3. 638. Shopping Offers

    In LeetCode Store, there are some kinds of items to sell. Each item has a price. However, there are ...

  4. 【leetcode】638. Shopping Offers

    题目如下: In LeetCode Store, there are some kinds of items to sell. Each item has a price. However, ther ...

  5. 【LeetCode】638. Shopping Offers 解题报告(Python & C++)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 DFS 回溯法 日期 题目地址:https://le ...

  6. Leetcode之深度优先搜索&回溯专题-638. 大礼包(Shopping Offers)

    Leetcode之深度优先搜索&回溯专题-638. 大礼包(Shopping Offers) 深度优先搜索的解题详细介绍,点击 在LeetCode商店中, 有许多在售的物品. 然而,也有一些大 ...

  7. 洛谷P2732 商店购物 Shopping Offers

    P2732 商店购物 Shopping Offers 23通过 41提交 题目提供者该用户不存在 标签USACO 难度提高+/省选- 提交  讨论  题解 最新讨论 暂时没有讨论 题目背景 在商店中, ...

  8. poj 1170 Shopping Offers

    Shopping Offers Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 4696   Accepted: 1967 D ...

  9. USACO 3.3 Shopping Offers

    Shopping OffersIOI'95 In a certain shop, each kind of product has an integer price. For example, the ...

随机推荐

  1. CentOS系统 Amoeba+MySql主从读写分离配置 适合新手傻瓜式教程!-----仅供参考!

    废话不说,直接开始: 一.安装mysql的三种方式,这里采用第2种(安装方式不再详解,请参照) http://www.cnblogs.com/babywaa/articles/4837946.html ...

  2. sklearn--回归

    一.线性回归 LinearRegression类就是我们平时所说的普通线性回归,它的损失函数如下所示: 对于这个损失函数,一般有梯度下降法和最小二乘法两种极小化损失函数的优化方法,而scikit-le ...

  3. 小辣椒之lombok-1.18.8.jar的使用

    1.下载lombok-1.18.8.jar包 http://central.maven.org/maven2/org/projectlombok/lombok/1.18.8/lombok-1.18.8 ...

  4. pip安装tesserocr时报错

    在Xubuntu上的python2虚拟环境中, 使用pip安装tesserocr时报错error: command 'x86_64-linux-gnu-gcc' failed with exit st ...

  5. zabbix验证微信

    在Zabbix服务端设置邮件报警,当被监控主机宕机或者达到触发器预设值时,会自动发送报警邮件到指定邮箱. 具体操作: 以下操作在Zabbix监控服务端进行 备注:Zabbix监控服务端 操作系统:Ce ...

  6. Error resolving template “pages”, template might not exist or might not be accessible by any of the configured Template Resolver 或者 springboot使用thymeleaf时报html没有结束标签

    application.properties配置文件 spring.thymeleaf.prefix=classpath:/templates/ spring.thymeleaf.suffix=.ht ...

  7. Python3下UnicodeDecodeError:‘ASCII’ codec cant decode..(128)

    今天准备用Keras跑一下LeNet的程序,结果总是编码出错 源代码是2.7写的,编码格式是utf-8.然后尝试网上各种方法不适用,最后还是解决了 源代码: data = gzip.open(r'C: ...

  8. JS用正则替换特殊字符

    'abcabce'.replace(/abc/g,'') "e" 'abcabce$E$'.replace(/$E$/g,'') "abcabce$E$" 'a ...

  9. 理解Event冒泡模型

    本文探索一下Event的冒泡过程和初学遇到的几个小bug DOM Event概述 Event接口是检测在DOM中的发生的所有事件,我们一直在用,而且从DOM的很早的版本就一直在用着.早期的网景(后来的 ...

  10. url传参特殊字符问题(+、%、#等)

    这样的话,你传的大多数带特殊符号的参数,都能在后台拿到,但是, url中可能用到的特殊字符及在url中的经过编码后的值:(此表格借鉴) 字符   特殊字符的含义 URL编码 #   用来标志特定的文档 ...