3.7练习题解

一共五道题:

1. PERKET:

 

观察容易发现n的值很小,所以我们可以考虑使用dfs的方法进行解答,首先我们可以考虑一共有n种配料,那么我们就可以考虑到可以选择1到n种配料数目,然后基于这个思路我们再对其进行判断当选择几种并且选择哪几种能够得到最小的酸甜度匹配,于是代码就如下:

#include<bits/stdc++.h>
using namespace std;
const int N=50;
int n,ans=1<<20,x=1,y=0;
int a[N],b[N],book[N];
void dfs(int c){if(c==0){ans=min(ans,abs(x-y));return;} for(int i=1;i<=n;i++){if(book[i]){x*=a[i];y+=b[i];--c;book[i]=0;dfs(c);book[i]=1;++c;x/=a[i];y-=b[i];}}
}
int main(){ios::sync_with_stdio(false);cin.tie(nullptr);cout.tie(nullptr);cin>>n;for(int i=1;i<=n;i++){cin>>a[i]>>b[i];}for(int i=1;i<=n;i++){x*=a[i];y+=b[i];}ans=min(ans,abs(x-y));for(int i=1;i<=n;i++){ans=min(ans,abs(a[i]-b[i]));}for(int i=2;i<n;i++){memset(book,1,sizeof(book));x=1;y=0;dfs(i);}cout<<ans<<endl;return 0;
}

 引入了book数组来标记在dfs过程中已经被使用过的配料,然后先把选择一种和n种配料的情况给算出来并更新最小的ans,接下来再判断当选择的配料数目为2到n-1的时候,对应的最小ans,并利用dfs进行不断更新。

2.迷宫:

 这道题目就是最经典和典型的dfs模板题目了,直接上代码吧:

#include<iostream>
#include<cstring>
using namespace std;
int n,m,t;
int startx,starty,p,q;
const int N=10;
int book[50][50],a[50][50];
int ans=0;
void dfs(int x,int y){if(x==p && y==q){++ans;return;} int next[4][2]={{0,1},{1,0},{0,-1},{-1,0}};int tx,ty;for(int i=0;i<=3;i++){tx=x+next[i][0];ty=y+next[i][1];if(tx < 1 || tx > n || ty < 1 || ty > m)continue;if(!book[tx][ty] && a[tx][ty]){book[tx][ty] = 1;dfs(tx,ty);if(a[tx][ty]) book[tx][ty]=0;}}
}
int main(){cin >> n >> m >> t;cin >> startx >> starty >> p >> q;memset(book,0,sizeof(book));for(int i=1;i<=n;i++){for(int j=1;j<=m;j++)a[i][j]=1;}for(int i=1;i<=t;i++){int x,y;cin>>x>>y;a[x][y]=0;}dfs(startx,starty);cout<<ans<<endl;return 0;
}

利用一个a数组代表是否有陷阱,利用book数组表示对于同一种方案是否有走过同一个格子,在dfs中利用一个二维数组来表示接下来一步的上下左右并进行枚举,不过我至今没有想明白为什么这只能获得70分。 

3.借教室:

