第二周

一、栈Stack,队列Queue

1)栈是一种运算受限的线性表。限定仅在表尾进行插入和删除操作的线性表。元素遵循先进后出的原则。

// 
#include <iostream>
using namespace std;
const int N=110;
int st[N];
int bottom=0,top=-1;
int n;
int main()
{
	cin>>n;
	while(n--)
	{
		cin>>st[++top];
	}
	while(top>=bottom)
	{
		cout<<st[top--]<<" ";
	}
	return 0;
}

1.empty函数
 语法:   bool empty();
 如当前堆栈为空,返回 true 否则返回false.

2.pop函数
 语法:   void pop();
 移除堆栈中最顶层元素

3.push函数
语法: void push( const TYPE &val );
将 val 值压栈,使其成为栈顶的第一个元素。如:
    stack<int> s;
    for(int i=0; i < 10; i++)
      s.push(i);

4.size函数
 语法:   size_type size();
 返当前堆栈中的元素数目。如:
    stack<int> s;
    for(int i=0; i < 10; i++)
      s.push(i);
    cout << "This stack has a size of " << s.size() << endl;
    
5.top函数
 语法:    TYPE &top();
 返回对栈顶元素的引用. 如下代码显现和清空一个堆栈
    while( !s.empty() ) 
    {
      cout << s.top() << " ";
      s.pop();//删除栈顶元素
    }
6.StackTraverse(S)
从栈底到栈顶依次对S的每个数据元素进行访问
  
7.ClearStack(&S)
将S清为空栈
   
8.DestoryStack(&S)
 销毁栈S     
    

2)队列
队列是一种特殊的线性表,特殊之处在于它只允许在表的前端(front)进行删除操作,而在表的后端(rear)进行插入操作,和栈一样,队列是一种操作受限制的线性表。进行插入操作的端称为队尾,进行删除操作的端称为队头。元素遵循先进先出的原则。

#include <iostream>
using namespace std;
const int N=110;
int st[N];
int front=0,rear=-1;
int n;
int main()
{
	cin>>n;
	while(n--)
	{
		cin>>st[++rear];
	}
	while(rear>=front)
	{
		cout<<st[front++]<<" ";
	}
	return 0;
}

1. TYPE &back();
back()返回一个引用,指向队列的最后一个元素。

2. bool empty();
empty()函数返回真(true)如果队列为空,否则返回假(false)。

3. TYPE &front();
front()返回队列第一个元素的引用。

4. void pop();
pop()函数删除队列队头的一个元素。

5. void push( const TYPE &val );
push()函数往队列队尾中加入一个元素。

6. size_type size();
size()返回队列中元素的个数。

二 、前缀与差分

- 前缀和
1)一维数组
前缀和其实就相当于数列中的前n项和,在m次询问求某个子序列的和时,如果每次都遍历序列,则时间复杂度为O(n*m),而若先统计前缀和,则时间复杂度为O(n+m)。

    //输入
    for(int i = 1; i <= n; i ++) scanf("%d", &a[i]);
    //计算前缀和
    for(int i = 1; i <= n; i ++) s[i] = s[i - 1] + a[i];
    while(m -- ){//m次询问
        int l, r;
        cin >> l >> r ;
        cout << s[r] - s[l - 1] << endl;//输出[l,r]区间的和
    }

2)二维数组
对于一个n*m的矩阵,给出q个询问,每次计算(x1, y1), (x2, y2)内的子矩阵和。

#include <iostream>
using namespace std;
const int N = 1010;
int a[N][N], s[N][N];
int main()
{
    int n, m, q;
    cin >> n >> m >> q;
    for(int i = 1; i <= n; i ++ )
       for(int j = 1; j <= m; j ++ )
       {
           //边输入边计算前缀和
           scanf("%d", &a[i][j]);
           s[i][j] = a[i][j] + s[i][j - 1] + s[i - 1][j] - s[i - 1][j - 1];
       }
    while(q -- )
    {
        int x1, y1, x2, y2;
        scanf("%d%d%d%d",&x1, &y1, &x2, &y2);
        //输出子矩阵和
        printf("%d\n", s[x2][y2] - s[x2][y1 - 1] - s[x1 - 1][y2] + s[x1 - 1][y1 - 1]);
    }
    return 0;
}

-差分
1)一维数组
差分数组的前缀和为原数组,在位置l上加C相当于对l之后的所有数都加C,因此还需要在r+1位置减去C,从而对数组a的多个数的操作转化为了对其差分数组b中两个数的操作。

