Leetcode---350周赛

news2024/9/20 14:37:04

题目列表

6901. 总行驶距离

6890. 找出分区值

6893. 特别的排列

6447. 给墙壁刷油漆

一、总行驶距离

很显然,这题单纯就是一道数学应用题,我们要明白最关键的一点 :只有当mainTank>=5并且additionalTank>0时,才能发生副油箱的油转移到主油箱,代码如下

int distanceTraveled(int mainTank, int additionalTank){
    int ans=0;
    while(mainTank/5){//这一条件等价于mainTank>=5
        ans+=50;
        mainTank-=5;
        if(additionalTank>=1){
            mainTank++;
            additionalTank--;
        }
    }
    ans+=mainTank*10;//记得加上主油箱中剩余的油所能跑的路程
    return ans;
}

二、找到分区值

 这题其实题目看懂就不算很难,就是让你将nums数组拆成俩个数组,找到第一个数组的max,第二个数组的min,返回max和min的最小差值,乍一看,这题好像需要枚举所有可能的拆分方法,但仔细看一下元素的个数范围,你就会知道这不现实,那么我们该怎么做?

首先,我们可以明确的是:我们可以通过分配数组元素,将任何一个数通过最大值或最小值拿出来,那么我们可不可以通过分配数组元素将任意两个数通过最大值和最小值的形式拿出来?

假设能这样操作,那么该问题就会变成找到两个数的差值最小的问题,而后面一个问题解决起来就会很容易,那么到底能不能这么操作呢?

我们假设原始数组中的两个数为x,y(x<=y),分成的两个数组分别是数组A和数组B,我们将x和大于y的数全部放到数组A,将剩余的数全部放入数组B,那么数组A的最小值就是x,数组B的最大值就是y,很显然我们能够选取出任意x,y

代码如下

int cmp(int*p1,int*p2){
    return *p1-*p2;
}
int findValueOfPartition(int* nums, int numsSize){
    qsort(nums,numsSize,sizeof(int),cmp);
    int ans=INT_MAX;
    for(int i=1;i<numsSize;i++){
        if(nums[i]-nums[i-1]<ans){
            ans=nums[i]-nums[i-1];
        }
    }
    return ans;
}

三、特别的排列

这题的思路其实不是很难,只要会回溯就能做出来,但是会超时,得用记忆化搜索,减少时间复杂度,或者直接用递推。

讲一下这类回溯的基本思路:首先要有数组记录数字有没有被使用过,因为需要考虑数字所在的位置问题,然后不断dfs找到适合当前位置的数字,直到将所有的数字都选上,记入答案 ,最后返回

这里的思路说的比较简单,具体的可以看该题的代码的逻辑

//注意:这是正常的回溯代码--->会超时
const int MOD=1e9+7;//题目要求答案取模,为了防止数字超出int的范围,我们将所有计算结果都取模
int* visited;//记录数字是否被使用
int dfs(int i,int depth,int n,int* nums){//i是前一个数的下标,depth记录用了几个数,n代表数的个数,nums数组
    if(depth==n){
        return 1;//返回1,代表找到一种组合方法
    }
    int res=0;
    for(int j=0;j<n;j++){
        if(!visited[j]&&(nums[i]%nums[j]==0||nums[j]%nums[i]==0)){
            visited[j]=1;
            res=(res+dfs(j,depth+1,n,nums))%MOD;
            visited[j]=0;
        }
    }
    return res%MOD;
}
int specialPerm(int* nums, int numsSize){
    int n=numsSize;
    int ans=0;
    visited=(int*)malloc(sizeof(int)*n);
    memset(visited,0,sizeof(int)*n);
    for(int i=0;i<n;i++){
        visited[i]=1;
        ans=(ans+dfs(i,1,n,nums))%MOD;
        visited[i]=0;
    }
    free(visited);
    return ans;
}

