【算法】昂贵的聘礼(dijkstra算法)

2023-11-05 03:28
文章标签 算法 dijkstra 聘礼 昂贵

本文主要是介绍【算法】昂贵的聘礼(dijkstra算法),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目

        年轻的探险家来到了一个印第安部落里。

        在那里他和酋长的女儿相爱了,于是便向酋长去求亲。

        酋长要他用 10000 个金币作为聘礼才答应把女儿嫁给他。

        探险家拿不出这么多金币,便请求酋长降低要求。

        酋长说:”嗯,如果你能够替我弄到大祭司的皮袄,我可以只要 8000 金币。如果你能够弄来他的水晶球,那么只要 5000 金币就行了。”

        探险家就跑到大祭司那里,向他要求皮袄或水晶球,大祭司要他用金币来换,或者替他弄来其他的东西,他可以降低价格。

        探险家于是又跑到其他地方,其他人也提出了类似的要求,或者直接用金币换,或者找到其他东西就可以降低价格。

        不过探险家没必要用多样东西去换一样东西,因为不会得到更低的价格。

        探险家现在很需要你的帮忙,让他用最少的金币娶到自己的心上人。

        另外他要告诉你的是,在这个部落里,等级观念十分森严。

        地位差距超过一定限制的两个人之间不会进行任何形式的直接接触,包括交易。

        他是一个外来人,所以可以不受这些限制。

        但是如果他和某个地位较低的人进行了交易,地位较高的的人不会再和他交易,他们认为这样等于是间接接触,反过来也一样。

        因此你需要在考虑所有的情况以后给他提供一个最好的方案。

        为了方便起见,我们把所有的物品从 1 开始进行编号,酋长的允诺也看作一个物品,并且编号总是 1。

每个物品都有对应的价格 P主人的地位等级 L以及一系列的替代品 Ti 该替代品所对应的”优惠” Vi

如果两人地位等级差距超过了 M,就不能”间接交易”。

你必须根据这些数据来计算出探险家最少需要多少金币才能娶到酋长的女儿。

输入格式

        输入第一行是两个整数 M,N,依次表示地位等级差距限制和物品的总数。

        接下来按照编号从小到大依次给出了 N 个物品的描述。

        每个物品的描述开头是三个非负整数 P、L、X,依次表示该物品的价格、主人的地位等级和替代品总数。

        接下来 X 行每行包括两个整数 T 和 V,分别表示替代品的编号和”优惠价格”。

输出格式

        输出最少需要的金币数。

数据范围

1 ≤ N ≤ 100
1 ≤ P ≤ 10000
1 ≤ L , M ≤ N
0 ≤ X < N

思路

我们可以根据以下样例绘制一张图:

样例:
1 4
10000 3 2
2 8000
3 5000
1000 2 1
4 200
3000 2 1
4 200
50 2 0

        由图可知,我们可以反向建图,从起始点出发到达所有点的距离为物品 i 的原价从点 i 到点 j 的距离为得到物品 i 之后物品 j 的价格。当我们建完图之后,很容易发现这个问题可以抽象成为一个最短路问题。

         对于阶级问题:for(int i = level[ 1 ] - m; i <= level[ 1 ]; i ++) res = min(res,dijkstra(i, i + m));     i 表示可以交换的下界,i + m表示可以交换的上界 ,循环m次即可得到最小的花费。

  

代码

