2023竞赛笔记合集

本文最后更新于:2 天前

2023/12/6

GESP C++等级测试四级2023 年 9 ⽉

2023/12/8

GESP C++等级测试四级2023 年 6 ⽉


希尔排序(递减增量排序)




1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
template <typename T>
void shell_sort(T array[], int length) {
int h = 1;
while (h < length / 3) {
h = 3 * h + 1;
}
while (h >= 1) {
for (int i = h; i < length; i++) {
for (int j = i; j >= h && array[j] < array[j - h]; j -= h) {
std::swap(array[j], array[j - h]);
}
}
h = h / 3;
}
}

归并排序

代码1(数组)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
void merge(const int *a, size_t aLen, const int *b, size_t bLen, int *c) {
size_t i = 0, j = 0, k = 0;
while (i < aLen && j < bLen) {
if (b[j] < a[i]) { // <!> 先判断 b[j] < a[i],保证稳定性
c[k] = b[j];
++j;
} else {
c[k] = a[i];
++i;
}
++k;
}
// 此时一个数组已空,另一个数组非空,将非空的数组并入 c 中
for (; i < aLen; ++i, ++k) c[k] = a[i];
for (; j < bLen; ++j, ++k) c[k] = b[j];
}

代码2(指针)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
void merge(const int *aBegin, const int *aEnd, const int *bBegin,
const int *bEnd, int *c) {
while (aBegin != aEnd && bBegin != bEnd) {
if (*bBegin < *aBegin) {
*c = *bBegin;
++bBegin;
} else {
*c = *aBegin;
++aBegin;
}
++c;
}
for (; aBegin != aEnd; ++aBegin, ++c) *c = *aBegin;
for (; bBegin != bEnd; ++bBegin, ++c) *c = *bBegin;
}

//也可使用 <algorithm> 库的 merge 函数,用法与上述指针式写法的相同。

1
2
3
4
5
6
7
8
9
10
11
void merge_sort(int *a, int l, int r) {
if (r - l <= 1) return;
// 分解
int mid = l + ((r - l) >> 1);
merge_sort(a, l, mid), merge_sort(a, mid, r);
// 合并
int tmp[1024] = {}; // 请结合实际情况设置 tmp 数组的长度(与 a 相同),或使用
// vector;先将合并的结果放在 tmp 里,再返回到数组 a
merge(a + l, a + mid, a + mid, a + r, tmp + l); // pointer-style merge
for (int i = l; i < r; ++i) a[i] = tmp[i];
}

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
void merge_sort(int *a, size_t n) {
int tmp[1024] = {}; // 请结合实际情况设置 tmp 数组的长度(与 a 相同),或使用
// vector;先将合并的结果放在 tmp 里,再返回到数组 a
for (size_t seg = 1; seg < n; seg <<= 1) {
for (size_t left1 = 0; left1 < n - seg;
left1 += seg + seg) { // n - seg: 如果最后只有一个段就不用合并
size_t right1 = left1 + seg;
size_t left2 = right1;
size_t right2 = std::min(left2 + seg, n); // <!> 注意最后一个段的边界
merge(a + left1, a + right1, a + left2, a + right2,
tmp + left1); // pointer-style merge
for (size_t i = left1; i < right2; ++i) a[i] = tmp[i];
}
}
}

2023/12/11

piWmGQK.jpg

2023/12/12

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
// P1192 台阶问题
#include<bits/stdc++.h>
using namespace std;
int mod=1e5+3;
int n,k,f[1000001];
int main()
{
cin>>n>>k;
f[0]=f[1]=1;//递推边界
for(int i=2;i<=n;i++)
for(int j=1;j<=k;j++)
if(i>=j)//当楼梯数大于迈步数
f[i]=(f[i]+f[i-j])%mod;

cout<<f[n]<<endl;
}

2023/12/23

x

本页面最后更新:2023/12/31 23:59:59


2023竞赛笔记合集
https://sepiatruck34735.github.io/2023/12/31/2023竞赛笔记合集/
作者
Minecraft-Sep & Minecraft-Nul.All rights reversed.
发布于
2023年12月31日
更新于
2024年2月1日
许可协议