C++ 的常见算法 之三

C++ 的常见算法 之三

  • 合并
    • merge
      • 使用实列
    • inplace_merge
      • 使用实列
    • set_difference
      • 使用实列
    • make_heap
      • 使用实列
    • sort_heap
      • 使用实列

合并

merge

将排序范围 [first1,last1) 和 [first2,last2) 中的元素合并到一个新范围中,该范围从 result 开始,所有元素均已排序。

  • 第一个版本使用operator< 来比较元素,
  • 第二个版本使用comp 来比较元素。两个范围中的元素应已根据相同的标准(operator< 或 comp)进行排序。结果范围也按此排序。

使用实列

#include <iostream> 
#include <algorithm>
#include <vector>using namespace std;struct personnel {string  name;int     age;
};void print_element(int val) { cout << val << " " ; }
void print_personnel(personnel person) { cout << person.name << " " << person.age << endl; }struct sort_class {bool operator() (personnel lf, personnel lr) { return (lf.age < lr.age);}
} sort_object;int main () {int first[] = {5,10,15,20,25};int second[] = {50,40,30,20,10};vector<int> v(10);vector<personnel> it_depart = {{"John", 30},{"Allison", 25},		  {"Sam", 29},		  {"Alice", 39},		  };vector<personnel> hr_depart = {{"Michael", 40},{"Tommy", 35},		  {"Eric", 49},		  {"Johnson", 59},		  };vector<personnel> all_employee(8);sort (first,first+5);sort (second,second+5);merge (first,first+5,second,second+5,v.begin());cout << "The resulting vector contains:";for_each(v.begin(), v.end(), [](int val)->void {cout << val << " ";}); cout << '\n';sort (it_depart.begin(), it_depart.end(), sort_object);cout << "After Sorted, IT depart:" << endl;for_each(it_depart.begin(), it_depart.end(), [](personnel emp)->void {cout << emp.name << " " << emp.age << endl;}); cout << '\n';sort (hr_depart.begin(), hr_depart.end(), sort_object);cout << "After Sorted, HR depart:" << endl;for_each(hr_depart.begin(), hr_depart.end(), [](personnel emp)->void {cout << emp.name << " " << emp.age << endl;}); cout << '\n';merge(it_depart.begin(), it_depart.end(), hr_depart.begin(), hr_depart.end(), all_employee.begin(), sort_object);cout << "After Merged, All employee:" << endl;cout << "All Employee:" << endl;for_each(all_employee.begin(), all_employee.end(), [](personnel emp)->void {cout << emp.name << " " << emp.age << endl;}); cout << '\n';return 0;
}

代码运行结果屏幕输出

The resulting vector contains:5 10 10 15 20 20 25 30 40 50
After Sorted, IT depart:
Allison 25
Sam 29
John 30
Alice 39After Sorted, HR depart:
Tommy 35
Michael 40
Eric 49
Johnson 59After Merged, All employee:
All Employee:
Allison 25
Sam 29
John 30
Tommy 35
Alice 39
Michael 40
Eric 49
Johnson 59

inplace_merge

合并两个连续的排序范围:[first,middle) 和 [middle,last),将结果放入合并的排序范围 [first,last) 中。

  • 第一个版本使用operator< 来比较元素,
template <class BidirectionalIterator>  
void inplace_merge (BidirectionalIterator first, BidirectionalIterator middle, BidirectionalIterator last);
  • 第二个版本使用comp 来比较元素。两个范围中的元素应已根据相同的标准(operator< 或 comp)进行排序。结果范围也按此排序。
template <class BidirectionalIterator, class Compare>  
void inplace_merge (BidirectionalIterator first, BidirectionalIterator middle,         BidirectionalIterator last, Compare comp);

该函数保留具有等效值的元素的相对顺序,第一个范围中的元素位于第二个范围中的等效元素之前。

使用实列