好,写到这一步,我们会发现超时,这里超时的原因和求较大值的斐波那契数列一样,相同的递归进行太多次,我们需要用数组记录我们已经计算过的dfs,这样之后我们在需要时,就不用计算直接返回,从而减少时间复杂度

而这题的难点在于:我们需要进行状态压缩之后才能进行记忆化搜索,而状态压缩在本题中就是将visited数组用二进制的数来表示

举个栗子:

 状态压缩后的代码如下:

//依旧会超时,但空间复杂度降低
const int MOD=1e9+7;
int visited;
int dfs(int i,int n,int* nums){
    if(visited==0){//visited==0,代表所有的数都被使用
        return 1;
    }
    int res=0;
    for(int j=0;j<n;j++){
        if((visited&(1<<j))&&(nums[i]%nums[j]==0||nums[j]%nums[i]==0)){
            visited^=(1<<j);
            res=(res+dfs(j,n,nums))%MOD;
            visited^=(1<<j);
        }
    }
    return res%MOD;
}
int specialPerm(int* nums, int numsSize){
    int n=numsSize;
    int ans=0;
    visited=(1<<n)-1;
    for(int i=0;i<n;i++){
        visited^=(1<<i);
        ans=(ans+dfs(i,n,nums))%MOD;
        visited^=(1<<i);
    }
    return ans;
}

下面我们只要有一个数组来储存已经计算过的dfs,从而实现记忆化搜索就行,但这个数组的形状(一维,二维...)大小是什么呢?

这里我们只要看dfs函数是由几个关键的参数决定的就行(因为dfs其实就是在求某种状态,我们要建立数组储存状态,肯定是看共多少种状态来决定数组的形状大小,而状态是由参数决定的,所以我们看参数),很显然nums是辅助型参数,不对dfs的状态产生影响,其他的都有影响,即两个参数=>二维数组,这两个参数的范围=>数组的大小

记忆化搜索的代码如下:

const int MOD=1e9+7;
int visited;
int**memo;
int dfs(int i,int n,int* nums){
    if(visited==0){
        return 1;
    }
    if(memo[i][visited]!=-1)
        return memo[i][visited];
    int res=0;
    for(int j=0;j<n;j++){
        if((visited&(1<<j))&&(nums[i]%nums[j]==0||nums[j]%nums[i]==0)){
            visited^=(1<<j);
            res=(res+dfs(j,n,nums))%MOD;
            visited^=(1<<j);
        }
    }
    return memo[i][visited]=res%MOD;
}
int specialPerm(int* nums, int numsSize){
    int n=numsSize;
    int ans=0;
    visited=(1<<n)-1;
    memo=(int**)malloc(sizeof(int*)*n);
    for(int i=0;i<n;i++){
        int s=1<<n;
        memo[i]=(int*)malloc(sizeof(int)*s);
        for(int j=0;j<s;j++){
            memo[i][j]=-1;
        }
    }
    for(int i=0;i<n;i++){
        visited^=(1<<i);
        ans=(ans+dfs(i,n,nums))%MOD;
        visited^=(1<<i);
    }
    for(int i=0;i<n;i++){
        free(memo[i]);
    }
    free(memo);
    return ans;
}

其实,这题还有一种写法,就是递推,我们可以直接将上面的记忆化搜索直接转换成递推

const int MOD=1e9+7;
int specialPerm(int* nums, int numsSize){
    int n=numsSize;
    int ans=0;
    int visited=(1<<n)-1;
    int s=1<<n;
    int f[n][s];
    memset(f,0,sizeof(f));
    for(int i=0;i<n;i++){
        f[i][0]=1;
    }
    //首先枚举状态,注意这里先枚举列,再枚举行,因为每一列的状态由它的上一列状态推出
    for(int i=1;i<s;i++){
        for(int j=0;j<n;j++){
            //然后计算每一个状态
            long long res=0;
            for(int k=0;k<n;k++){
                if((i&(1<<k))&&(nums[j]%nums[k]==0)||(nums[k]%nums[j]==0)){
                    res=(res+f[k][i^(1<<k)])%MOD;
                }
            }
            f[j][i]=res;
        }
    }
    for(int i=0;i<n;i++)
        ans=(ans+f[i][visited^(1<<i)])%MOD;
    return ans;
}