#include <iostream>
using namespace std;
const int N = 1e5 + 10;
int n, m;
int a[N], b[N];
void insert(int l, int r, int c)
{
    b[l] += c;
    b[r + 1] -= c;
}
int main(){
    scanf("%d%d", &n, &m);
    for(int i = 1; i <= n; i ++ ) 
        scanf("%d", &a[i]);//得到a的差分数组
    for(int i = 1; i <= n; i ++ ) 
        insert(i, i, a[i]);
    while(m -- )
    {
        //进行m次操作,转化为对差分数组b的操作
        int l, r, c;
        scanf("%d%d%d", &l, &r, &c);
        insert(l, r, c);
    }
    //由差分数组b求和得到原数组a
    for(int i = 1; i <= n; i ++ ) b[i] = b[i - 1] + b[i];
    for(int i = 1; i <= n; i ++ ) printf("%d ", b[i]);
    return 0;
}

2)二维数组

#include <iostream>
using namespace std;
const int N = 1010;
int n, m, q;
int a[N][N], b[N][N];
//对原矩阵的操作转化为对其差分矩阵的操作
void insert(int x1, int y1, int x2, int y2, int c)
{
    b[x1][y1] += c;
    b[x1][y2 + 1] -= c;
    b[x2 + 1][y1] -= c;
    b[x2 + 1][y2 + 1] += c;
}
int main()
{
    scanf("%d%d%d", &n, &m, &q);
    //边输入a矩阵边计算其差分矩阵b
    for(int i = 1; i <= n; i ++)
      for(int j = 1; j <= m; j ++ )
      {
          scanf("%d", &a[i][j]);
          insert(i, j, i, j, a[i][j]);//计算差分矩阵b
      }
    //q次操作
    while(q -- )
    {
        int x1, y1, x2, y2, c;
        cin >> x1 >> y1 >> x2 >> y2 >> c;
        insert(x1, y1, x2, y2, c);
    }
    //对差分矩阵b求前缀和得到q次操作后的a矩阵
    for(int i = 1; i <= n; i ++)
      for(int j = 1; j <= m; j ++ )
         b[i][j] += b[i - 1][j] + b[i][j - 1] - b[i - 1][j - 1];
    //输出
    for(int i = 1; i <= n; i ++)
    {
        for(int j = 1; j <= m; j ++ ) printf("%d ", b[i][j]);
        puts(" ");
    }
    
    return 0;
}

三、位运算

- 四种运算符

1)与 &
  只有x,y都是1时结果是1 其余情况是0;
  eg:1&1=1,1&0=0
2)或 |
  x,y中只要有一个是1,结果就是1,其余情况是0;
  eg:1|0=00|0=0
3)非 !
  如果x是0,!x=1,x是1的话,运算结果是04)异或 ^ 
  x^ y 相同是0 不同是1;
  eg: 1^ 1=00^ 0=00^1=1;
 

- 整型

 有符号整型 (int 32位二进制):整数:0 负数:1
  1:000000000001
  -1:10000000000001
 无符号整型(unsigned int 16位二进制):最高位为组成该数的一部分
 

- 反码与补码

原码 100000111
反码 011111000
补码:
  ①如果是正数:正数的补码=原码。即100000111
  ②如果是负数:负数的补码就是在原码基础上加1。即011111000+1=011111001
  

- 奇数偶数

 奇数的二进制最低位是1,3的二进制是11,5的二进制是101;
 偶数的二进制最低位是0,2的二进制是10,4的二进制是100;
  b (b&1)=1 那么b就是奇数否则b是偶数

- 左移右移

 左移: 1<<x 01 10 100 1 2 4 相当于扩大2的倍数。
 右移: 1>>x 100 10 01 4 2 1 相当于除以2。

四、二分

- 整数二分

//整数二分 (需要考虑边界问题)
//有两个模板 
//模板一 (寻找第一个大于等于x的数 )
//[l,mid],[mid+1,r]
int b[100000]
int erfen(int x , int m)//x是要寻找的数m右边界
{
	
	int l=0,r=m;
	while(l < r)
	{
		mid = l + (r-l) / 2;//或者mid=(l+r)/2 
		if(b[mid] >= x)
		r=mid;
		else
		l=mid+1;
		
	}
	if(b[l]==x)
	return 1;
	else 
	return -1;	
 } 
//模板二(寻找最后一个大于等于x的数) 
//[l,mid-1],[mid,r];
int bsearch_2(int l, int r)
{
    while (l < r)
    {
        int mid = (l + r + 1)/2;//注意加一防止死循环
        if (b[mid] <= x) 
		l = mid;
        else 
		r = mid - 1;
    }
   if(b[l]==x)
	return 1;
	else 
	return -1;	
}

- 浮点数二分

//求平方根
int main()
{
    double x;
    cin >> x;
    
    double l = 0, r = x;
    while(r - l >= 1e-6)
    {
        double mid = l + r >> 1;
        if (mid * mid >= x) r =mid;
        else l = mid;
    }
    
    cout << mid << endl;
 
    return 0;
}

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值