寒假第一周总结
一.排序
1.1快速排序
算法思想:分治,将要排序的部分分为两部分,通过比较互换是得左半部分比由半部分都要小,再对两部分继续以上步骤,可以递归实现。
【模版题】
下面展示一些 代码
。
#include <iostream>
using namespace std;
#define N 100005
int a[N];
void sort(int a[],int l,int r){
if(l >= r)return;//先判断退出条件
int i = l - 1,j = r + 1,x =a[r + l>>1];//r + l >> 1就等同于(r + l)/2,这里l - 1和r + 1是因为下面用的是do,while,用x来分割
while(i < j){
do i++;while(a[i] < x);//i 从小开始增加,找到第一个大于x的值
do j--;while(a[j] > x);//j 从大开始减,找到第一个小于x的值
if(i < j) swap(a[i],a[j]);//将他们互换可以让小的去左部分,大的去右边
}
sort(a,l,j),sort(a,j+1,r);//再对左右两部分分别操作
}
int n;
int main() {
cin >> n;
for(int i = 1;i <= n;i++) {
cin >> a[i];
}
sort(a,1,n);
for(int i = 1;i <= n;i++) {
cout<< a[i]<<' ';
}
return 0;
}
主要思路:找基准值,不断找左半部分大于基准值的与右半部分小于基准值的进行互换,让醉半部分都小于基准值,右半部分大于基准值。
1.2归并排序
主要思想:将数组不断二分,对每个子数组进行排序,然后将已经排序的数组合并为一个有序的数组。
【模版题】
下面展示一些 代码
。
#include<iostream>
using namespace std;
const int N = 1000010;
int n;
int q[N], tmp[N];
void merge_sort( int q[], int l, int r)//传入要排序数组,传入左右边界
{
if(l >= r) return;//运用递归函数一定要先判断结束条件避免死循环
int mid = l + r >> 1;//l + r >> 1等同于(l + r)/2,找到中间位置接下来分解
merge_sort(q, l, mid),merge_sort(q, mid + 1, r);//将数组不断分解直到只有一个元素,在这里默认只有一个元素的时候是有序的
int k = 0, i = l, j = mid + 1;//这里i和j为分成的两部分的起始位置,开始合并数组
while(i <= mid && j <= r)
if(q[i] < q[j]) tmp[k++] = q[i++];//左右两部分开始遍历,将娇小的一个先放进tmp里这样tmp就是有序的
else tmp[k++] = q[j++];
while(i <= mid) tmp[k++] = q[i++];//这里是合并的时候可能左部分或有部分都放进了tmp里,因为两部分都是有序的所以直接把剩余的加入
while(j <= r) tmp[k++] = q[j++];//看上行解释
for(i = l, j = 0; i <= r; i++, j++) q[i] = tmp[j];//传入的左右区间是l,r这里重新把tmp的值赋给q
}
int main()
{
scanf("%d", &n);
for(int i = 0; i < n; i++) scanf("%d", &q[i]);
merge_sort(q, 0, n - 1);
for(int i = 0; i < n; i++) printf("%d ", q[i]);
return 0;
}
【例题】求逆序对的个数
思路:将数组不断二分但是他们的下标还是从大到小的,这里我们只要在归并排序的前提下稍微一变就可以了,当出现了q[ i ] > q[ j ],又因为是有序的,说明左半部分剩余的元素都能与当前的这个q[ j ]组成逆序对
主要修改部分:
下面展示一些 主要修改部分(就归并排序)
。
while(i <= mid && j <= r)
if(q[i] <= q[j]) tmp[k++] = q[i++];
else tmp[k++] = q[j++],ans+=mid-i+1;//这里是关键,因为左部分i <= mid,下标只差需要加一,比如3 ~ 5 ——>5 - 3 + 1 = 3;
下面展示一些 完整代码
。
#include<iostream>
using namespace std;
const int N = 1000010;
int n;
long long ans;
int q[N], tmp[N];
void merge_sort( int q[], int l, int r)
{
if(l >= r) return;
int mid = l + r >> 1;
merge_sort(q, l, mid),merge_sort(q, mid + 1, r);
int k = 0, i = l, j = mid + 1;
while(i <= mid && j <= r)
if(q[i] <= q[j]) tmp[k++] = q[i++];
else tmp[k++] = q[j++],ans+=mid-i+1;
while(i <= mid) tmp[k++] = q[i++];
while(j <= r) tmp[k++] = q[j++];
for(i = l, j = 0; i <= r; i++, j++) q[i] = tmp[j];
}
int main()
{
scanf("%d", &n);
for(int i = 1; i <= n; i++) scanf("%d", &q[i]);
merge_sort(q, 1, n );
cout << ans<< endl;
return 0;
}
二.二分
思想:首先确定搜索区间,然后比确认新的搜索区间,再进行归并查找时间复杂度为O(logN),比较快,(有序)
有个基本的框架
下面展示一些 基本框架
。
int l = 0, r = n - 1;
while (l < r)
{
int mid = l + r + 1 >> 1;
if (q[mid] <= x) l = mid;
else r = mid - 1;
}
cout << l << endl;
【模版题】
思路:通过用两次二分找到左右区间,每次比较中间值
下面展示一些 参考代码
。
#include <iostream>
using namespace std;
const int N = 500010;
int n, m;
int q[N];
int main()
{
scanf("%d%d", &n, &m);
for (int i = 0; i < n; i ++ ) scanf("%d", &q[i]);
while (m -- )
{
int x;
scanf("%d", &x);
int l = 0, r = n - 1;
// 二分查起始位置
while (l < r)
{
int mid = l + r >> 1;
if (q[mid] >= x) r = mid;
else l = mid + 1;
}
if (q[l] != x) cout << "-1 -1" << endl;
else
{
cout << l << ' ';
int l = 0, r = n - 1;
while (l < r)
{
int mid = l + r + 1 >> 1;
if (q[mid] <= x) l = mid;
else r = mid - 1;
}
cout << l << endl;
}
}
return 0;
}
三.高精度
一般都有A + B , A - B , A * B(高**低) , A / B(高 / 低)。
这里用数组存储
3.1加法
直接上模版题
思路:输入字符串代表要相加的数,每一位逆序存入数组,便于从个位对齐,再用一个数组存储相加之后的值,用temp存储进位。
下面展示一些 内联代码片
。
#include<iostream>
#include<stdio.h>
#include<cstring>
#include<cmath>
using namespace std;
#define N 100005
int a[N],b[N],all[N];
char ac[N],bc[N];
int len1,len2,len3;
int main(){
cin>>ac>>bc;
len1 = strlen(ac);//计算位数
len2 = strlen(bc);
len3 = max(len1,len2) + 1;//可能进位需要预留
int temp = 0;//进位
for(int i = 0;i < len1;i ++) a[i] = ac[len1 - i - 1] - '0';//字符要转化为数字,逆序相加
for(int i = 0;i < len2;i ++) b[i] = bc[len2 - i - 1] - '0';
for(int i = 0;i < len3;i ++){
all[i] = a[i] + b[i] + temp;//模拟算式相加
temp = all[i]/10;
all[i] %= 10;//取个位
}
while(len3 > 0 && all[len3] == 0)len3--;
for(int i = len3;i >= 0;i --){
cout<<all[i];
}
return 0;
}
3.2减法
思路;同样模拟数学中的计算过程,但是需要判断一下第一个数与第二个数的大小,逐位相减;
【模版题】
下面展示一些 代码
。
#include<iostream>
#include<vector>
using namespace std;//需要计算当前结果和处理借位和进位
bool cmp(vector <int>A,vector <int>B)
{
if(A.size()!=B.size()) return A.size()>B.size();
for(int i=A.size()-1;i>=0;i--){
if(A[i]!=B[i])
return A[i]>B[i];
}
return true;
}
vector<int> sub(vector <int>A,vector<int>B)
{
vector<int>v;
for(int i=0,t=0;i < A.size();i++){
t=A[i]-t;
if(i < B.size())t-=B[i];
v.push_back((t+10)%10);
if(t<0)t = 1;
else t=0;
}
while(v.size()>1 && v.back()==0) v.pop_back();
return v;
}
int main()
{
string a,b;
vector <int>v1,v2;
cin>>a>>b;
vector <int>c;
for(int i=a.size()-1;i>=0;i--)v1.push_back(a[i]-'0');//字符转化为数字
for(int i=b.size()-1;i>=0;i--)v2.push_back(b[i]-'0');
if(cmp(v1,v2)) c=sub(v1,v2);
else {
c=sub(v2,v1);
cout<<'-';
}
for(int i=c.size()-1;i>=0;i--)
cout<<c[i];
cout<<endl;
return 0;
}
3.3乘法
高精度*低精度,尽量把低精度数看成一个整体,用一个变量存储进位后剩余的数用于加到下一位;
【模版】
下面展示一些 参考代码
。
#include<iostream>
#include<vector>
using namespace std;
vector<int>mul(vector<int>A,int b)
{
vector<int>c;
int t=0;
for(int i=0;i<A.size()||t;i++){
if(i<A.size())t+=A[i]*b;//
c.push_back(t%10);
t/=10;
}
while(c.size()>1 && c.back()==0) c.pop_back();
return c;
}
int main()
{
string a;
int b;
cin>>a>>b;
vector <int>v;
for(int i=a.size()-1;i>=0;i--) v.push_back(a[i]-'0');//记得转化为数字做的时候忘记了
auto c=mul(v,b);//auto 很好用它会根据c的类型而定
for(int i=c.size()-1;i>=0;i--) cout<<c[i];
cout<<endl;
return 0;
}
3.4除法
高精度 ÷ 低精度 除法从数学算式可以看出来是高位对齐,所以不需要倒序
下面展示一些 内联代码片
。
#include<iostream>
#include<vector>
#include<algorithm>
using namespace std;
vector<int> div(vector<int> A,int b,int &r){
vector<int> C;
r = 0;
for(int i = A.size() - 1;i >= 0;i--){
r = r*10 + A[i];
C.push_back(r/b);
r %= b;
}
reverse(C.begin(),C.end());
while(C.size() > 1 && C.back() == 0) C.pop_back();
return C;
}
int main(){
string a;
int b;
cin >> a >> b;
vector<int> A;
for(int i = a.size()-1;i >= 0;i--) A.push_back(a[i]-'0');//记得转化为数字
int r;
auto C = div(A,b,r);//auto自动根据输入判断类型
for(int i = C.size()-1;i >=0;i--) cout << C[i] ;
cout << endl;
cout << r << endl;
return 0;
}
四.前缀和和差分
前缀和最好是从下标1开始不用处理边界问题
4.1一维前缀和
用另一个数组sum来存储前面的和,sum[ i - 1] + a[i];
【模版】
下面展示一些 内联代码片
。
#include <iostream>
using namespace std;
const int N = 1e5 + 10;
int n,q,a[N],s[N];
int main()
{
cin >> n >> q;
for(int i = 1;i <= n;i++) cin >> a[i];
for(int i = 1;i <= n;i++) s[i] = s[i - 1] + a[i];//记录前i项的和
while(q--)
{
int l,r;
cin >> l >> r;
if(l > r)//注意l和r的大小关系
{
int temp = l;
l = r;
r = temp;
}
cout << s[r] - s[l - 1] << endl;//记得l - 1,l那一位也要加上
}
return 0;
}
4.2二维前缀和
放在一个二维表格中来看,两条边还是一位前缀和,但其他位置就是sum[i][j] = sum[i - 1][j] + sum[i][j - 1] - sum[i - 1][j - 1] + sum[i][j];
计算的时候有重复减掉的区域要加上
【模版】
下面展示一些 参考代码
。
#include <iostream>
using namespace std;
const int N=1005;
int 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++)
{
cin>>s[i][j];
s[i][j]+=s[i-1][j]+s[i][j-1]-s[i-1][j-1];
}
}
while(q--)
{
int x1,y1,x2,y2;
cin>>x1>>y1>>x2>>y2;
cout<<s[x2][y2]-s[x1-1][y2]-s[x2][y1-1]+s[x1-1][y1-1]<<endl;
}
return 0;
}
4.3差分
建造一个新数组q对于存入l ~ r都加c,只需要b[l] += c,b[r + 1] -= c,再对b数组进行前缀和操作加上原数组就是最后的值
【模版】
下面展示一些 差分代码
。
#include<iostream>
using namespace std;
int n,m;
const int N = 100005;
int a[N],b[N];
void insert(int l ,int r,int c){
b[l]+=c;
b[r+1]-=c;
}
int main(){
cin >> n >> m;
for(int i = 1; i <= n;i++) scanf("%d",&a[i]);
for (int i = 1; i <= n; i ++ ) insert(i, i, a[i]);
while(m --){
int l , r, c;
cin >> l >> r >> c;
insert(l,r,c);
}
for(int i = 1;i <= n;i++) b[i] += b[i-1];//对b进行前缀和,这样就实现了l到r都加上了目标值
for(int i = 1;i <= n;i++) printf("%d ",b[i]);
return 0;
}
4.4二维差分
如果给b[ i ][ j ]加上c则右下角的数都会加上c
这样就需要特殊处理
这样就把不需要加的部分给减掉了
要加上a[x2 + 1][y2 + 1] = c.因为右下角被减了两遍
【模版】
下面展示一些 内联代码片
。
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
int a[1010][1010];
void fun(int x1,int y1,int x2,int y2,int c){//主要代码
a[x1][y1]+=c;
a[x2+1][y1]-=c;
a[x1][y2+1]-=c;
a[x2+1][y2+1]+=c;
}
int main(){
int n,m,q;
cin>>n>>m>>q;
int t;
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++){
cin>>t;
fun(i,j,i,j,t);
}
int x1,x2,y1,y2,k;
while(q--){
scanf("%d%d%d%d%d",&x1,&y1,&x2,&y2,&k);
fun(x1,y1,x2,y2,k);
}
for(int i=1;i<=n;i++){
for(int j=1;j<=m;j++){
a[i][j]+=a[i-1][j]+a[i][j-1]-a[i-1][j-1];
printf("%d ",a[i][j]);
}
cout <<endl;
}
return 0;
}
五.双指针算法
有左右指针还有快慢指针等,优点可以通过特殊指针操作节省时间,可以让指针逐渐靠近达到目的,比较灵活好用,避免循环1嵌套;
一般都是先想到暴力的解法然后再想优化
【模版】
下面展示一些 参考代码
。
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
#define N 100005
int n,ans;
int a[N],b[N];
int main(){
cin >> n;
for(int i = 0;i < n;i++) {
cin >> a[i];
}
int i = 0,j=0;
for( i = 0,j=0;i < n;i++) {
b[a[i]]++;//记录每个数字出现的次数
while(b[a[i]] > 1){//说明已经有重复数字j开始向右移动
b[a[j]]--;
j++;
}
ans=max(ans,i-j+1);
}
cout<<ans;
return 0;
}
六.位运算
【例题】
***思路:***这里用一种比较巧妙的思路,定义一个变量f为一于目标值相与如果为一答案加一然后f = f << 1;直到f == 0 为止
下面展示一些 内联代码片
。
#include<iostream>
using namespace std;
int n;
int sum(int x){
int cnt = 0;
int f = 1;
while(f){//当 f 左移32位后就是0
if(f&x){
cnt++;
}
f = f << 1;//逐渐左移只有相与一位是一其他都是零
}
return cnt;
}
int main(){
cin>>n;
for(int i = 1;i <= n;i ++){
int p;
cin>>p;
cout<<sum(p)<<' ';
}
return 0;
}
七.离散化
将跨度很大但是个数不多的数据离散化成连续的数据
1.先排序:对要离散化的数据排序,可以使用sort
2.去重:然后去重,c++可以使用unique()和lower_bound,
3.索引
可以节省空间
sort(alls.begin(),alls.end());
alls.erase(unique(alls.begin(),alls.end()),alls.end());
对于unique 返回的是去重后的末尾的迭代器,注意重复部分并没有被去掉只是变成了任意值放在了后面所以要用erase去除
auto 可以根据输入自动判断类型
【例题】
下面展示一些 内联代码片
。
#include<iostream>
#include<vector>
#include<algorithm>
using namespace std;
const int N = 300005;
int n,m;
typedef pair<int ,int> pII;
int a[N],s[N];//a[N]数组存储离散化后对应的下标的值,s[N]是a的前缀和
vector<int> alls;//alls是存储所有的要用到的下标,具体离散化的主题
vector<pII> add,query;//键值对,记录插入和询问的操作,
int find(int x){//主要部分二分查找离散化后对应的坐标
int l = 0,r = alls.size() - 1;
while(l < r){
int mid = l + r >> 1;
if(alls[mid] >= x) r = mid;
else{
l = mid + 1;
}
}
return r + 1;
}
int main(){
cin>>n>>m;
for(int i = 0;i < n;i ++){
int x,c;
cin>>x>>c;
add.push_back({x,c});
alls.push_back(x);//把要用到的坐标记录
}
for(int i = 0;i < m;i ++){
int l , r;
cin>>l>>r;
query.push_back({l,r});
alls.push_back(l);//把要用到的坐标记录
alls.push_back(r);//把要用到的坐标记录
}
sort(alls.begin(),alls.end());
alls.erase(unique(alls.begin(),alls.end()),alls.end());//必须是在有序的前提下一般伴随sort
//unique是去重操作返回的是去重后的末尾的迭代器再结合erase去掉重复的
for(auto item : add){
int x = find(item.first);
a[x] +=item.second;
}
for(int i = 1;i <= alls.size();i ++)s[i] = s[i - 1] + a[i];
for(auto item : query){
int l = find(item.first),r = find(item.second);//l,r必须是离散化之后的值
cout<<s[r] - s[l - 1]<< endl;
}
return 0;
}
八.区间合并
要用到pair键值对
关于pair
定义:typedef pair<int ,int > PII
头文件:#include
赋值:push_back({a,b})
排序:sort默认先排序以第一个数即first,再根据second排序
vecotr 遍历for(auto item:query)
访问:PII a:a.first , a.second
【例题】
直接对键值对排序看一下是否相交或包含,存入vector最后输出size()
下面展示一些 参考代码
。
#include<iostream>
#include<vector>
#include<algorithm>
using namespace std;
typedef long long ll;
typedef pair<int ,int >PII;
vector<PII>segs;
void merge(vector<PII> & segs)
{
vector<PII>res;
sort(segs.begin(),segs.end());
int st = -2e9,ed = -2e9;
for(auto seg:segs)
{
if(ed < seg.first){
if(st != -2e9) res.push_back({st , ed});
st = seg.first, ed = seg.second;
}
else ed = max(ed , seg.second);//注意右边要取最大,因为可能出现包含和相交
}
if(st != -2e9) res.push_back({st , ed});
segs = res;
}
int main()
{
int n;
cin >> n;
while(n --){
int l ,r;
cin >> l >> r;
segs.push_back({l ,r});
}
merge(segs);
cout << segs.size() <<endl;
return 0;
}