四、给墙壁刷油漆

 

 这题其实和上一题很相似,思路:一面墙要么是免费刷的消耗时间,要么是付费刷的增加时间和金额,取较小值,即dfs(i,j) = min { dfs(i - 1 ,j - 1) ,dfs( i - 1,j + time[ i ] ) + cost[ i ] } 

递归边界:1. j>=i+1,即剩下的墙可以免费刷,返回0     

                  2. i<0并且j<i+1,即没有墙可刷,但是还欠费,该方案不符合条件,返回一个正无穷(就是一个无法作为答案的正数,目的是在取较小值时,不影响答案取值)

递归入口:一开始,从最后一面墙开始(下标是n-1),时间为0,dfs(n-1,0)

代码如下:

//正常的递归:超时
long long dfs(int i,int j,int* cost,int* time)//剩余第0~i面墙,剩余花费的时间j,返回所需要的金额
{
    if(j>i)return 0;//等价j>=i+1,这里的i是用下标表示的墙的个数
    if(i<0)return INT_MAX;//i<0&&j<=i,即欠费时间
    return fmin(dfs(i-1,j-1,cost,time),dfs(i-1,j+time[i],cost,time)+cost[i]);
}
int paintWalls(int* cost, int costSize, int* time, int timeSize){
    int n=costSize;
    return dfs(n-1,0,cost,time);
}

//记忆化搜索--可以过
//dfs(i,j)=fmin(dfs(i-1,j),dfs(i-1,j-time[i])+cost[i])
#define MIN(x,y) ((x)>(y)?(y):(x))//这是宏定义,或者你定义函数都行,用来比较大小
//以下全局变量是为了减少函数参数个数,使dfs函数的逻辑更加清晰,当然把它们放入参数中也是可以的
int*Cost;
int*Time;
int**memo;
int N;
int dfs(int i,int j){
    if(j>=i+1)return 0;//如果免费的时间>=要刷的墙的数量,那么剩下的墙直接免费
    if(i<0)return INT_MAX/2;//如果墙全刷完后,j<i+1=0,返回正无穷(该值取决于题目的可能答案区间,和函数返回值类型)
    if(memo[i][j+N-1]!=-1)
        return memo[i][j+N-1];
    return memo[i][j+N-1]=MIN(dfs(i-1,j-1),dfs(i-1,j+Time[i])+Cost[i]);
}
int paintWalls(int* cost, int costSize, int* time, int timeSize){
    int n=costSize;
    N=costSize;
    Cost=cost;
    Time=time;
    memo=(int**)malloc(sizeof(int*)*n);
    for(int i=0;i<n;i++){
        memo[i]=(int*)malloc(sizeof(int)*(2*n));
        for(int j=0;j<2*n;j++){
            memo[i][j]=-1;
        }
    }
    int res = dfs(n-1,0);
    for(int i=0;i<n;i++){
        free(memo[i]);
    }
    free(memo);
    return res;
}

上面代码中的memo数组的第二维度(时间维度)的范围是[-(n-1),n],即最多连续有n-1面墙免费和n面墙的免费时间,其他的状态会被直接返回0或INT_MAX/2,而要表示负的情况我们只能将每个数加上n-1,得到[0,2n-1],而这是下标范围,我们数组大小就是2n

当然,这题其实也是一种01背包问题的变形,以后找时间出一期01背包问题的详解。

关注我,让你了解更多周赛题目!!!

不要忘记点赞,评论加收藏哦!!!!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/674477.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

操作系统 - 操作系统结构

