寒假集训笔记

注意事项(并不完整):
  (1)i式筛求素数(用bool型数组,大的数int会爆MLE)
  (2)for(int j = i;j <= n/i;j++) 能除的就不要乘(i*j可能会爆int)
  (3)快速幂别忘了long long
一、memset (40 -> 6亿)
  1.int
    0x7f一个很大的数(略小于0x7fffffff)
    0xaf一个很小的数
  2.double
    127一个很大的数
  3.16进制:0xff
  4.无符号整型
    ffffffff一个很大的数
二、最短路(负权回路无最短路)
  1.Floyd
    最简单的最短路径算法
    可以计算图中任意两点之间的最短路径
    Floyd算法的时间复杂度是n^3
    适合于出现负权边的情况
    代码实现最为简单
  2.Dijstra
    时间稳定的计算一个点到其他所有点的最短路径算法
    不能用于计算带负权的图,一般使用堆优化
  3.Bellman ford
    可以处理带负权边的最短路
    但不能处理带负环的最短路(此时不存在最短路)
    因复杂度较高,一般不建议使用
  4.Spfa
    复杂度O(玄学)
    一般在实测堆优化dijstra算法无法通过时使用
    可能更快也可能更慢,且可以被卡成O(n*m)
    //PS:堆优化存的是节点,不是边;利用邻接链表时别忘取cs[i].to
三、并查集(树型数据结构)
  1.操作
  (1)初始化
    for(int i = 1;i <= n;i++) father[i] = i;
  (2)查询:查找根节点
  (3)路径压缩

1 (查找并路径压缩)
2 int find(int x){
3   if(father[x] != x) father[x] = find(father[x]);
4   return father[x];
5 }

  (4)合并:两个元素所在集合合并为一个集合

1 void unionn(int x,int y){
2   x = find(x);
3   y = find(y);
4   father[y] = x;
5 }

  2.用某个元素所在树的根节点表示该点所在的集合
  3.找爹 for(int i = 1;i <= n;i++) if(find(i) == i) sum++;
四、离散化
  1.map

1 #include<map>
2 map<string,int> mp;
3 string a;
4 cin >> a;
5 mp[a] = 0;
6 mp["Hello"] = 50;//映射
7 mp.find(a);//查找
8 mp.clear()//清空

  2.STL(例题:程序自动分析)
五、最小生成树
  1.Prim
    min[v] 表示蓝点v与白点相连的最小边权
  2.Kruskal(克鲁斯卡尔)
    利用并查集来实现
六、vector

 1 #include <vector>
 2 vector<int> a;
 3 push_back 在数组的最后添加一个数据
 4 pop_back 去掉数组的最后一个数据
 5 at 得到编号位置的数据
 6 begin 得到数组头的指针
 7 end 得到数组的最后一个单元+1的指针
 8 front 得到数组头的引用
 9 back 得到数组的最后一个单元的引用
10 max_size 得到vector最大可以是多大
11 capacity 当前vector分配的大小
12 size 当前使用数据的大小
13 resize 改变当前使用数据的大小,如果它比当前使用的大,者填充默认值
14 reserve 改变当前vecotr所分配空间的大小
15 erase 删除指针指向的数据项
16 clear 清空当前的vector
17 rbegin 将vector反转后的开始指针返回(其实就是原来的end-1)
18 rend 将vector反转构的结束指针返回(其实就是原来的begin-1)
19 empty 判断vector是否为空
20 swap 与另一个vector交换数据

七、树状数组
  (单点修改,区间查询->普通树状数组)
  (单点查询,区间修改->差分)
  1. 负数的二进制是反码的补码

  原码 补码 反码
1 0000 0001 0000 0001

0000 0001

-1 1000 0001 1111 1110 1111 1111

 

 

 

 

 

  2.x&(-x) 返回x转二进制后最后一个1的位置

  3.构造方法

    (1)每个树状数组的元素由2^k个初始数组元素相加而成

    (2)树状数组中某个节点i的父节点的下标为i+lowbit(i)
    (3)树状数组中某个节点i的兄弟(左边的)下标为i-lowbit(i)

1 int lowbit(int x){
2     return x & (-x);
3 }

  4.单点修改 (时间复杂度O(lgn))
    若要对原数组的某个元素加上一个val值
    应该对涉及到该点的所有树状数组的点进行修改
    也就是沿父节点一路往上,直到区间的最右端

1 void update(int x,int val){
2     while(x <= n){
3         c[x] += val;
4         x += lowbit(x);
5     }
6 }

  5.询问前缀和 (时间复杂度O(lgn))
    对于询问的区间,从右端点开始
    每次累加,每次移到左边兄弟的树状数组值
    即每次下标减lowbit值

1 int query(int x){
2     int sum = 0;
3     while(x){
4         sum += c[x];
5         x -= lowbit(x);
6     }
7 }

  6.区间查询

1 int getsum(int x){
2     int s = 0;
3     while(x > 0){
4         s += c[x];
5         x -= lowbit(x);
6     }
7     return s;
8 }

  7.优化(卡常小技巧)
    (1)非递归函数前加inline,反复调用时可优化读取速度
    (2)快读快输

 1 //快读
 2 inline int read(){
 3     int x = 0,f = 1;
 4     char c = getchar();
 5     while(c > '9'||c < '0'){
 6         if(c == '-') f = -1;
 7         c = getchar();
 8     }
 9     while(c >= '0'&&c <= '9'){
10         x = (x << 3) + (x << 1) + (c ^ 48);
11         c = getchar();
12     }
13     return x*f;
14 }

  8.差分
    把每个数和它前一个数的差值来建立树状数组
    某位置原本所代表的数,就是从开始到这个位置的区间和
    将区间[l,r]加上x的话
    只需要d[l] += x;d[r+1] -= x;
八、二维树状数组
  1.规律
    不管是横坐标还是纵坐标
    将他们单独拿出来,他们都符合x += lowbit(x)
  2.修改

1 void updata(int x,int y,int key){
2     for(int i = x;i <= n;i += lowbit(i)){//注意结束位置,n还是maxn,maxn可能会小于n,不能全部修改
3         for(int j = y;j <= n;j += lowbit(j))
4         c[i][j] += key;
5     }
6 } 

  3.查询

1 int sum(int x,int y){
2   int result = 0;
3   for(int i = x;i > 0;i -= lowbit(i)){
4     for(int j = y;j > 0;j -= lowbit(j))
5       result += c[i][j];
6   }
7   return result;
8 }

  4.二维前缀和

1           x1,y2
2 ________________ ____________ x2,y2
3 |                 |
4 |                 |
5 |_______________x1,y1     |x2,y1
6 |         |       |
7 |         |       |
8 |         |        |
9 |_______________|____________|

  矩形内的和:
  query(x2,y2)-query(x1-1,y2)-query(x2,y1-1)+query(x1-1,y1-1);
九、异或
  性质:满足交换律和结合律
  交换律:A ^ B = B ^ A;
  结合律:A ^ (B ^ C) = (A ^ B) ^ C;
  恒等律:X ^ 0 = X;
  归零律:X ^ X = 0;
  自反:A ^ B ^ B = A ^ 0 = A;
  对于任意的X:X ^ (-1) = -X-1;
  如果A ^ B = C成立,那么A ^ B = C,B ^ C = A;

上一篇:继续(3n+1)猜想 C++(g++)


下一篇:现代计算机入门games101