#include <iostream>
#include <algorithm>
#include <vector>using namespace std;struct personnel {string  name;int     age;
};void print_personnel(personnel person) 
{ cout << person.name << " " << person.age << endl; }struct sort_class {bool operator() (personnel lf, personnel lr) { return (lf.age < lr.age);}
} sort_object;int main () {int first[] = {5,10,15,20,25};int second[] = {50,40,30,2,1};vector<personnel> it_depart = {{"John", 30},{"Allison", 25},		  {"Sam", 29},		  {"Alice", 39},		  };vector<personnel> hr_depart = {{"Michael", 40},{"Tommy", 18},		  {"Eric", 19},		  {"Johnson", 59},		  };vector<personnel> all_employee(8);vector<personnel>::iterator it_emp;vector<int> v(10);vector<int>::iterator it;sort (first,first+5);sort (second,second+5);it=copy (first, first+5, v.begin());copy (second,second+5,it);cout << "Before merge:";for_each(v.begin(), v.end(), [](int val)->void{cout << val << " ";});cout << endl;inplace_merge (v.begin(),v.begin()+5,v.end());cout << " After merge:";for_each(v.begin(), v.end(), [](int val)->void{cout << val << " ";});cout << endl;sort (it_depart.begin(), it_depart.end(), sort_object);sort (hr_depart.begin(), hr_depart.end(), sort_object);it_emp=copy (it_depart.begin(), it_depart.end(), all_employee.begin());copy (hr_depart.begin(), hr_depart.end(), it_emp);cout << "Employee before merge:" << endl;for_each(all_employee.begin(), all_employee.end(), [](personnel emp)->void {cout << emp.name << " " << emp.age << endl;});cout << endl;inplace_merge (all_employee.begin(), all_employee.begin()+4, all_employee.end(), sort_object);cout << "Employee after merge:" << endl;for_each(all_employee.begin(), all_employee.end(), [](personnel emp)->void {cout << emp.name << " " << emp.age << endl;});cout << endl;return 0;
}

代码运行结果屏幕输出

Before merge:5 10 15 20 25 1 2 30 40 50After merge:1 2 5 10 15 20 25 30 40 50
Employee before merge:
Allison 25
Sam 29
John 30
Alice 39
Tommy 18
Eric 19
Michael 40
Johnson 59Employee after merge:
Tommy 18
Eric 19
Allison 25
Sam 29
John 30
Alice 39
Michael 40
Johnson 59

set_difference

构造一个排序范围,其元素是,排序范围 [first1,last1)和排序范围 [first2,last2)之间的差。

两个集合的差是由第一个集合中存在,但第二个集合中不存在的元素形成的。函数复制的元素始终来自第一个范围,且顺序相同。

对于支持值多次出现的容器,差异包括给定值在第一个范围中出现的次数减去第二个范围中匹配元素的数量(保留顺序)。

  • 第一个版本使用operator< 来比较元素
  • 第二个版本使用comp 来比较元素。如果 (!(a<b) && !(b<a)) 或 if (!comp(a,b) && !comp(b,a)),则 a 和 b 两个元素被视为等效。

范围中的元素应已根据相同的标准(operator< 或 comp)进行排序。结果范围也按此排序。

使用实列

#include <iostream>
#include <algorithm>
#include <vector>using namespace std;struct personnel {string  name;int     age;
};void print_personnel(personnel person) { cout << person.name << " " << person.age << endl; }struct sort_class {bool operator() (personnel lf, personnel lr) { return (lf.age < lr.age);}
} sort_object;struct equal_class {bool operator() (personnel lf, personnel lr) { return ((lf.name != lr.name) || (lf.age != lr.age));}
} equal_object;int main () {vector<int> first = {15,10,50,20,25,2};vector<int> second = {50,40,30,20,10};vector<personnel> it_depart = {{"John", 30},{"Allison", 25},		  {"Sam", 29},		  {"Alice", 30},		  };vector<personnel> all_employee = {{"John", 30},{"Allison", 25},		  {"Sam", 29},		  {"Alice", 39},		  {"Tommy", 35},		  {"Eric", 49},		  {"Johnson", 59},		  };vector<personnel> other_employee(all_employee.size());vector<int> v(11); vector<int>::iterator it;vector<personnel>::iterator it_emp;sort (first.begin(), first.end()); sort (second.begin(), second.end());it=set_difference (first.begin(), first.end(),second.begin(), second.end(), v.begin());v.resize(it-v.begin());cout << "The difference has " << (v.size()) << " elements:\n";for_each(v.begin(), v.end(), [](int val)->void {cout << val << " ";});cout << endl;sort (it_depart.begin(), it_depart.end(), sort_object); sort (all_employee.begin(), all_employee.end(), sort_object);it_emp=set_difference (all_employee.begin(), all_employee.end(),it_depart.begin(), it_depart.end(), other_employee.begin(), equal_object);other_employee.resize(it_emp - other_employee.begin());cout << "Other employee:" << endl;for_each(other_employee.begin(), other_employee.end(), [](personnel emp)->void {cout << emp.name << " " << emp.age << endl;});cout << endl;return 0;
}

代码运行结果屏幕输出