✅作者简介&#xff1a;人工智能专业本科在读&#xff0c;喜欢计算机与编程&#xff0c;写博客记录自己的学习历程。 &#x1f34e;个人主页&#xff1a;小嗷犬的个人主页 &#x1f34a;个人网站&#xff1a;小嗷犬的技术小站 &#x1f96d;个人信条&#xff1a;为天地立心&…

【Kubernetes资源篇】StatefulSet无状态服务管理入门实战详解

文章目录 一、StatefulSet理论知识1、StatefulSet Pod控制器特性2、什么是有状态服务和无状态服务&#xff1f;3、Deployment和StatefulSet区别 二、案例&#xff1a;StatefulSet资源实战演示1、创建WEB站点并验证StatefulSet特点2、StatefulSet滚动更新 三、总结 一、Stateful…

JavaFX第四篇 Button按钮和事件处理

JavaFX第四篇 Button按钮和事件处理 1. 代码2. 讲解3. 代码仓库 上一篇我们讲解了Hello World演示&#xff0c;主要用到Label标签的功能&#xff0c; 这里我们要介绍的是最常用的控件之一&#xff1a;按钮 在现在的软件开发过程中还没发现没有用到按钮的应用&#xff0c; 基本上…

【数据库原理与实践】第八章至第十章作业汇总(更新中)

填空题部分&#xff1a; Chp 8 安全性与完整性 part 1&#xff1a; 数据库的安全性是指保护数据库以防止不合法的使用所造成的&#xff08; 数据泄露、更改或破坏 &#xff09;。计算机系统有三类安全性问题&#xff0c;即&#xff08; 技术安全 &#xff09;、管理安全和…

ROS:计算图

目录 一、ROS计算图简介二、节点&#xff08;Node&#xff09;三、节点管理器&#xff08;Master&#xff09;四、消息&#xff08;Message&#xff09;五、话题&#xff08;Topic&#xff09;六、服务&#xff08;Service&#xff09;七、动作&#xff08;Action&#xff09;八…

LabVIEW开发燃油阀自动性能测试系统

LabVIEW开发燃油阀自动性能测试系统 燃油阀是航空燃油控制系统的核心部件。燃油阀的流量滞后直接影响控制精度、稳定性和可靠性&#xff0c;而燃油阀生产的性能测试是至关重要的步骤。但是&#xff0c;由于流动滞后是非线性的&#xff0c;因此很难控制和进行实时测试。随着厂家…

CSS3-浮动

&#xff08;了解&#xff09;标准流&#xff1a;又称文档流&#xff0c;是浏览器在渲染显示网页内容时默认采用的一套排版规则&#xff0c;规定了应该以何种方式排列元素 常见标准流排版规则&#xff1a; 1. 块级元素&#xff1a;从上往下&#xff0c;垂直布局&am…

Matplotlib---柱形图

1. 绘制简单的柱状图 bar函数用于绘制柱状图。下面是bar函数的语法格式&#xff1a; bar(x, height, width0.8, bottomNone, aligncenter, dataNone, **kwargs)参数解释&#xff1a; x&#xff1a;指定每个柱子的x轴坐标。height&#xff1a;指定每个柱子的高度。width&…

TypeScript ~ 掌握基本类型 ②

作者 : SYFStrive 博客首页 : HomePage &#x1f4dc;&#xff1a; TypeScript ~ TS &#x1f4cc;&#xff1a;个人社区&#xff08;欢迎大佬们加入&#xff09; &#x1f449;&#xff1a;社区链接&#x1f517; &#x1f4cc;&#xff1a;觉得文章不错可以点点关注 &…

vue的学习

