230th Weekly Leetcode Contest
题目二
题目描述
5690. 最接近目标价格的甜点成本 难度:中等-中等
题解
解法一:三进制状态压缩
考虑到baseCosts、toppingCosts的长度最多都为10,每一种辅料都有加0、1、2份的选择,因此可以考虑三进制状态压缩求解。类似二进制的状态压缩。
以10种辅料为例。



直到curJ为0。
代码:
1 class Solution {
2 public int closestCost(int[] baseCosts, int[] toppingCosts, int target) {
3 Arrays.sort(baseCosts);
4 Arrays.sort(toppingCosts);
5 int nearestCost=0;//记录最接近的成本
6 int minGap = Integer.MAX_VALUE;//记录目前最接近的成本和目标成本target的差
7 for(int i=0;i<baseCosts.length;i++){//每一种基料
8 for(int j=0;j<Math.pow(3,toppingCosts.length);j++){//三进制位串
9 int cost = baseCosts[i];
10 int curJ = j;//现在的位串
11 int index = 0;
12 while(curJ!=0){//取位串的每一位
13 int curBit = curJ % 3;//位串的当前一位
14 curJ /= 3;
15 cost += toppingCosts[index]*curBit;
16 index++;
17 }
18 if(Math.abs(cost-target)==0){//若现在的成本恰好是target,直接返回
19 return target;
20 }else if(Math.abs(cost-target)<minGap){//若目前的成本更接近target,更新nearestCost和minGap;
21 minGap = Math.abs(cost-target);
22 nearestCost = cost;
23 }else if(Math.abs(cost-target)==minGap){//若目前的成本和已经找到的最接近成本与目标成本的差距相等,取成本较小者(题目要求)
24 if(cost<nearestCost){
25 nearestCost = cost;
26 }
27 }
28 }
29 }
30 return nearestCost;
31 }
32 }
略微优化代码:
当目前的cost>target 且 Math.abs(cost-target)>minGap时,由于cost随着位串的遍历单调不减,这种方案可以不再考虑,肯定不优于已经求得的成本,因此可以剪枝。
1 class Solution {
2 public int closestCost(int[] baseCosts, int[] toppingCosts, int target) {
3 Arrays.sort(baseCosts);
4 Arrays.sort(toppingCosts);
5 int nearestCost=Integer.MAX_VALUE;//记录最接近的成本
6 int minGap = Integer.MAX_VALUE;//记录目前最接近的成本和目标成本target的差
7 for(int i=0;i<baseCosts.length;i++){//每一种基料
8 for(int j=0;j<Math.pow(3,toppingCosts.length);j++){//三进制位串
9 int cost = baseCosts[i];
10 int curJ = j;//现在的位串
11 int index = 0;
12 boolean isCut = false;
13 while(curJ!=0){//取位串的每一位
14 int curBit = curJ % 3;//位串的当前一位
15 curJ /= 3;
16 cost += toppingCosts[index]*curBit;
17 index++;
18 if(cost>target&&Math.abs(cost-target)>minGap){//剪枝
19 isCut = true;
20 break;
21 }
22 }
23 if(isCut==true){//剪枝
24 continue;
25 }
26 if(Math.abs(cost-target)==0){//若现在的成本恰好是target,直接返回
27 return target;
28 }else if(Math.abs(cost-target)<minGap){//若目前的成本更接近target,更新nearestCost和minGap;
29 minGap = Math.abs(cost-target);
30 nearestCost = cost;
31 }else if(Math.abs(cost-target)==minGap){//若目前的成本和已经找到的最接近成本与目标成本的差距相等,取成本较小者(题目要求)
32 if(cost<nearestCost){
33 nearestCost = cost;
34 }
35 }
36 }
37 }
38 return nearestCost;
39 }
40 }
解法二:动态规划
本题数据范围很小,暴力枚举辅料组合就可以通过,但时间复杂度为指数级。
把问题转化为背包问题,可以将时间复杂度降低到多项式级别。
- 因为每种辅料最多可以用两次,所以直接把每种辅料变成两个。
- 基料必须且只能选一种,可以首先处理好。
题目三
题目描述
5691. 通过最少操作次数使数组的和相等 难度:中等-中等
题解-贪心法
看到题目后,总结出以下要点:
- 总和大的数组中元素减小等同于总和小的数组中元素增大(这样转化后,我们在求得每个数组的总和后,两个总和不再狭义地代表每个数组的总和)
- 要实现最小操作次数,考虑从最小的元素开始增大,增大到最大,如1直接变成6;或从最大的元素开始减小,减小到最小,如6直接变成1。
- 如何判断是否已经满足条件呢?既然我们是保证每次改变量都是最大,那么,如果这次改变后,最好是两个总和相等,那么功德圆满,即使不相等,但若实现了两个总和的反转(即大小关系反转),那么因为我们是最大的改变量,我们只需在心中将这次的改变量减小一点,如原来是2直接到6,我们可以改变2到5、4等等,肯定能实现两个总和的相等。
- 题目中固定1-6的数据范围,我们可以指定一个大小为6的数组inc[] (为了方便,inc[0]不用),inc[1]代表两个数组中1或6的个数(总和大的数组中的1可以增大到6、总和小的数组中的6可以减小到1),inc[2]代表两个数组中2或5的个数(总和大的数组中的2可以增大到6、总和小的数组中的5可以减小到1)
1 class Solution {
2 public int minOperations(int[] nums1, int[] nums2) {
3 if(Math.min(nums1.length,nums2.length)*6<Math.max(nums1.length,nums2.length)*1){
4 return -1;
5 }
6 int nums1Sum = 0;
7 int nums2Sum = 0;
8 for(int i=0;i<nums1.length;i++){//统计nums1的和
9 nums1Sum += nums1[i];
10 }
11 for(int i=0;i<nums2.length;i++){//统计nums2的和
12 nums2Sum += nums2[i];
13 }
14 if(nums1Sum==nums2Sum){
15 return 0;
16 }
17 int[] inc = new int[6];//统计nums1 和 nums2中可以增大或减小的数的个数,即1 ~ 5的个数;为了方便,数组容量多1,inc[0]不用
18 if(nums1Sum<nums2Sum){
19 for(int i=0;i<nums1.length;i++){//nums1Sum较小,考虑nums1中可以增大的数
20 if(nums1[i]<6){ // 1 ~ 5
21 inc[nums1[i]]++;
22 }
23 }
24 for(int i=0;i<nums2.length;i++){//nums2Sum较大,考虑nums2中可以减小的数
25 if(nums2[i]>1){ // 2 ~ 6 nums2减小相当于nums1增大,如6减小到1,同1增加到6t;又如5减小到1同2增加到6
26 inc[7-nums2[i]]++;
27 }
28 }
29 int cnt=0;
30 for(int i=1;i<=5;i++){//此后的nums1Sum已经不是真正意义上的nums1的和
31 while(inc[i]!=0){
32 nums1Sum += 6 - i;
33 inc[i]--;
34 cnt++;
35 if(nums1Sum>=nums2Sum){
36 return cnt;
37 }
38 }
39 }
40 }else if(nums1Sum>nums2Sum){
41 for(int i=0;i<nums2.length;i++){//nums2Sum较小,考虑nums2中可以增大的数
42 if(nums2[i]<6){ // 1 ~ 5
43 inc[nums2[i]]++;
44 }
45 }
46 for(int i=0;i<nums1.length;i++){//nums1Sum较大,考虑nums1中可以减小的数
47 if(nums1[i]>1){ // 2 ~ 6 nums1减小相当于nums2增大,如6减小到1,同1增加到6;又如5减小到1同2增加到6
48 inc[7-nums1[i]]++;
49 }
50 }
51 int cnt=0;
52 for(int i=1;i<=5;i++){
53 while(inc[i]!=0){
54 nums2Sum += 6 - i;
55 inc[i]--;
56 cnt++;
57 if(nums2Sum>=nums1Sum){
58 return cnt;
59 }
60 }
61 }
62 }
63 return -1;
64 }
65 }
题目三
题目描述
5691. 通过最少操作次数使数组的和相等 难度:困难-简单
题解
230th Weekly Leetcode Contest的更多相关文章
- LeetCode Contest 166
LeetCode Contest 166 第一次知道LeetCode 也有比赛. 很久没有打过这种线上的比赛,很激动. 直接写题解吧 第一题 很弱智 class Solution { public: ...
- leetcode contest 20
Q1: 520. Detect Capital Given a word, you need to judge whether the usage of capitals in it is right ...
- leetcode bugfree note
463. Island Perimeterhttps://leetcode.com/problems/island-perimeter/就是逐一遍历所有的cell,用分离的cell总的的边数减去重叠的 ...
- [LeetCode] Reverse Words in a String III 翻转字符串中的单词之三
Given a string, you need to reverse the order of characters in each word within a sentence while sti ...
- C#版(击败97.76%的提交) - Leetcode 557. 反转字符串中的单词 III - 题解
版权声明: 本文为博主Bravo Yeung(知乎UserName同名)的原创文章,欲转载请先私信获博主允许,转载时请附上网址 http://blog.csdn.net/lzuacm. Leetcod ...
- LeetCode算法题-Reverse Words in a String III(Java实现)
这是悦乐书的第259次更新,第272篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第126题(顺位题号是557).给定一个字符串,您需要反转句子中每个单词中的字符顺序,同 ...
- Leetcode#557. Reverse Words in a String III(反转字符串中的单词 III)
题目描述 给定一个字符串,你需要反转字符串中每个单词的字符顺序,同时仍保留空格和单词的初始顺序. 示例 1: 输入: "Let's take LeetCode contest" 输 ...
- 【leetcode】557. Reverse Words in a String III
Algorithm [leetcode]557. Reverse Words in a String III https://leetcode.com/problems/reverse-words-i ...
- leetCode 557. Reverse Words in a String I
Input: "Let's take LeetCode contest" Output: "s'teL ekat edoCteeL tsetnoc" 解:输入一 ...
随机推荐
- Docker Swarm(六)Label 节点标签与服务约束
前言 多节点 Swarm 集群下,可能节点的配置不同(比如 CPU.内存等),部署着不同类型的服务(比如 Web服务.Job服务等),当这些服务以 Service 或者 Stack 的形式部署到集群, ...
- Docker镜像的仓库及底层依赖的核心技术(3)
一.docker镜像的仓库 仓库分为公共仓库和私有仓库 DockerHub的官方仓库:https://hub.docker.com DockerPool社区仓库:https://dl.dockerpo ...
- exec函数族实例解析-(转自blankqdb)
fork()函数通过系统调用创建一个与原来进程(父进程)几乎完全相同的进程(子进程是父进程的副本,它将获得父进程数据空间.堆.栈等资源的副本.注意,子进程持有的是上述存储空间的"副本&quo ...
- System Verilog MCDF(二)
整形器的接口时序: reg,grant是维持了两个clk的. chid ,length在发送数据期间不可以变化. 第一个data数据必须在start上升沿的同一个clk发送. reg,grant两者之 ...
- TheSuperego 实验五 团队作业2:毕业设计选题系统
项目 内容 课程班级博客链接 https://edu.cnblogs.com/campus/xbsf/2018CST 这个作业要求链接 https://www.cnblogs.com/nwnu-dai ...
- RISC-V与DSA计算机架构
RISC-V与DSA计算机架构 相信所有和计算机体系结构打过交道的朋友们都看过David Patterson与John Hennessy的煌煌巨作,<计算机体系架构:量化研究方法>.两位在 ...
- AlexeyAB DarkNet YOLOv3框架解析与应用实践(一)
AlexeyAB DarkNet YOLOv3框架解析与应用实践(一) Darknet: C语言中的开源神经网络 Darknet是一个用C和CUDA编写的开源神经网络框架.它速度快,易于安装,支持C ...
- BEP 7:CUDA外部内存管理插件(下)
BEP 7:CUDA外部内存管理插件(下) Numba依赖 向库中添加EMM插件的实现自然会使Numba成为库的依赖项,而以前可能没有.为了使依赖关系可选,如果需要的话,可以有条件地实例化并注册EMM ...
- CodeGen结构循环回路
CodeGen结构循环回路 structure循环是一个模板文件构造,它允许您迭代CodeGen拥有的有关结构的集合.为了使用结构循环,必须同时基于多个存储库结构生成代码. CodeGen可以通过以下 ...
- .h5图像文件(数据集)的读取并存储 工具贴(二)
概述 H5文件是层次数据格式第5代的版本(Hierarchical Data Format,HDF5),它是用于存储科学数据的一种文件格式和库文件.由美国超级计算中心与应用中心研发的文件格式,用以存储 ...