找中值的c语言源代码 80

在一个比较大的数组中,寻找中值。要求是O(n)复杂度.c源代码.谢谢,急用.... 在一个比较大的数组中,寻找中值。要求是O(n)复杂度.
c源代码.谢谢,急用.
展开
 我来答
478617
2015-01-21 · TA获得超过875个赞
知道小有建树答主
回答量:725
采纳率:100%
帮助的人:98.5万
展开全部
#include <iostream>
#include <cstdlib>
#include <ctime>
#include <algorithm>

using namespace std;

 
void swap(int A[], int i, int j)
{
if (i != j)
{
int t = A[i];
A[i] = A[j];
A[j] =t;
}
}

int partition(int A[], int p, int r)
{
    int x = A[r];
    int i = p - 1;
    for (int j = p; j <= r-1; j++)
{
if (A[j] <= x)
{
i++;
swap(A, i, j);
}
}
swap(A, i+1, r);
return i+1;
}

int randomized_partition(int A[], int p, int r)
{
srand(time(NULL));
int i = p + rand() % (r-p+1);
swap(A, i, r);
return partition(A, p, r);
}

int randomized_select(int A[], int p, int r, int i)
{
if (p == r) return A[p];
    int q = randomized_partition(A, p, r);
    int k = q - p + 1;

    if (i == k)     return A[q];
    else if (i < k) return randomized_select(A, p, q-1, i);
    else            return randomized_select(A, q+1, r, i-k);
}

void triplet_adjust(int A[], int i, int step)
{
int j = i + step;
    int k = i + 2 * step;
    if (A[i] < A[j])
{
if (A[k] < A[i])      swap(A, i, j);
else if (A[k] < A[j]) swap(A, j, k);
    }
    else
{
if (A[i] < A[k])      swap(A, i, j);
else if (A[k] > A[j]) swap(A, j, k);
}
}

double mean(int A[], int n)
{
double f = A[0];
for (int i = 1; i < n; i++) f += A[i];
return f / n;
}

int approximate_median(int A[], int r)
{
int step = 1;
int size = 1;
int i;
    for (int j = 0; j < r; j++) size *= 3;
    for (int k = 0; k < r; k++)
{
i = (step - 1) / 2;
while (i < size)
{
triplet_adjust(A, i, step);
i += (3 * step);
}
step *= 3;
}
return A[(size - 1) / 2];
}

void selection_sort(int A[], int left, int size, int step)
{
int min;
int i, j;
for (i = left; i < left + (size-1) * step; i += step)
{
min = i;
for (j = i+step; j < left + size; j += step)
{
if (A[j] < A[min]) min = j;
}
swap(A, i, j);
}
}

template<class T, class S>
void ncmerge_sort(T * _A, S _N)
{
S step, ins, l, m, r, pos0, pos1;
T * _T = new T[_N];

for(step = 2; step < _N * 2; step *= 2)
{
for(l = 0; l + step / 2 < _N; l += step)
{
m = l + step / 2;
r = m + step / 2 < _N ? m + step / 2 : _N;
pos0 = l; pos1 = m; ins  = 0;
while(pos0 < m && pos1 < r)
            {
if(_A[pos1] < _A[pos0]) _T[ins++] = _A[pos1++];
else                    _T[ins++] = _A[pos0++];
}
while(pos0 < m) _T[ins++] = _A[pos0++];
while(pos1 < r) _T[ins++] = _A[pos1++];
while(ins  > 0) _A[--r]   = _T[--ins];
}
}
delete [] _T;
}




const int SORT_NUM_THRESHOD = 50;

int approximate_median_any_n(int A[], int size)
{
bool left_to_right = false;
int left = 0;
int step = 1;
int i;
while (size > SORT_NUM_THRESHOD)
{
left_to_right = !left_to_right;
int rem = size % 3;
if (left_to_right) i = left;
else               i = left + (3 + rem) * step;
for (int j = 0; j < (size / 3 - 1); j++)
{
triplet_adjust(A, i, step);
i += 3 * step;
}
        if (left_to_right) left += step;
        else
{
i = left;
left += (1 + rem) * step;
        }
selection_sort(A, i, 3 + rem, step);
if (rem == 2)
{
if (left_to_right) swap(A, i+step, i+2*step);
else               swap(A, i+2*step, i+3*step);
}
step *= 3;
size = size / 3;
}
selection_sort(A, left, size, step);
return A[left + step * int( (size-1) / 2 )];
}

int main(int argc, char* argv[]) 
{
const int DEFAULT_N = 50000000;
    int N;
if (argc < 2)
{
N = DEFAULT_N;
}
else
{
N = atoi(argv[1]);
}
cout << "N = " << N << endl;
clock_t t = clock();
cout << clock() << endl;
int* A = new int[N];
cout << clock() << endl;
srand(time(NULL));
for (int i=0; i<N; i++)
{
A[i] = rand() % N;
}

clock_t t1 = clock();
cout << t1 << endl;
double avg = mean(A, N);
cout << avg << endl;
clock_t t2 = clock();
cout << t2 << endl;
int m = approximate_median_any_n(A, N); // 近似中值选择算法,不太精确,但速度快点
cout << m << endl;
clock_t t3 = clock();
cout << t3 << endl;
m = randomized_select(A, 0, N-1, N/2); // 随机选择法
cout << m << endl;
clock_t t4 = clock();
cout << t4 << endl;
//std::sort(A, A+N); // 标准快速排序
ncmerge_sort(A, N); // 原地并归排序, 排序法
m = A[N/2];
cout << m << endl;
clock_t t5 = clock();
cout << t5 << endl;
cout << endl;
double dt = (double) (t2-t1) / CLOCKS_PER_SEC;
cout << dt << "  " ;
dt = (double) (t3-t2) / CLOCKS_PER_SEC;
cout << dt << "  " ;
dt = (double) (t4-t3) / CLOCKS_PER_SEC;
cout << dt << "  " ;
dt = (double) (t5-t4) / CLOCKS_PER_SEC;
cout << dt << "  " ;
    cout << endl;

 return 0;
}
禅心如水静
2015-01-20 · 超过13用户采纳过TA的回答
知道答主
回答量:47
采纳率:0%
帮助的人:17.4万
展开全部
给你提供个思路,假如一个100000长度的数组,里面的内容是1到500的随机数,那么你可以建立一个长度为500的数组,开始时每个元素为0,每取一个值第二个数组下标与该值对应的元素加一。然后从第二个数组的第一个值开始加,(a[1]+a[2]...如果值等于50000了就输出该下标。(我学的是c++而且好久没写了,估计写出来要好久,提供个思路给你,估计你能比我更快出答案)
追问
每取一个值第二个数组下标与该值对应的元素加一, 这个是数组是什么样的,可以举个例子吗?想不明白
已赞过 已踩过<
你对这个回答的评价是?
评论 收起
推荐律师服务: 若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询

为你推荐:

下载百度知道APP,抢鲜体验
使用百度知道APP,立即抢鲜体验。你的手机镜头里或许有别人想知道的答案。
扫描二维码下载
×

类别

我们会通过消息、邮箱等方式尽快将举报结果通知您。

说明

0/200

提交
取消

辅 助

模 式