The difference has 3 elements:
2 15 25
Other employee:
Tommy 35
Alice 39
Eric 49
Johnson 59

make_heap

重新排列 [first,last) 范围内的元素,使它们形成堆。

堆是一种组织元素序列的方法,它允许在任何时刻,重复地快速检索具有最高值的元素(使用 pop_heap),同时允许快速插入新元素(使用 push_heap)。

具有最高值的元素始终是第一个元素。其他元素的顺序取决于特定的实现,但所有与堆相关的函数都是一致的。

  • 使用operator<(对于第一个版本)或
  • comp(对于第二个版本)来比较元素:具有最高值的元素是与范围中的每个其他元素进行比较时将返回 false 的元素。

标准容器适配器priority_queue自动调用make_heap、push_heap和pop_heap来维护容器的堆属性。

使用实列

#include <iostream>
#include <algorithm>
#include <vector>using namespace std;int main () {vector<int> v = {10,20,30,5,15};make_heap (v.begin(),v.end());cout << "Initial heap: ";for_each(v.begin(),v.end(), [](int val)->void {cout << val << " ";});cout << endl;cout << "initial max heap   : " << v.front() << '\n';pop_heap(v.begin(),v.end()); cout << "after pop_head heap: ";for_each(v.begin(),v.end(), [](int val)->void {cout << val << " ";});cout << endl;v.pop_back();cout << "after pop_back vector: ";for_each(v.begin(),v.end(), [](int val)->void {cout << val << " ";});cout << endl;cout << "max heap after pop : " << v.front() << '\n';v.push_back(99); cout << "after push_back vector: ";for_each(v.begin(),v.end(), [](int val)->void {cout << val << " ";});cout << endl;push_heap (v.begin(),v.end());cout << "after push_head heap: ";for_each(v.begin(),v.end(), [](int val)->void {cout << val << " ";});cout << endl;cout << "max heap after push: " << v.front() << '\n';return 0;
}

代码运行结果屏幕输出:

Initial heap: 30 20 10 5 15
initial max heap   : 30
after pop_head heap: 20 15 10 5 30
after pop_back vector: 20 15 10 5
max heap after pop : 20
after push_back vector: 20 15 10 5 99
after push_head heap: 99 20 10 5 15
max heap after push: 99

sort_heap

将堆范围 [first,last) 中的元素按升序或降序排序。

  • 第一个版本使用operator<来比较元素,
  • 第二个版本使用comp来比较元素,这应该与构造堆时使用的相同。

该序列失去了其作为堆的属性。

使用实列

#include <iostream> 
#include <algorithm>
#include <vector>using namespace std;struct personnel {string  name;int     age;
};struct sort_class {bool operator() (personnel lf, personnel lr) {return ((lf.name < lr.name) || ((lf.name == lr.name) && (lf.age < lr.age)));}
} sort_object;int main () {vector<int> v = {10,20,30,5,15};vector<personnel> depart = {{"John", 30},{"Allison", 25},		  {"Sam", 29},		  {"Sam", 49},		  {"Alice", 39},		  };personnel new_incomer = {"Alice", 29};make_heap (v.begin(),v.end());cout << "initial max heap   : " << v.front() << '\n';pop_heap (v.begin(),v.end());v.pop_back();cout << "max heap after pop : " << v.front() << '\n';v.push_back(99);push_heap (v.begin(),v.end());cout << "max heap after push: " << v.front() << '\n';sort_heap (v.begin(),v.end());cout << "final sorted range :";for_each(v.begin(),v.end(), [](int val)->void {cout << val << " ";});;cout << endl;//make_heap (depart.begin(), depart.end(), sort_object);cout << "After make_heap:" << endl;cout << "initial max heap   : " << depart.front().name << " " << depart.front().age << '\n';for_each(depart.begin(), depart.end(), [](personnel emp)->void {cout << " [ " << emp.name << " " << emp.age << " ] ";}); cout << '\n';depart.push_back(new_incomer);cout << "After push_back:" << endl;for_each(depart.begin(), depart.end(), [](personnel emp)->void {cout << " [ " << emp.name << " " << emp.age << " ] ";}); cout << '\n';push_heap (depart.begin(), depart.end(), sort_object);cout << "After make_heap:" << endl;cout << "max heap after push: " << depart.front().name << " " << depart.front().age << '\n';for_each(depart.begin(), depart.end(), [](personnel emp)->void {cout << " [ " << emp.name << " " << emp.age << " ] ";}); cout << '\n';sort_heap (depart.begin(), depart.end(), sort_object);cout << "final sorted range :" << endl;for_each(depart.begin(), depart.end(), [](personnel emp)->void {cout << " [ " << emp.name << "," << emp.age << " ] ";});cout << endl;return 0;
}

