voidquick_sort(int q[], int l, int r) { //判断边界,只有一个元素或没有元素必然有序直接返回。 if (l >= r) return;
//先移动再判断,所以初始化要多一格。 int i = l - 1, j = r + 1, x = q[l + r >> 1]; while (i < j) { do i ++ ; while (q[i] < x); do j -- ; while (q[j] > x); if (i < j) swap(q[i], q[j]); }
// C = A + B, A >= 0, B >= 0 vector<int> add(vector<int> &A, vector<int> &B) { vector<int> C;
int t = 0; //进位 for (int i = 0; i < A.size() || i < B.size(); i ++ ) { if (i < A.size()) t += A[i]; if (i < B.size()) t += B[i]; C.push_back(t % 10); t /= 10; }
if (t) C.push_back(t); return C; }
高精度减法
存储方式与加法一致
大小判别放到使用时判断
步骤:
从个位开始遍历
设置借位位,初始为t = 0
先减去上一位的t,按位加上两个整数每一位的值(需判断减数下标是否在其大小内)
合并处理减完后大于零小于零的情况。若小于零,还要令t = 1,表明需要借位。
循环结束后,去掉前导零,同时令位数不少于1
模板
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
// C = A - B, 满足A >= B, A >= 0, B >= 0 vector<int> sub(vector<int> &A, vector<int> &B) { vector<int> C;
int t = 0; for (int i = 0; i < A.size(); i ++ ) { t = A[i] - t; if (i < B.size()) t -= B[i]; C.push_back((t + 10) % 10); //合并处理减完后大于零小于零的情况 if (t < 0) t = 1; else t = 0; }
// C = A * b, A >= 0, b >= 0 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; }
高精度除以低精度
为了方便与其他运算结合,存储方式仍然采用之前的
从最高位开始遍历,与之前不同
答案的存储方式不同,返回前要使用algorithm中的reverse方法将元素倒转
注意顺序:先倒转再去前导零
模板
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
// A / b = C ... r, A >= 0, b > 0 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; }
前缀和与差分
前缀和
如何求\(S_i\):
\[
S_i=a_1+a_2+\cdots+a_i , S_0=0
\]
1 2 3 4
for(int i = 1; i <= n; i++) { s[i] = s[i-1]+a[i]; }