谈STL的重要应用与实现
1. priority_queue
无理由在广搜和解决贪心问题用处很大,首先作为广搜的挑选最优状态如Astar,然后是当做堆给最短路和最小生成树做优化,用的是贪心思想。
定义优先级,和正常的反过来。
bool operator < (const number1 &a)const {
return x > a.x; // 从小到大 ,x 小的 优先级别高
}
优先队列的实现就是一个堆,队尾进入一个新元素就和父节点比较上升,出对就调整堆。
上一道纯贪心和优先队列完美结合的例题 HDU 4544
#include <iostream> #include <cstdio> #include <queue> #include <algorithm> using namespace std; #define MAXN 100006 #define LL long long int Blood[MAXN]; bool cmp(int a, int b) { return a > b; } struct Node { int D, cost; bool operator <(const Node &tmp) const { return cost > tmp.cost; } } arrow[MAXN]; int N, M; priority_queue<Node> Q; bool cmp_Node(Node a, Node b) { return a.D > b.D; } LL solve() { LL Min_cost = 0; int pos = 0; for (int i = 0; i < N; i++) { while (pos < M && Blood[i] <= arrow[pos].D) Q.push(arrow[pos++]); if (Q.empty()) { return -1; } Min_cost += Q.top().cost; Q.pop(); } return Min_cost; } int main() { // freopen("data_tmp.txt", "r", stdin); int i; while (~scanf("%d%d", &N, &M)) { while (!Q.empty()) Q.pop(); for (i = 0; i < N; i++) scanf("%d", &Blood[i]); for (i = 0; i < M; i++) scanf("%d", &arrow[i].D); for (i = 0; i < M; i++) scanf("%d", &arrow[i].cost); sort(Blood, Blood + N, cmp); sort(arrow, arrow + M, cmp_Node); if (arrow[0].D < Blood[0]) { puts("No"); continue; } LL ans = solve(); if (ans == -1) puts("No"); else printf("%I64d\n", ans); } return 0; }
2. set 与 hash_set
应用,首先是集合操作,交并补对称差,和查找.count() ,指向键值>=k的第一个元素.lower_bound() ,指向键值>k的第一个元素.upper_bound(),it-- 就得到<=val 的最大值。
其次是对于动态插入删除有序的题,或者维护状态(最远曼哈顿距离),可以用set快速找到应该插入的位置,如insert和lower_bound 都会返回位置的迭代器。
上一道很有代表性的set应用 (hash_set用法和set一致,很多情况下hash_set效率更高,且hash表空间分配上使用了一个近似于倍增的素数表,最开始取第一个素数,当空间不足时就使用下一个素数)
内部实现为一颗红黑树。
//给出一些能引起周围一定区域内炸弹爆炸的炸弹,按给出顺序引爆 //输出每次爆炸会引起几个炸弹的爆炸 //unique()去重操作,返回去重后尾地址 //lower_bound(first,last,val)求first到last区间内大于等于val第一个值 #include<queue> #include<set> #include<algorithm> #include<iostream> #include<cmath> using namespace std; const int maxn = 100015; int myhash[maxn]; struct node { int x, y, d; } mine[maxn]; struct node2 { int y, id; node2(int a, int b) { y = a; id = b; } bool operator<(const node2 &tmp) const { return y < tmp.y; } }; multiset<node2> p[maxn]; multiset<node2>::iterator LL, RR, it; queue<int> q; bool vis[maxn]; int main() { int n; while (scanf("%d", &n) != EOF, n) { for (int i = 0; i < n; i++) { scanf("%d%d%d", &mine[i].x, &mine[i].y, &mine[i].d); myhash[i] = mine[i].x; //在 x 方向离散化 } sort(myhash, myhash + n); int cnt = unique(myhash, myhash + n) - myhash; for (int i = 0; i < cnt; i++) p[i].clear(); memset(vis, 0, sizeof(vis)); for (int i = 0; i < n; i++) { int pos = lower_bound(myhash, myhash + cnt, mine[i].x) - myhash; p[pos].insert(node2(mine[i].y, i)); // i 是原序号 } int id; scanf("%d", &id); while (!q.empty()) q.pop(); int ans = 0; q.push(id); vis[id] = 1; while (!q.empty()) { int now_id = q.front(); //点燃第一个 q.pop(); ans++; int first_pos = lower_bound(myhash, myhash + cnt, mine[now_id].x - mine[now_id].d) - myhash; int last_pos = upper_bound(myhash, myhash + cnt, mine[now_id].x + mine[now_id].d) - myhash; //首先由横坐标确定大致的 x 寻找范围 for (int pos = first_pos; pos < last_pos; pos++) { int dy = mine[now_id].d - abs(mine[now_id].x - myhash[pos]); //lower_bound 与upper_bound配合确定上下界 LL = p[pos].lower_bound(node2(mine[now_id].y - dy, 0)); RR = p[pos].upper_bound(node2(mine[now_id].y + dy, 0)); for (it = LL; it != RR; it++) { if (vis[it->id] == 0) { vis[it->id] = 1; q.push(it->id); } } p[pos].erase(LL, RR); //删除已经爆炸的点 区间删除 } } printf("%d\n", ans); } return 0; }
3. map
首先是映射,如规定10000000000对应1,而不能开那么大的bool数组吧,那就用map就可以。
其次它有自排序功能,所以对于处理字典序很有用。
map内部自建一颗红黑树(一种非严格意义上的平衡二叉树),这颗树具有对数据自动排序的功能,所以在map内部所有的数据都是有序的。
//给出一系列要忽略的单词,这些单词以外的单词都看作关键字。给出一些标题,按这些关键字的字典序给标题排序 #include <iostream> #include <cstdio> #include <map> #include <set> #include <cctype> #include <algorithm> using namespace std; const int maxn = 220; int main() { string k, st; int a = 0; set<string> o; multimap<string, string> r; while (cin >> k && k != "::") o.insert(k); getchar(); while (getline (cin, st)) { for (int i = 0; i < st.size(); i++) st[i] = tolower(st[i]); for (int i = 0; i < st.size(); i++) { if (!isalpha(st[i])) continue; string t; int rec = i; while (i < st.size() && isalpha(st[i])) t += st[i++]; if (!o.count(t)) { for (int j = 0; j < t.size(); j++) t[j] = toupper(t[j]); string temp = st; temp.replace(rec, t.size(), t); r.insert(make_pair(t, temp)); } } } for (multimap<string, string>::iterator i = r.begin(); i != r.end(); i++) cout << i -> second << endl; return 0; }
4.heap
头文件 #include <algorithm>
建立堆 make_heap(_First, _Last, _Comp)
默认是建立最大堆的。对int类型,可以在第三个参数传入greater<int>()得到最小堆。
在堆中添加数据 push_heap (_First, _Last) 要先在最后加入数据,再调用push_heap()在原堆上调整
在堆中删除数据
pop_heap(_First, _Last) 要先调用pop_heap()清除第一个并调整成堆,再删除数据
堆排序 sort_heap(_First, _Last) 稳定排序
#include <algorithm> using namespace std; int num[20]; void debug() { for (int i = 0; i < 11; i++) printf("%d ", num[i]); puts(""); } int main() { memset(num, -1, sizeof(num)); for (int i = 0; i < 10; i++) num[i] = rand() % 100; debug(); make_heap(num, num + 10); debug(); num[10] = 100; push_heap(num, num + 11); debug(); pop_heap(num, num + 11); num[10] = -1; debug(); return 0; }
执行结果
原数组 41 67 34 0 69 24 78 58 62 64 -1
建堆后 78 69 41 62 67 24 34 58 0 64 -1
添加100调整后 100 78 41 62 69 24 34 58 0 64 67
清除堆顶调整后 78 69 41 62 67 24 34 58 0 64 -1
POJ 2051
//============================================================================ // Name : Test.cpp // Author : // Version : // Copyright : Your copyright notice // Description : Hello World in C++, Ansi-style //============================================================================ #include <iostream> #include <algorithm> using namespace std; struct Node { int id, p, time; bool operator <(const Node &tmp) const { return (time == tmp.time) ? id > tmp.id : time > tmp.time; } } node[4000]; int main() { char opr[10]; int cnt = 0; while (scanf("%s", opr) && opr[0] != '#') { scanf("%d%d", &node[cnt].id, &node[cnt].p); node[cnt].time = node[cnt].p; cnt++; } make_heap(node, node + cnt); int k; scanf("%d", &k); while (k--) { printf("%d\n", node[0].id); pop_heap(node, node + cnt); node[cnt - 1].time += node[cnt - 1].p; push_heap(node, node + cnt); } return 0; }
5.string
有关函数
string类的构造函数: string(const char *s); //用c字符串s初始化 string(int n,char c); //用n个字符c初始化 此外,string类还支持默认构造函数和复制构造函数,如string s1;string s2="hello";都是正确的写法。当构造的string太长而无法表达时会抛出length_error异常 string类的字符操作: const char &operator[](int n)const; const char &at(int n)const; char &operator[](int n); char &at(int n); operator[]和at()均返回当前字符串中第n个字符的位置,但at函数提供范围检查,当越界时会抛出out_of_range异常,下标运算符[]不提供检查访问。 const char *data()const;//返回一个非null终止的c字符数组 const char *c_str()const;//返回一个以null终止的c字符串 int copy(char *s, int n, int pos = 0) const;//把当前串中以pos开始的n个字符拷贝到以s为起始位置的字符数组中,返回实际拷贝的数目 string的特性描述: int capacity()const; //返回当前容量(即string中不必增加内存即可存放的元素个数) int max_size()const; //返回string对象中可存放的最大字符串的长度 int size()const; //返回当前字符串的大小 int length()const; //返回当前字符串的长度 bool empty()const; //当前字符串是否为空 void resize(int len,char c);//把字符串当前大小置为len,并用字符c填充不足的部分 string类的输入输出操作: string类重载运算符operator>>用于输入,同样重载运算符operator<<用于输出操作。 函数getline(istream &in,string &s);用于从输入流in中读取字符串到s中,以换行符'\n'分开。 string的赋值: string &operator=(const string &s);//把字符串s赋给当前字符串 string &assign(const char *s);//用c类型字符串s赋值 string &assign(const char *s,int n);//用c字符串s开始的n个字符赋值 string &assign(const string &s);//把字符串s赋给当前字符串 string &assign(int n,char c);//用n个字符c赋值给当前字符串 string &assign(const string &s,int start,int n);//把字符串s中从start开始的n个字符赋给当前字符串 string &assign(const_iterator first,const_itertor last);//把first和last迭代器之间的部分赋给字符串 string的连接: string &operator+=(const string &s);//把字符串s连接到当前字符串的结尾 string &append(const char *s); //把c类型字符串s连接到当前字符串结尾 string &append(const char *s,int n);//把c类型字符串s的前n个字符连接到当前字符串结尾 string &append(const string &s); //同operator+=() string &append(const string &s,int pos,int n);//把字符串s中从pos开始的n个字符连接到当前字符串的结尾 string &append(int n,char c); //在当前字符串结尾添加n个字符c string &append(const_iterator first,const_iterator last);//把迭代器first和last之间的部分连接到当前字符串的结尾 string的比较: bool operator==(const string &s1,const string &s2)const;//比较两个字符串是否相等 运算符">","<",">=","<=","!="均被重载用于字符串的比较; int compare(const string &s) const;//比较当前字符串和s的大小 int compare(int pos, int n,const string &s)const;//比较当前字符串从pos开始的n个字符组成的字符串与s的大小 int compare(int pos, int n,const string &s,int pos2,int n2)const;//比较当前字符串从pos开始的n个字符组成的字符串与s中pos2开始的n2个字符组成的字符串的大小 int compare(const char *s) const; int compare(int pos, int n,const char *s) const; int compare(int pos, int n,const char *s, int pos2) const; compare函数在>时返回1,<时返回-1,==时返回0 string的子串: string substr(int pos = 0,int n = npos) const;//返回pos开始的n个字符组成的字符串 string的交换: void swap(string &s2); //交换当前字符串与s2的值 string类的查找函数: int find(char c, int pos = 0) const;//从pos开始查找字符c在当前字符串的位置 int find(const char *s, int pos = 0) const;//从pos开始查找字符串s在当前串中的位置 int find(const char *s, int pos, int n) const;//从pos开始查找字符串s中前n个字符在当前串中的位置 int find(const string &s, int pos = 0) const;//从pos开始查找字符串s在当前串中的位置 //查找成功时返回所在位置,失败返回string::npos的值 int rfind(char c, int pos = npos) const;//从pos开始从后向前查找字符c在当前串中的位置 int rfind(const char *s, int pos = npos) const; int rfind(const char *s, int pos, int n = npos) const; int rfind(const string &s,int pos = npos) const; //从pos开始从后向前查找字符串s中前n个字符组成的字符串在当前串中的位置,成功返回所在位置,失败时返回string::npos的值 int find_first_of(char c, int pos = 0) const;//从pos开始查找字符c第一次出现的位置 int find_first_of(const char *s, int pos = 0) const; int find_first_of(const char *s, int pos, int n) const; int find_first_of(const string &s,int pos = 0) const; //从pos开始查找当前串中第一个在s的前n个字符组成的数组里的字符的位置。查找失败返回string::npos int find_first_not_of(char c, int pos = 0) const; int find_first_not_of(const char *s, int pos = 0) const; int find_first_not_of(const char *s, int pos,int n) const; int find_first_not_of(const string &s,int pos = 0) const; //从当前串中查找第一个不在串s中的字符出现的位置,失败返回string::npos int find_last_of(char c, int pos = npos) const; int find_last_of(const char *s, int pos = npos) const; int find_last_of(const char *s, int pos, int n = npos) const; int find_last_of(const string &s,int pos = npos) const; int find_last_not_of(char c, int pos = npos) const; int find_last_not_of(const char *s, int pos = npos) const; int find_last_not_of(const char *s, int pos, int n) const; int find_last_not_of(const string &s,int pos = npos) const; //find_last_of和find_last_not_of与find_first_of和find_first_not_of相似,只不过是从后向前查找 string类的替换函数: string &replace(int p0, int n0,const char *s);//删除从p0开始的n0个字符,然后在p0处插入串s string &replace(int p0, int n0,const char *s, int n);//删除p0开始的n0个字符,然后在p0处插入字符串s的前n个字符 string &replace(int p0, int n0,const string &s);//删除从p0开始的n0个字符,然后在p0处插入串s string &replace(int p0, int n0,const string &s, int pos, int n);//删除p0开始的n0个字符,然后在p0处插入串s中从pos开始的n个字符 string &replace(int p0, int n0,int n, char c);//删除p0开始的n0个字符,然后在p0处插入n个字符c string &replace(iterator first0, iterator last0,const char *s);//把[first0,last0)之间的部分替换为字符串s string &replace(iterator first0, iterator last0,const char *s, int n);//把[first0,last0)之间的部分替换为s的前n个字符 string &replace(iterator first0, iterator last0,const string &s);//把[first0,last0)之间的部分替换为串s string &replace(iterator first0, iterator last0,int n, char c);//把[first0,last0)之间的部分替换为n个字符c string &replace(iterator first0, iterator last0,const_iterator first, const_iterator last);//把[first0,last0)之间的部分替换成[first,last)之间的字符串 string类的插入函数: string &insert(int p0, const char *s); string &insert(int p0, const char *s, int n); string &insert(int p0,const string &s); string &insert(int p0,const string &s, int pos, int n); //前4个函数在p0位置插入字符串s中pos开始的前n个字符 string &insert(int p0, int n, char c);//此函数在p0处插入n个字符c iterator insert(iterator it, char c);//在it处插入字符c,返回插入后迭代器的位置 void insert(iterator it, const_iterator first, const_iterator last);//在it处插入[first,last)之间的字符 void insert(iterator it, int n, char c);//在it处插入n个字符c string类的删除函数 iterator erase(iterator first, iterator last);//删除[first,last)之间的所有字符,返回删除后迭代器的位置 iterator erase(iterator it);//删除it指向的字符,返回删除后迭代器的位置 string &erase(int pos = 0, int n = npos);//删除pos开始的n个字符,返回修改后的字符串 string类的迭代器处理: string类提供了向前和向后遍历的迭代器iterator,迭代器提供了访问各个字符的语法,类似于指针操作,迭代器不检查范围。 用string::iterator或string::const_iterator声明迭代器变量,const_iterator不允许改变迭代的内容。常用迭代器函数有: const_iterator begin()const; iterator begin(); //返回string的起始位置 const_iterator end()const; iterator end(); //返回string的最后一个字符后面的位置 const_iterator rbegin()const; iterator rbegin(); //返回string的最后一个字符的位置 const_iterator rend()const; iterator rend(); //返回string第一个字符位置的前面 rbegin和rend用于从后向前的迭代访问,通过设置迭代器string::reverse_iterator,string::const_reverse_iterator实现 字符串流处理: 通过定义ostringstream和istringstream变量实现,<sstream>头文件中 例如: string input("hello,this is a test"); istringstream is(input); string s1,s2,s3,s4; is>>s1>>s2>>s3>>s4;//s1="hello,this",s2="is",s3="a",s4="test" ostringstream os; os<<s1<<s2<<s3<<s4; cout<<os.str();
6.stable_sort
sort优点一大堆,一个缺点就是它不是一种稳定的排序。要追求稳定性,只好用stable_sort了。
在各种排序算法中,合并排序是稳定的,但一般的合并排序需要额外的O(N)的存储空间,而这个条件不是一定能够满足的。所以在stable_sort内部,首先判断是否有足够的额外空间,有的话就使用普通合并函数,总的时间复杂性和快速排序一个数量级,都是O(N*logN)。
如果没有额外空间,使用了一个merge_without_buffer的关键函数进行就地合并,这个合并过程不需要额外的存储空间(递归的堆栈除外),但时间复杂度变成O(N*logN),这种情况下,总的stable_sort时间复杂度是O(N*logN*logN)。