基础算法(一)
快速排序
void quick_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]);
}
quick_sort(q, l, j), quick_sort(q, j + 1, r);
}
[平均的]时间复杂度:
O
(
n
l
o
g
n
)
O(n log n)
O(nlogn) (以2为底的
l
o
g
n
logn
logn) [最坏]时间复杂度:
O
(
n
2
)
O(n^2)
O(n2) 快排的期望是
n
/
2
n/2
n/2,递归的层数的期望是
l
o
g
2
n
log2n
log2n层,所以平均的时间复杂度:
O
(
n
l
o
g
n
)
O(n log n)
O(nlogn)
归并排序
注意快排和递归的区别,快排是先排序再递归两边,递归是先递归再排序之类的其他操作
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 ++ ];
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];
}
时间复杂度:
O
(
n
l
o
g
n
)
O(n log n)
O(nlogn) (以2为底的
l
o
g
n
logn
logn) 归并排序的递归树,树种每层元元素的个数最多是n,也就代表着每层最多进行n次比较,而递归树最多只有log2n层(n除以log2n次就除为1了),合在一起就是
n
l
o
g
2
n
nlog2n
nlog2n了。 
整数二分
很好的二分查找解释的视频:https://www.bilibili.com/video/BV1d54y1q7k7?from=search&seid=7982797064148706619 C++的库函数 
y总的 有单调性一定可以二分,没有单调性有可能也可以二分,
bool check(int x) {}
int bsearch_1(int l, int r)
{
while (l < r)
{
int mid = l + r >> 1;
if (check(mid)) r = mid;
else l = mid + 1;
}
return l;
}
int bsearch_2(int l, int r)
{
while (l < r)
{
int mid = l + r + 1 >> 1;
if (check(mid)) l = mid;
else r = mid - 1;
}
return l;
}
浮点数二分
bool check(double x) {}
double bsearch_3(double l, double r)
{
const double eps = 1e-6;
while (r - l > eps)
{
double mid = (l + r) / 2;
if (check(mid)) r = mid;
else l = mid;
}
return l;
}
|