#include<bits/stdc++.h>
using namespace std;
const int N = 110,INF = 0x3f3f3f3f;
int n,m;// n等级差距限制,m物品个数
int w[N][N],level[N];//level数组储存的是第i个物品的主人所处的阶级,w数组储存点i到点j的边权
int dist[N];// 表示当前买到第i件物品价格的最小值
bool st[N];// 表示当前物品的价格是否为最小值int dijkstra(int down,int up)
{memset(dist,0x3f,sizeof(dist));// 将价格初始化为正无穷memset(st,0,sizeof(st));// 所有物品都没有确定最小价格dist[0] = 0;// 将起始点价格初始化为0int i = n;while(i --)// 循环n次确定n个物品的最小价格{int t = -1;// 在没有找到下一个可以确定价格已经最低的物品编号之前,t = -1for(int j = 0; j <= n; j ++)// 本次循环确定价格最小的物品编号if(!st[j] && (t == -1 || dist[t] > dist[j]))t = j;st[t] = true;// 该物品已经确定为最小价格,标记一下for(int j = 1; j <= n; j ++)// 使用这个已经确定最小价格的点对其他点进行更新if(level[j] >= down && level[j] <= up)// 如果阶级不符合条件,则不进行更新dist[j] = min(dist[j],dist[t] + w[t][j]);}return dist[1];
}int main()
{cin >> m >> n;// m等级差距限制,n物品总数memset(w,0x3f,sizeof(w));// 将数组w初始化for(int i = 1;i <= n; i ++) w[i][i] = 0;for(int i = 1;i <= n; i ++){int price,cnt;// price表示物品的价格,cnt表示替代品的数量cin >> price >> level[i] >> cnt;// 依次输入物品的价值,物品主人的阶级,代替品的数量w[0][i] = min(price,w[0][i]);// 起始点到该物品的距离(物品原价)while(cnt --){int id,cost;cin >> id >> cost;// 输入替代品的数量与价格w[id][i] = min(w[id][i],cost);//保留边权最小的值}}int res = INF;// 将答案初始化为正无穷for(int i = level[1] - m; i <= level[1]; i ++) res = min(res,dijkstra(i, i + m));// i表示可以交换的下界,i + m表示可以交换的上界cout << res << endl;return 0;}

这篇关于【算法】昂贵的聘礼(dijkstra算法)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/347097

相关文章

代码随想录算法训练营:12/60

非科班学习算法day12 | LeetCode150:逆波兰表达式 ,Leetcode239: 滑动窗口最大值  目录 介绍 一、基础概念补充: 1.c++字符串转为数字 1. std::stoi, std::stol, std::stoll, std::stoul, std::stoull(最常用) 2. std::stringstream 3. std::atoi, std

人工智能机器学习算法总结神经网络算法(前向及反向传播)

1.定义,意义和优缺点 定义: 神经网络算法是一种模仿人类大脑神经元之间连接方式的机器学习算法。通过多层神经元的组合和激活函数的非线性转换,神经网络能够学习数据的特征和模式,实现对复杂数据的建模和预测。(我们可以借助人类的神经元模型来更好的帮助我们理解该算法的本质,不过这里需要说明的是,虽然名字是神经网络,并且结构等等也是借鉴了神经网络,但其原型以及算法本质上还和生物层面的神经网络运行原理存在

大林 PID 算法

Dahlin PID算法是一种用于控制和调节系统的比例积分延迟算法。以下是一个简单的C语言实现示例: #include <stdio.h>// DALIN PID 结构体定义typedef struct {float SetPoint; // 设定点float Proportion; // 比例float Integral; // 积分float Derivative; // 微分flo

LeetCode 算法:二叉树的中序遍历 c++

原题链接🔗:二叉树的中序遍历 难度:简单⭐️ 题目 给定一个二叉树的根节点 root ,返回 它的 中序 遍历 。 示例 1: 输入:root = [1,null,2,3] 输出:[1,3,2] 示例 2: 输入:root = [] 输出:[] 示例 3: 输入:root = [1] 输出:[1] 提示: 树中节点数目在范围 [0, 100] 内 -100 <= Node.

【Java算法】滑动窗口 下

​ ​    🔥个人主页: 中草药 🔥专栏:【算法工作坊】算法实战揭秘 🦌一.水果成篮 题目链接:904.水果成篮 ​ 算法原理 算法原理是使用“滑动窗口”(Sliding Window)策略,结合哈希表(Map)来高效地统计窗口内不同水果的种类数量。以下是详细分析: 初始化:创建一个空的哈希表 map 用来存储每种水果的数量,初始化左右指针 left

ROS2从入门到精通4-4:局部控制插件开发案例(以PID算法为例)

目录 0 专栏介绍1 控制插件编写模板1.1 构造控制插件类1.2 注册并导出插件1.3 编译与使用插件 2 基于PID的路径跟踪原理3 控制插件开发案例(PID算法)常见问题 0 专栏介绍 本专栏旨在通过对ROS2的系统学习,掌握ROS2底层基本分布式原理,并具有机器人建模和应用ROS2进行实际项目的开发和调试的工程能力。 🚀详情:《ROS2从入门到精通》 1 控制插

算法与数据结构面试宝典——回溯算法详解(C#,C++)

文章目录 1. 回溯算法的定义及应用场景2. 回溯算法的基本思想3. 递推关系式与回溯算法的建立4. 状态转移方法5. 边界条件与结束条件6. 算法的具体实现过程7. 回溯算法在C#,C++中的实际应用案例C#示例C++示例 8. 总结回溯算法的主要特点与应用价值 回溯算法是一种通过尝试各种可能的组合来找到所有解的算法。这种算法通常用于解决组合问题,如排列、组合、棋盘游

【图像识别系统】昆虫识别Python+卷积神经网络算法+人工智能+深度学习+机器学习+TensorFlow+ResNet50

一、介绍 昆虫识别系统,使用Python作为主要开发语言。通过TensorFlow搭建ResNet50卷积神经网络算法(CNN)模型。通过对10种常见的昆虫图片数据集(‘蜜蜂’, ‘甲虫’, ‘蝴蝶’, ‘蝉’, ‘蜻蜓’, ‘蚱蜢’, ‘蛾’, ‘蝎子’, ‘蜗牛’, ‘蜘蛛’)进行训练,得到一个识别精度较高的H5格式模型文件,然后使用Django搭建Web网页端可视化操作界面,实现用户上传一

【数据结构与算法 经典例题】使用队列实现栈(图文详解)

💓 博客主页:倔强的石头的CSDN主页               📝Gitee主页:倔强的石头的gitee主页    ⏩ 文章专栏:《数据结构与算法 经典例题》C语言                                   期待您的关注 ​​ 目录  一、问题描述 二、前置知识 三、解题思路 四、C语言实现代码 🍃队列实现代码:

算法11—判断一个树是不是二叉查询树

问题: 给定一个二叉树,判断它是否是二叉查询树。 思路: 要判断是否是二叉查询树,标准就是看每一个节点是否满足:1、左节点及以下节点的值比它小;2、右节点及以下节点的值比它大。当然,前提是子节点都存在的情况。所以,我们需要从根节点不断向下递归,只要所有节点都满足,那么就是BST,否则,就不是。 代码: [java]  view plain copy pri