代码运行结果屏幕输出:

initial max heap   : 30
max heap after pop : 20
max heap after push: 99
final sorted range :5 10 15 20 99
After make_heap:
initial max heap   : Sam 49[ Sam 49 ]  [ John 30 ]  [ Sam 29 ]  [ Allison 25 ]  [ Alice 39 ]
After push_back:[ Sam 49 ]  [ John 30 ]  [ Sam 29 ]  [ Allison 25 ]  [ Alice 39 ]  [ Alice 29 ]
After make_heap:
max heap after push: Sam 49[ Sam 49 ]  [ John 30 ]  [ Sam 29 ]  [ Allison 25 ]  [ Alice 39 ]  [ Alice 29 ]
final sorted range :[ Alice,29 ]  [ Alice,39 ]  [ Allison,25 ]  [ John,30 ]  [ Sam,29 ]  [ Sam,49 ]

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

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

相关文章

【常用知识点-Java】创建文件夹

Author&#xff1a;赵志乾 Date&#xff1a;2024-07-04 Declaration&#xff1a;All Right Reserved&#xff01;&#xff01;&#xff01; 1. 简介 java.io.File提供了mkdir()和mkdirs()方法创建文件夹&#xff0c;两者区别&#xff1a;mkdir()仅创建单层文件夹&#xff0c;如…

编程五大原则实现思路介绍

在软件工程中&#xff0c;存在一些被广泛认可的设计原则&#xff0c;它们被称为编程的五大原则&#xff0c;也常被简称为SOLID原则。这些原则旨在促进更好的软件设计&#xff0c;提高可维护性和可扩展性。以下是每个原则的概述以及它们的实现思路&#xff1a; 1. 单一职责原则…

Linux 命令总结

今天&#xff0c;给小伙伴们带来一篇 Linux 命令总结的非常全的文章&#xff0c;也是我们平时工作中使用率非常高的操作命令&#xff0c;命令有点多&#xff0c;建议小伙伴们可以先收藏后阅读。 1.基本命令 uname -m 显示机器的处理器架构 uname -r 显示正在使用的内核版本 dmi…

入门PHP就来我这(纯干货)08

~~~~ 有胆量你就来跟着路老师卷起来&#xff01; -- 纯干货&#xff0c;技术知识分享 ~~~~ 路老师给大家分享PHP语言的知识了&#xff0c;旨在想让大家入门PHP&#xff0c;并深入了解PHP语言。 1 PHP对象的高级应用 1.1 final关键字 final 最终的、最后的。被final修饰过的类…

采用B/S模式 可跨平台使用的数据采集监控平台!

数据采集监控平台是一款专注于工业企业生产设备管理、数据采集、数据分析、数据管理、数据存储、数据传输等的软件系统。系统具备丰富的接口&#xff0c;配置灵活&#xff0c;方便部署&#xff0c;通过采集企业生产设备的数据集中处理&#xff0c;将各个信息孤岛有机连接&#…

技术赋能政务服务:VR导视与AI客服在政务大厅的创新应用

在数字化转型的浪潮中&#xff0c;政务大厅作为服务民众的前沿阵地&#xff0c;其服务效率和质量直接影响着政府形象和民众满意度。然而&#xff0c;许多政务大厅仍面临着缺乏智能化导航系统的挑战&#xff0c;这不仅增加了群众的办事难度&#xff0c;也降低了服务效率。维小帮…

ArcEngine获取投影坐标,地理坐标,垂直坐标系统以及3或6分度带的代码

本代码是用于质检中获取图层中平面坐标系统是否采用“2000国家大地坐标系(CGCS2000)高程系统是否采用“1985国家高程基准”。然后封装的一个方法 /// <summary>/// 平面坐标系统是否采用“2000国家大地坐标系(CGCS2000)高程系统是否采用“1985国家高程基准”。/// &l…

Windows的磁盘管理

&#x1f4d1;打牌 &#xff1a; da pai ge的个人主页 &#x1f324;️个人专栏 &#xff1a; da pai ge的博客专栏 ☁️宝剑锋从磨砺出&#xff0c;梅花香自苦寒来 第一动态磁盘管理 问题&#xff1a;…

C语言 求第一天摘的桃子数

