2022年ACM算法题以及答案 .pdf
《2022年ACM算法题以及答案 .pdf》由会员分享,可在线阅读,更多相关《2022年ACM算法题以及答案 .pdf(30页珍藏版)》请在淘文阁 - 分享文档赚钱的网站上搜索。
1、学而不思则惘,思而不学则殆ACM 算法题使用 C+实现在做这些题目之前必须了解vector(数组),list(链表)、deque(双端队列)、queue(队列) , priority_queue (优先队列)Stack(栈) 、set(集合)、map(键值对),mutilset 、mutilmap 。stack 堆栈,没有迭代器,支持push()方法。后进先出,top()返回最顶端的元素,pop()剔除最顶元素deque 双端队列,支持迭代器,有push_back()方法,跟vector 差不多,比vector 多了个pop_front,push_front 方法queue 队列,先进先出,不
2、支持迭代器,有push()方法, pop()剔除第一个元素,front() 返回第一个元素vector 使用vector 是 C+标准模板库中的部分内容,它是一个多功能的,能够操作多种数据结构和算法的模板类和函数库。vector 之所以被认为是一个容器,是因为它能够像容器一样存放各种类型的对象, 简单地说 vector 是一个能够存放任意类型的动态数组,能够增加和压缩数据。为了可以使用vector,必须在你的头文件中包含下面的代码:#include vector 属于 std 命名域的,因此需要通过命名限定,如下完成你的代码:using std:vector; vector v; 或者连在一起
3、,使用全名:std:vector v; 建议使用全局的命名域方式:using namespace std; 1.vector 的声明vector c; 创建一个空的vector vector c1(c2); 创建一个vector c1,并用 c2 去初始化c1 vector c(n) ; 创建一个含有n 个 ElemType 类型数据的vector; vector c(n,elem); 创建一个含有n 个 ElemType 类型数据的vector,并全部初始化为 elem; c.vector(); 销毁所有数据,释放资源 ; 2.vector 容器中常用的函数。(c 为一个容器对象)c.pus
4、h_back(elem); 在容器最后位置添加一个元素elem c.pop_back(); 删除容器最后位置处的元素c.at(index); 返回指定 index 位置处的元素c.begin(); 返回指向容器最开始位置数据的指针c.end(); 返回指向容器最后一个数据单元的指针+1 c.front(); 返回容器最开始单元数据的引用c.back(); 返回容器最后一个数据的引用c.max_size(); 返回容器的最大容量c.size(); 返回当前容器中实际存放元素的个数c.capacity(); 同 c.size() c.resize(); 重新设置vector 的容量c.reserv
5、e(); 同 c.resize() c.erase(p); 删除指针p 指向位置的数据,返回下指向下一个数据位置的指针(迭名师归纳总结 精品学习资料 - - - - - - - - - - - - - - -精心整理归纳 精选学习资料 - - - - - - - - - - - - - - - 第 1 页,共 30 页 - - - - - - - - - 学而不思则惘,思而不学则殆代器)c.erase(begin,end) 删除 begin,end 区间的数据,返回指向下一个数据位置的指针(迭代器)c.clear(); 清除所有数据c.rbegin(); 将 vector 反转后的开始指针返回
6、(其实就是原来的end-1) c.rend(); 将 vector 反转后的结束指针返回(其实就是原来的begin-1) c.empty(); 判断容器是否为空,若为空返回true,否则返回false c1.swap(c2); 交换两个容器中的数据c.insert(p,elem); 在指针 p 指向的位置插入数据elem,返回指向elem 位置的指针c.insert(p,n,elem); 在位置 p 插入 n 个 elem 数据,无返回值c.insert(p,begin,end) 在位置 p 插入在区间 begin,end)的数据,无返回值3.vector 中的操作operator 如:c.i
7、; 同 at()函数的作用相同,即取容器中的数据。list 使用:STL中的 list 就是一双向链表,可高效地进行插入删除元素。list不支持随机访问。所以没有 at(pos)和 operator。list对象 list1, list2 分别有元素 list1(1,2,3),list2(4,5,6) 。list:iterator it;list成员说明constructor 构造函数destructor 析构函数operator= 赋值重载运算符assign 分配值front 返回第一个元素的引用back 返回最后一元素的引用begin 返回第一个元素的指针(iterator) end 返回
8、最后一个元素的下一位置的指针rbegin 返回链表最后一元素的后向指针(reverse_iterator or const) rend 返回链表第一元素的下一位置的后向指针push_back 增加一元素到链表尾push_front 增加一元素到链表头pop_back pop_back()删除链表尾的一个元素pop_front 删除链表头的一元素clear 删除所有元素erase 删除一个元素或一个区域的元素(两个重载 ) 名师归纳总结 精品学习资料 - - - - - - - - - - - - - - -精心整理归纳 精选学习资料 - - - - - - - - - - - - - - -
9、第 2 页,共 30 页 - - - - - - - - - 学而不思则惘,思而不学则殆remove 删除链表中匹配值的元素( 匹配元素全部删除) remove_if 删除条件满足的元素( 遍历一次链表 ) ,参数为自定义的回调函数empty 判断是否链表为空max_size 返回链表最大可能长度size 返回链表中元素个数resize 重新定义链表长度(两重载函数 ) reverse 反转链表sort 对链表排序,默认升序merge 合并两个有序链表并使之有序splice 对两个链表进行结合( 三个重载函数 ) 结合后第二个链表清空insert 在指定位置插入一个或多个元素(三个重载函数)
10、swap 交换两个链表(两个重载 ) unique 删除相邻重复元素Deque 双端队列使用:容器 deque 和 vector 非常相似, 操作函数基本一致。它采用动态数组来管理元素,提供随机存取, 可以在头尾两端进行快速安插和删除元素操作。特别要注意,除了头尾两端,在任何地方安插与删除元素,都将导致指向deque元素的任何pointers references iterators 失效。包括的头文件为:#include using namespace std; 声明一个deque时,一般需要前缀std: ,如 std:deque c; 因为类型deque是一个定义在namespace st
11、d内的 template 。构造函数:deque c ; /产生一个空的deque,其中没有任何元素deque c1(c2); / 产生另一个同型deque的副本(所有元素都被拷贝)deque c(n) ; /产生一个大小为n的 deque deque c(n , elem) ; /产生一个大小为n 的 deque,/每个元素值都是elem。dequer c(begin,end); / 产生一个deque,以区间 begin ; end /做为元素初值析构函数:c. deque() ;销毁所有元素,并释放内存。非变动性操作c.size(); /返回当前的元素数量c.empty(); /判断大小
12、是否为零。等同于c.size() = 0,但可能更快名师归纳总结 精品学习资料 - - - - - - - - - - - - - - -精心整理归纳 精选学习资料 - - - - - - - - - - - - - - - 第 3 页,共 30 页 - - - - - - - - - 学而不思则惘,思而不学则殆c.max_size(); /可容纳元素的最大数量c.at(idx) ; /返回索引为idx 所标示的元素。如果idx 越界,抛出out_of_range cidx ; /返回索引idx 所标示的元素。不进行范围检查c.front() ; /返回第一个元素,不检查元素是否存在c.bac
13、k(); /返回最后一个元素c.begin(); /返回一个随机迭代器,指向第一个元素c.end(); /返回一个随机迭代器,指向最后元素的下一位置变动性操作:c1 = c2 ; /将 c2 的所有元素赋值给c1; c.assign(n , elem); /将 n 个 elem 副本赋值给c c.assing(beg , end); /将区间 beg;end中的元素赋值给c; c.push_back(elem); /在尾部添加元素elem c.pop_back() ; /移除最后一个元素(但不回传)c.push_front() ; /在头部添加元素elem c.pop_front() ; /移
14、除头部一个元素(但不回传)c.erase(pos) ; /移除 pos 位置上的元素,返回一元素位置/如 c.erase( c.begin() + 5) /移除第五个元素c.insert(pos , elem); /在 pos 位置插入一个元素elem,并返回新元素的位置c.insert(pos , n , elem); / 在 pos 位置插入n 个元素 elem,无返回值c.insert(pos , beg , end); c.resize(num); /将容器大小改为num。可更大或更小。c.resize(num , elem); /将容器大小改为num,新增元素都为elem c.cle
15、ar(); /移除所有元素,将容器清空PS:Deque和 Vector 是智能容器, 删除或者增加元素时,其他位置与元素会进行相应的移动。queue的使用queue 模板类的定义在头文件中。与 stack 模板类很相似,queue 模板类也需要两个模板参数,一个是元素类型,一个容器类型,元素类型是必要的,容器类型是可选的,默认为deque 类型。定义 queue 对象的示例代码如下:queue q1; queue q2; queue 的基本操作有:入队,如例: q.push(x); 将 x 接到队列的末端。出队,如例: q.pop(); 弹出队列的第一个元素,注意,并不会返回被弹出元素的值。访
16、问队首元素,如例:q.front() ,即最早被压入队列的元素。访问队尾元素,如例:q.back(),即最后被压入队列的元素。判断队列空,如例:q.empty(),当队列空时,返回true。访问队列中的元素个数,如例:q.size() 名师归纳总结 精品学习资料 - - - - - - - - - - - - - - -精心整理归纳 精选学习资料 - - - - - - - - - - - - - - - 第 4 页,共 30 页 - - - - - - - - - 学而不思则惘,思而不学则殆priority_queue 的使用在头文件中,还定义了另一个非常有用的模板类priority_que
17、ue( 优先队列)。优先队列与队列的差别在于优先队列不是按照入队的顺序出队,而是按照队列中元素的优先权顺序出队(默认为大者优先,也可以通过指定算子来指定自己的优先顺序)。priority_queue 模板类有三个模板参数,第一个是元素类型,第二个容器类型,第三个是比较算子。其中后两个都可以省略,默认容器为vector,默认算子为less,即小的往前排,大的往后排(出队时序列尾的元素出队)。定义 priority_queue 对象的示例代码如下:priority_queue q1; priority_queue pair q2; / 注意在两个尖括号之间一定要留空格。priority_queue
18、int, vector, greater q3; / 定义小的先出队priority_queue 的基本操作与queue 相同。初学者在使用priority_queue 时,最困难的可能就是如何定义比较算子了。如果是基本数据类型,或已定义了比较运算符的类,可以直接用STL 的 less 算子和 greater 算子默认为使用less 算子,即小的往前排,大的先出队。如果要定义自己的比较算子,方法有多种,这里介绍其中的一种:重载比较运算符。优先队列试图将两个元素x 和 y 代入比较运算符(对 less 算子,调用xy),若结果为真,则x 排在 y 前面, y 将先于 x 出队,反之,则将y 排在
19、 x 前面, x 将先出队。看下面这个简单的示例:#include #include using namespace std; class T public: int x, y, z; T(int a, int b, int c):x(a), y(b), z(c) ; bool operator (const T &t1, const T &t2) return t1.z t2.z; / 按照 z 的顺序来决定t1 和 t2 的顺序 main() priority_queue q; q.push(T(4,4,3); q.push(T(2,2,5); q.push(T(1,5,4); 名师归纳总
20、结 精品学习资料 - - - - - - - - - - - - - - -精心整理归纳 精选学习资料 - - - - - - - - - - - - - - - 第 5 页,共 30 页 - - - - - - - - - 学而不思则惘,思而不学则殆q.push(T(3,3,6); while (!q.empty() T t = q.top(); q.pop(); cout t.x t.y t.z endl; return 1; 输出结果为 (注意是按照z 的顺序从大到小出队的):3 3 6 2 2 5 1 5 4 4 4 3 Statck 使用statck 栈中的数据是先进后出的(Firs
21、t In Last Out, FILO ) 。栈只有一个出口,允许新增元素(只能在栈顶上增加) 、移出元素(只能移出栈顶元素)、取得栈顶元素等操作。set 和 multiset的功能和所有关联式容器类似,通常使用平衡二叉树完成。事实上,set 和 multiset 通常以红黑树实作而成。自动排序的优点是使得搜寻元素时具有良好的性能,具有对数时间复杂度。但是造成的一个缺点就是:名师归纳总结 精品学习资料 - - - - - - - - - - - - - - -精心整理归纳 精选学习资料 - - - - - - - - - - - - - - - 第 6 页,共 30 页 - - - - - -
22、 - - - 学而不思则惘,思而不学则殆不能直接改变元素值。因为这样会打乱原有的顺序。改变元素值的方法是:先删除旧元素,再插入新元素。存取元素只能通过迭代器,从迭代器的角度看,元素值是常数。三、操作函数构造函数和析构函数set 的形式可以是:有两种方式可以定义排序准则:1、以 template参数定义:cppview plaincopyprint?1.set int ,greater col1; 此时,排序准则就是型别的一部分。型别系统确保只有排序准则相同的容器才能被合并。程序实例:cppview plaincopyprint?1.#include 2.#include 3.usingname
23、space std; 名师归纳总结 精品学习资料 - - - - - - - - - - - - - - -精心整理归纳 精选学习资料 - - - - - - - - - - - - - - - 第 7 页,共 30 页 - - - - - - - - - 学而不思则惘,思而不学则殆4.5.int main() 6. 7. set s1; 8. setint ,greater s2; 9.10.for ( int i = 1;i 6;+i) 11. 12. s1.insert(i); 13. s2.insert(i); 14. 15.if (s1 = s2) 16. cout c1 equal
24、s c2 ! endl; 17.else18. cout c1 not equals c2 ! endl; 19. 程序运行会报错。但是如果把s1 的排序准则也指定为greater 便运行成功。2、以构造函数参数定义。这种情况下, 同一个型别可以运用不同的排序准则,而排序准则的初始值或状态也可以不同。如果执行期才获得排序准则,而且需要用到不同的排序准则,这种方式可以派上用场。程序实例:cppview plaincopyprint?1.#include 2.#include print.hpp3.#include 4.usingnamespace std; 5.6.template 7.clas
25、s RuntimeCmp 8.public: 9.enum cmp_modenormal,reverse; 10.private: 11. cmp_mode mode; 12.public: 13. RuntimeCmp(cmp_mode m = normal):mode(m) 14.15.bool operator()(const T &t1,const T &t2) 16. 17.return mode = normal ? t1 t2 : t2 t1; 18. 名师归纳总结 精品学习资料 - - - - - - - - - - - - - - -精心整理归纳 精选学习资料 - - - -
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 2022年ACM算法题以及答案 2022 ACM 算法 以及 答案
限制150内