#include <iostream>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long LL;
const int N = 1000010;
int n, m;
int w[N];//这个数组用来存储n天中每一天对应的可借出去的教室
int l[N], r[N], d[N];//l和r分别表示对应天数的差分端点,d数组则表示对应需要借出的教室
LL b[N];//当进行二分筛选的时候这一个数组就是拿来进行记录当订单数量为mid的时候每一天需要的教室数量
bool check(int mid)
{memset(b, 0, sizeof b);for (int i = 1; i <= mid; i ++ ){b[l[i]] += d[i];b[r[i] + 1] -= d[i];//对端点进行加减}for (int i = 1; i <= n; i ++ ){b[i] += b[i - 1];if (b[i] > w[i]) return false;//如果当天的需求大于所能提供的教室数量则说明要求的值在mid的左边}return true;
}
int main()
{scanf("%d%d", &n, &m);for (int i = 1; i <= n; i ++ )scanf("%d", &w[i]);for (int i = 1; i <= m; i ++ )scanf("%d%d%d", &d[i], &l[i], &r[i]);int l = 0, r = m;while (l < r){int mid = l + r + 1 >> 1;//我们的二分求出来的r是最后一份能恰好满足教室需求的订单,r+1则是第一个无法满足的订单,记得右移if (check(mid)) l = mid;else r = mid - 1;}if (r == m) puts("0");else printf("-1\n%d\n", r + 1);return 0;
}

 这道题目的方法就是差分加二分,订单的数量越多,那么剩余的空余教室的数量肯定会减少,因此我们考虑使用二分的方法进行查找,对于每一份订单,我们都有一个区间,在这个区间上加上订单所对应的借教室的数量,但是我们会发现题目当中,n和m的极限数据是10的6次方,如果对于每一个订单我们都将从第l天到第r天进行枚举并加上借教室的数量的话会导致tle,所以我们考虑使用差分的方式进行加速处理,这样处理的时间复杂度为O(n+m)logm。

以下是二分的两种模板:

bool check(int x) {/* ... */} // 检查x是否满足某种性质// 区间[l, r]被划分成[l, mid]和[mid + 1, r]时使用:
int bsearch_1(int l, int r)
{while (l < r){int mid = l + r >> 1;if (check(mid)) r = mid;    // check()判断mid是否满足性质else l = mid + 1;}return l;
}
// 区间[l, r]被划分成[l, mid - 1]和[mid, r]时使用:
int bsearch_2(int l, int r)
{while (l < r){int mid = l + r + 1 >> 1;if (check(mid)) l = mid;else r = mid - 1;}return l;
}

4.分巧克力:

这道题目的话 难度比排教室要小很多了,我们可以考虑使用二分来找到所给的巧克力总共可以分成的边长最大的数量大于k的答案,那么我们的代码可以这样去写:

#include<iostream>
using namespace std;
const int N=1e5+100;
int w[N],h[N];
int n,k;
bool pd(int x){int ans=0;for(int i=1;i<=n;i++){ans += (w[i]/x) * (h[i]/x);}if(ans>=k) return true;else return false;
}
int main(){cin>>n>>k;for(int i=1;i<=n;i++){cin>>w[i]>>h[i];}int l=1,r=100000;while(l<r){int mid=l+r+1>>1;if(pd(mid)) l=mid;else r=mid-1;}cout<<r<<endl;return 0;
}

5.技能升级

这是一道很难的题目,题解如下:

一定要记得开long long:

#include <iostream>
#include <cmath>
#define N 100005
using namespace std;
int n, m, l = 0, r = 1e6, mid, now, cnt, a[N], b[N];
long long ans;
long long sum (int r, int n, int t) // 求和
{int l = r - t * (n - 1);return (long long) (l + r) * n >> 1;
}
bool check (int x)
{long long res = 0;for (int i = 1; i <= n; i ++){if (a[i] > x) // 前提条件{res += ceil ((double) (a[i] - x) / b[i]);// 累计第 i 个技能发动的次数}}return res <= m; // 判断是否合法
}
int main ()
{cin >> n >> m;for (int i = 1; i <= n; i ++){cin >> a[i] >> b[i];}while (l < r) // 二分{mid = l + r >> 1;if (check (mid)) // 如果 x = mid 合法{r = mid;}else{l = mid + 1;}}for (int i = 1; i <= n; i ++){if (a[i] > l) // 前提条件{now = ceil ((double) (a[i] - l) / b[i]), cnt += now;// now 是第 i 个技能发动的次数,cnt 则是总共升级了多少次ans += sum (a[i], now, b[i]);// 总升级攻击力累加上右端点为 a[i],项数为 now,公差为 b[i] 的等差数列和}}cout << ans + (long long) l * (m - cnt); // 答案还要记得加上那些等于 l 的攻击力增加return 0;
}

也可以用另一种二分模板,代码如下:

#include<bits/stdc++.h>
#define int long long
using namespace std;
int n,m,ans;
const int N=1e3+100;
int a[N],d[N];
int sum(int An,int n,int dx){int A1=An-(n-1)*dx;return n*(A1+An)/2;
}
bool check(int x){int cnt=0;for(int i=1;i<=n;i++){if(a[i]>x)	cnt += ceil( (double)(a[i]-x) / d[i]); }return cnt>=m;
}signed main(){ios::sync_with_stdio(false);cin.tie(nullptr);cout.tie(nullptr);int l=0,r=1e6;cin>>n>>m;for(int i=1;i<=n;i++){cin>>a[i]>>d[i];}while(l<r){int mid=l+r+1>>1;if(check(mid)) l=mid;else r=mid-1;}int cnt=0;for(int i=1;i<=n;i++){if(a[i]>r){int now=ceil((double)(a[i]-r)/d[i]);cnt+=now;ans += sum(a[i],now,d[i]);}}cout<<ans+(m-cnt)*r;return 0;	
}

感谢您的观看。

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

任务调度新境界:探秘ScheduledExecutorService的异步魔力

欢迎来到我的博客&#xff0c;代码的世界里&#xff0c;每一行都是一个故事 任务调度新境界&#xff1a;探秘ScheduledExecutorService的异步魔力 前言ScheduledExecutorService的基本概念基本概念&#xff1a;为何它是 Java 中任务调度的首选工具&#xff1a;基本用法&#xf…

MySQL-查询SQL语句的执行过程:连接器->查询缓存(8就没了)->分析器->优化器->执行器->返回结果

MySQL-查询SQL语句的执行过程&#xff1a;连接器->查询缓存<8就没了>->分析器->优化器->执行器->返回结果 查询SQL语句的执行过程1、主要步骤2、实用案例 查询SQL语句的执行过程 1、主要步骤 在MySQL中&#xff0c;一条查询SQL语句的执行过程非常复杂且…

day14_异常

今日内容 零、 复习昨日 一、日期类 二、异常 零、 复习昨日 1为什么要重写toString Object类toString返回的是对象名字地址,无意义子类重写toString() 返回的对象属性内容 2为什么要重写equals Object类equals判断是对象的地址值是否相等,无意义子类重写equals,为了判断对象的…

如何写一份简单的产品说明书,教程奉上

如果你是一位新晋产品经理&#xff0c;或者正在研发新产品&#xff0c;并且心中惴惴不安因为未知的产品说明书制作环节&#xff0c;那么今天你就来对地方了。本篇文章将教你如何创建一份简单明了的产品说明书。让我们开始吧&#xff01; 首先&#xff0c;明确产品说明书的目标。…

Leetcoder Day42| 动态规划part09 打家劫舍问题

198.打家劫舍 你是一个专业的小偷&#xff0c;计划偷窃沿街的房屋。每间房内都藏有一定的现金&#xff0c;影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统&#xff0c;如果两间相邻的房屋在同一晚上被小偷闯入&#xff0c;系统会自动报警。 给定一个代表每个房…

从新手到大师:顶级PPT学习网站推荐,让你的幻灯片脱颖而出!

介绍&#xff1a;PowerPoint&#xff0c;简称PPT&#xff0c;是微软公司开发的一款演示程序&#xff0c;也是Microsoft Office套件的重要组成部分之一。 PPT允许用户通过幻灯片的形式创建和展示信息&#xff0c;这些幻灯片可以包含文本、图形、图表、视频、音频等多种元素。用户…

Linux-网络-011

1网络协议模型 1.1【OSI】协议模型 1.1.1应用层 实际发送的数据应用层:HTTP 超文本传输协议HTTPS FTP 文件传输协议TFTP 简单文本传输协议SMTP 邮件传输协议MQTT TELNET ..1.1.2表示层 发送的数据是否加密1.1.3会话层 是否建立会话连接1.1.4传输层 数据…

解决 Chrome 无法代理外网,但Safari等其他软件可以

文章目录 1. 问题2. 解决方法 1. 问题 开启代理之后&#xff0c;Chrome 浏览器仍然无法访问外网。但是用 Mac 的 Safari 浏览器或者其他的软件已经可以访问外网。 2. 解决方法 发现 Chrome 浏览器的某些拓展程序会影响代理&#xff0c;可以关闭 Chrome 的拓展程序试试&#…

二维码门楼牌管理系统应用场景:数据管理的智慧新选择

文章目录 前言一、数据管理部门的智慧工具二、助力决策制定与优质服务提供三、二维码门楼牌管理系统的优势四、展望未来 前言 随着科技的飞速发展&#xff0c;二维码门楼牌管理系统正逐渐成为城市管理的智慧新选择。该系统不仅提升了数据管理效率&#xff0c;还为政府和企业提…

软件系统开发安全指南-word

应用系统设计安全主要涵盖以下几点&#xff1a; 1、应用系统架构安全设计 2、应用系统软件功能安全设计 3、应用系统存储安全设计 4、应用系统通讯安全设计 5、应用系统数据库安全设计 应用系统测试安全包含&#xff1a; 1、测试前置要求 2、测试方法及测试内容 3、测试环境及人…

idea Gradle 控制台中文乱码

如下图所示&#xff0c;idea 中的 Gradle 控制台中文乱码&#xff1a; 解决方法&#xff0c;如下图所示&#xff1a; 注意&#xff1a;如果你的 idea 使用 crack 等方式破解了&#xff0c;那么你可能需要在文件 crack-2023\jetbra\vmoptions\idea.vmoptions 中进行配置&#xf…

Linux 开发工具 yum、git、gdb

目录 一、yum 1、软件包 2、rzsz 3、注意事项 4、查看软件包 5、安装软件 6、卸载软件 二、git操作 1、克隆三板斧 2、第一次使用会出现以下情况&#xff1a; 未配置用户名和邮箱&#xff1a; push后弹出提示 三、gdb使用 1、背景 2、使用方法 例一&#xff1a…