猴子第一天摘下若干个桃子&#xff0c;当即吃了一半&#xff0c;还不过瘾&#xff0c;又吃了一个&#xff0c;第二天早上又将剩下的桃子吃掉一半又多吃了一个&#xff0c;以后每天早上都吃了前一天剩下的一半零一个。到10天早上再想吃时就只剩一个桃子了&#xff0c;求第一天共…

Jackson与Json、Json和各种Java数据类型的互相转化

jackson是什么 json是最常用的数据交换格式 Jackson是最流行的Json库 首先对于这种JSON序列化的库其实有非常多&#xff0c;比如我们熟悉的Gson&#xff0c;Fastjson等等&#xff0c;当然技术没有完全的好坏&#xff0c;但是从使用情况和社区生态等方面综合看来&#xff0c;Ja…

【讲解下AI Native应用中的模型微调】

&#x1f308;个人主页: 程序员不想敲代码啊 &#x1f3c6;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f44d;点赞⭐评论⭐收藏 &#x1f91d;希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff0c;让我们共…

Jenkins 任务中的 java.lang.InterruptedException 异常解析与解决

个人名片 &#x1f393;作者简介&#xff1a;java领域优质创作者 &#x1f310;个人主页&#xff1a;码农阿豪 &#x1f4de;工作室&#xff1a;新空间代码工作室&#xff08;提供各种软件服务&#xff09; &#x1f48c;个人邮箱&#xff1a;[2435024119qq.com] &#x1f4f1…

Symfony中的MVC架构实现:构建高效Web应用的基石

Symfony中的MVC架构实现&#xff1a;构建高效Web应用的基石 Symfony是一个高度灵活的PHP框架&#xff0c;用于构建Web应用和API。它遵循模型-视图-控制器&#xff08;Model-View-Controller&#xff0c;简称MVC&#xff09;设计模式&#xff0c;将应用分为三个核心组件&#x…

深入探索:scikit-learn中递归特征消除(RFE)的奥秘

深入探索&#xff1a;scikit-learn中递归特征消除(RFE)的奥秘 在机器学习的世界里&#xff0c;特征选择是一项至关重要的任务。它不仅能够提高模型的性能&#xff0c;还能减少模型的复杂度&#xff0c;避免过拟合。scikit-learn&#xff0c;作为Python中一个广泛使用的机器学习…

列表渲染 v-for

列表渲染v-for 使用v-for指令基于数组渲染一个列表&#xff0c;v-for指令的值需要使用item in/of items形式的特殊语法&#xff0c;其中items是源数据的数组&#xff0c;而item是迭代的别名。 代码实例&#xff1a; <template> <div><p v-for"item in na…

Python实战,怎么删除垃圾文件

注意:本文的下载教程,与以下文章的思路有相同点,也有不同点,最终目标只是让读者从多维度去熟练掌握本知识点。 下载教程: Python项目开发实战_删除垃圾文件_编程案例解析实例详解课程教程.pdf 在处理删除垃圾文件的Python实战任务时,我们需要首先明确“垃圾文件”的定义。…

SolidityFoundry 安全审计测试 绕过isContract()校验

名称&#xff1a; 绕过isContract()校验 https://github.com/XuHugo/solidityproject/tree/master/vulnerable-defi 描述&#xff1a; 出于安全原因&#xff0c;某些智能合约方法被定义为只接受来自外部自有账户(EOA)的调用&#xff0c;而不接受来自其他智能合约的调用。 依…

LeetCode258-各位相加

# 题目 给定一个非负整数 num&#xff0c;反复将各个位上的数字相加&#xff0c;直到结果为一位数。返回这个结果。 示例 1: 输入: num 38 输出: 2 解释: 各位相加的过程为&#xff1a; 38 --> 3 8 --> 11 11 --> 1 1 --> 2 由于 2 是一位数&#xff0c;所以返…

邮件客户端程序

①&#xff1a;网页邮箱 优点&#xff1a;简单 缺点&#xff1a; 没有新邮件实时提醒&#xff0c; 邮件保存在对方服务器 ②&#xff1a;邮件客户端工具&#xff08;Out look&#xff09; 优点&#xff1a; 有新邮件实时提醒 邮件可以选择下载本地&#xff0c;还是保存服务器…

[C++][CMake][生成可执行文件][上]详细讲解

目录 0.准备工作1.添加CMakeLists.txt文件2.执行cmake命令3.变量定义4.指定使用的C标准5.指定输出路径 0.准备工作 add.c#include <stdio.h> #include "head.h"int add(int a, int b) {return ab; }sub.c#include <stdio.h> #include "head.h"…