title: VUE 一、Vue简介 1.1 简介 ::: tip Vue (读音 /vjuː/&#xff0c;类似于 view) 是一套用于构建用户界面的渐进式的js框架&#xff0c;发布于 2014 年 2 月。与其它大型框架不同的是&#xff0c;Vue 被设计为可以自底向上逐层应用。Vue 的核心库只关注视图层&#xff0…

F. Gardening Friends(树的直径)

Problem - 1822F - Codeforces 两位朋友Alisa和Yuki在他们的花园里种了一棵有n个顶点的树。树是一个无向图&#xff0c;没有循环、回路或多重边。这棵树中的每条边都有一个长度为k。最初&#xff0c;顶点1是树的根。 Alisa和Yuki种植这棵树不仅仅是为了好玩&#xff0c;而是想…

基于SpringBoot的流浪动物救助平台

1.引言 系统管理也都将通过计算机进行整体智能化操作&#xff0c;对于流浪动物救助平台所牵扯的管理及数据保存都是非常多的&#xff0c;例如首页、个人中心、会员管理、志愿者管理、流浪动物信息管理、领养信息管理、取消领养信息管理、志愿团队活动管理、志愿者申请表管理、…

【性能设计篇】数据库拓展

前两篇文章介绍了分布式存储的机制&#xff0c;为保证数据的高性能以及可拓展&#xff0c;采用分片/分区机制。为保证数据的高可用性&#xff0c;采用复制/镜像机制存储数据。一份数据存储多份。而这两种方式在数据中&#xff0c;就是分片/分区机制。分库分表/读写分离。 读写…

掌握这个关键技术,让你的APP开发事半功倍!——Flutter与其他方案的区别

Flutter动机是什么&#xff0c;解决啥痛点&#xff1f;优势在哪里&#xff1f; 介绍Flutter的历史背景和运行机制&#xff0c;并以界面渲染过程为例与你讲述其实现原理。 1 Flutter的历史背景 为不同的操作系统开发拥有相同功能的应用程序&#xff0c;开发人员只有两个选择&…

gitTortoise图形化工具下载步骤

一&#xff0c;简介 本文主要介绍如何下载安装gitTortoise图形化工具来管理和提交代码。 二&#xff0c;步骤介绍 2.1 安装包下载 下载地址&#xff1a;https://download.tortoisegit.org/tgit/ 打开后&#xff0c;界面如下&#xff1a; 点击选择最新的稳定release版本&am…

二分查找算法及实现

概念 二分查找算法也成为折半查找,是一种非常高效的工作于有序数组的查找算法. 实现 需求 现有一个有序数组 arr [1,2,3,4,5,6,7,8,9], 目标值target 2 ,要求返回目标值target在数组arr中的索引,没有则返回-1; 代码实现 private int binarySearch(int[] arr, int target)…

Android加快你的编译速度

工欲善其事&#xff0c;必先利其器。如果每次运行项目都要花费5-10分钟&#xff0c;那人的心态都要崩了。 Gradle构建流程 Gradle 的生命周期可以分为大的三个部分&#xff1a;初始化阶段&#xff08;Initialization Phase)&#xff0c;配置阶段&#xff08;Configuration Pha…

Linux学习之网络管理和配置文件

在CentOS 7中有两种网络配置方法&#xff1a;SysV&#xff08;也称为Sys 5&#xff09;和Systemd。 SysV的命令如下&#xff1a; service network start|stop|restart|status chkconfig --list network Systemd的命令如下&#xff1a; systemctl list-unit-files NetworkManage…

Linux->线程同步

目录 前言&#xff1a; 1 线程同步引入 2 条件变量 2.1 线程饥饿 2.2 条件变量接口 2.3 添加条件变量 3 生产者和消费者模型 前言&#xff1a; 本篇主要讲解了关于线程同步的相关知识&#xff0c;还有生产者和消费者模型的认识和使用。 1 线程同步引入 在讲解线程同步之…

【Unity3D】基于深度和法线纹理的边缘检测方法

1 前言 边缘检测特效中使用屏后处理技术&#xff0c;通过卷积运算计算梯度&#xff0c;检测每个像素周围像素的亮度差异&#xff0c;以识别是否是边缘像素&#xff1b;选中物体描边特效中也使用了屏后处理技术&#xff0c;通过 CommandBuffer 获取目标物体渲染后的模板纹理&…