当前位置:文档之家› 常用经典算法(整理中)

常用经典算法(整理中)

常用经典算法(整理中)
常用经典算法(整理中)

C语言经典算法及程序

算法(Algorithm):计算机解题的基本思想方法和步骤。算法的描述:是对要解决一个问题或要完成一项任务所采取的方法和步骤的描述,包括需要什么数据(输入什么数据、输出什么结果)、采用什么结构、使用什么语句以及如何安排这些语句等。通常使用自然语言、结构化流程图、伪代码等来描述算法。

一、一些简单算法

1.求两个整数的最大公约数、最小公倍数

2.判断素数

3.验证哥德巴赫猜想

4.超级素数

5.猴子选大王

6.数的全排列

7.迭代法求平方根

二、排序算法

1.选择排序

2.冒泡排序

3.插入排序

4.快速排序

5.第K小元素

6.二分查找法

三、高精度数算法

1.已知P,且P×S=11...1,求S及1的个数

2.高精度数加法

3.高精度数减法

4.高精度数乘法

5.高精度数除法

6.高精度数阶乘

7. Fibonacci数列

四、数据结构相关问题

1.左右括号配对

2.多项式相加

3.N叉树

五、复杂算法

1.N女王问题

六、动态规划实例应用

1.求序列的最大连续序列和

2.求序列的最长下降子序列长度

3.数塔问题(解法一)

4.数塔问题(解法二)

一、一些简单算法

1.求两个整数的最大公约数、最小公倍数

最大公约数算法:(最小公倍数=两个整数之积/最大公约数)

(1) 对于已知两数m,n,使得m>n;

(2) m除以n得余数r;

(3) 若r=0,则n为求得的最大公约数,算法结束;否则执行(4);

(4) m←n,n←r,再重复执行(2)。

程序:

#include "stdio.h"

int main( )

{ int nm,r,n,m,t;

printf("please input two numbers:\n");

scan f("%d,%d”,&m,&n);

nm=n*m;

if (mn*/

r=m%n;

while (r!=0)

{ m=n; n=r; r=m%n; }

printf("最大公约数:%d\n",n);

printf("最小公倍数:%d\n",nm/n);

return 0;}

2.判断素数(只能被1或本身整除的数称为素数)

算法:把m作为被除数,将2—INT(sqrt(m) )作为除数,如果都除不尽,m就是素数,否则就不是。

将其写成一函数,若为素数返回1,不是则返回0

#include "stdio.h"

#include "math.h"

int prime( int m)

{int i,k;

if(m==2) return 1;

if(m<=1||m%2==0) return 0;

k=(int)sqrt(m);

for(i=3;i<=k;i=i+2)

if(m%i==0) return 0;

return 1;

}

int main( )

{int n;

scanf("%d",&n);

if(prime(n)) printf("%d是素数",n);

else printf("%d不是素数",n);

}

3.验证哥德巴赫猜想

(任意一个大于等于6的偶数都可以分解为两个素数之和)

算法:n为大于等于6的任一偶数,可分解为n1和n2两个数,分别检查n1和n2是否为素数,如都是,则为一组解。如n1不是素数,就不必再检查n2是否素数。先从n1=3开始(到n/2结束),检验n1和n2(n2=N-n1)是否素数。然后使n1+2 再检验n1、n2是否素数,…直到n1=n/2为止。本程序可利用上个程序中的prime()函数

程序:

#include "stdio.h"

int main( )

{ int x,i;

printf("please input a even number(>=6):\n");

do

scanf("%d",&x);

while (x<6||x%2!=0);

for(i=2;i<=x/2;i++)

if (prime(i)&&prime(x-i))

{printf("%d+%d\n",i,x-i);

printf("验证成功!”); break;

}

return 0;

}

4.超级素数

一个n位超级素数是指一个n位正整数,它的前1位,前2位, . . . , 前n位均为素数,例如,7331是个4位超级素数,因为7,73,733,7331均为素数。由键盘输入n (n<9), 然后输出全部的n位超级素数。本程序可利用之前程序中的prime()函数

#include

#include

int n;

void find(int k,long x)

{ long i;

if (k==n) printf("%ld\n",x);

else

for (i=1;i<=9;i=i+2)

if (prime(x*10+i)) find(k+1,x*10+i);

}

int main( )

{ printf("input the number of digit:\n");

do

scanf("%d",&n);

while(n<=0||n>9);

find(0,0);

return 0;

}

5.猴子选大王

从键盘上输入两整数N(<30),K(<20),表示N只猴子顺时针围成一圈,从第一只猴子开始,顺时针数到第K只猴子,令其离开队伍,最后剩下的一只猴子为大王,输出大王的编号.

程序:

#include

#include

int main( )

{int a[40]={0};

int i,j,count,k;

scanf("%d,%d",&n,&k);

for(j=1;j<=n-1;j++)

{count=0;

while(count

{

if(a[i]==0) count++;

i=(i+1)%n;

}

a[(i-1+n)%n]=j;

}

for (i=0;i<=n-1;i++)

if(a[i]==0)

{printf("the result is %d\n",i+1);

break;

}

return 0;

}

Joseph问题

题目描述::k个好人与k个坏蛋站一圈,前k个都是好人,从1开始报数,报到m的枪毙,下一个再从1开始报数,以此类推!求一个数m,当剩下k个人时,满足他们都是好人。

输入:

仅有的一个数字是k(0 < k <14)。

输出:

使得最先出列的k个人都是坏人的m的最小值。

输入样例:

4

输出样例:

30

程序:

#include

long k, m, start;

int check(long remain){

long result = ( start+m-1 ) % remain;

if ( k<=result) {

start = result; return 1;

}

else return 0;

}

int main(){

long i, find = 0;

scanf("%ld", &k);

m = k;

while( find= =0) {

find = 1; start = 0;

for (i = 0; i < k; i++)

if (!check( 2*k-i)){

find = 0; break;

}

m++;

}

printf("%ld\n",m-1);

return 0;

}

6.数的全排列

给定两个自然数n,r(n>r)输出从数1到n中按降序顺序取r个自然数的所有组合(按降序输出)。

例如,n=5,r=3时,输出的结果是:

5 4 3

5 4 2

5 4 1

5 3 2

5 3 1

5 2 1

4 3 2

4 3 1

4 2 1

3 2 1

7.以下函数sqt是用迭代法求某数a的平方根。已知求平方根的迭代公式为:x n+1=(x n+a/x n)/2,要求前后两次求出的差的绝对值小于10-5,请填空实现上述功能。#include

#include

double sqt(double a){

double x0,x1;

x0=a/2;

x1=(x0+a/x0)/2;

while(fabs(x1-x0)>= 0.00001){

x0=x1;

x1=(x0+a/x0)/2;

}

return x1;

}

int main(){

double a;

printf("Input a positive number:\n");

scanf("%lf",&a);

printf("The square root of %lf is %lf\n",a,sqt(a));

return 0;

}

二、排序算法

1.选择排序(升序)

算法:

1)对有n个数的序列(存放在数组a[n]中),从中选出最小的数,与第1个数交换位置;

2)对第(i=0…n-2)次来说,从尚未排序的其余n-1-i个数中选最小的数,与第i个数交换位置;

选择排序是不稳定的。算法复杂度O(n2)

程序:

#include "stdio.h"

#define N 10

int main( )

{ int i,j,min,t,a[N];

printf("\ninput %d numbers:\n",N);

for(i=0;i<=N-1;i++) scanf("%d",&a[i]);

for(i=0;i

{ min=i;

for(j=i+1;j<=N-1;j++) /*找到最小值所在的下标*/

if(a[min]>a[j]) min=j;

if(i!=min) {t=a[i]; a[i]=a[min]; a[min]=t; } /*交换*/

}

for(i=0;i<=N-1;i++) printf("%5d ",a[i]);

return 0;

}

算法:(将相邻两个数比较,小的调到前头)

1)有n个数(存放在数组a[n]中),第一趟将每相邻两个数比较,小的调到前头,经n-1次两两相邻比较后,最大的数已"沉底",放在最后一个位置,小数上升"浮起";

2)第(i=0…n-2)趟对余下的n-i个数(最大的数已"沉底")按上法比较,经n-1-i次两两相邻比较后得第i大的数;

冒泡排序是稳定的。算法时间复杂度O(n2)

程序:

#include "stdio.h"

#define N 10

int main( )

{ int a[N];

int i, j, t;

printf("input %d numbers\n",N);

for(i=0;i

for(i=0;i

if(a[j]>a[j+1]) { t=a[j]; a[j]=a[j+1]; a[j+1]=t; }

for(i=0;i

return 0;

}

算法:将未排序元素依次插入到己排序序列中,最终形成有序序列。1)有n个数(存放在数组a[n]中),首先将a[0]看成有序序列

2)将a[1]—a[n-1]依次插入到有序序列中

3)插入元素t=a[i]的方法:此时有序序列为a[0]—a[i-1],

从a[j](j从i-1到0)处开始检查,若a[j]>t,则将a[j]后移1位即a[j+1]=a[j],否则a[j+1]=t,停止检查

直接插入排序是稳定的。算法时间复杂度O(n2)

程序:

#include "stdio.h"

#define N 10

int main( )

{ int a[N];

int i,j,t;

printf("input %d numbers\n",N);

for(i=0;i

for(i=1;i

{ t=a[i]; j=i-1;

while(a[j]>t && j>=0]) { a[j+1]=a[j]; j--; }

a[j+1]=t;

}

for(i=0;i

return 0;

}

4.快速排序(升序)

算法思想简单描述:

快速排序是对冒泡排序的一种本质改进。它的基本思想是通过一趟扫描后,使得排序序列的长度能大幅度地减少。在冒泡排序中,一次扫描只能确保最大数值的数移到正确位置,而待排序序列的长度可能只减少1。快速排序通过一趟扫描,就能确保某个数(以它为基准点吧)的左边各数都比它小,右边各数都比它大。然后又用同样的方法处理它左右两边的数,直到基准点的左右只有一个元素为止。快速排序是不稳定的。最理想情况算法时间复杂度O(nlog2n),最坏O(n2)

#define N 100

int a[N];

void quick_sort( int low, int high)

{int i, j, t,key;

if (low < high)

{ i = low; j = high; key= a[low];

while (i

{ while (i=key) j--; /*右边找到比key小的数*/

while (i

if (i

{t=a[j]; a[j]=a[i]; a[i]=t;

j--; i++;

}

}

if(j!=low)

{t=a[j]; a[j] = a[low]; a[low]=t; }/*一遍扫描完后,交换a[j],a[low]*/ quick_sort(low,i-1); /*对基准点左边的数再执行快速排序*/

quick_sort(i+1,high); /*对基准点右边的数再执行快速排序*/

}

}

5.第K小元素

将n个元素放在一个数组中,然后取出第k个元素为标准m,把n个元素重新排列:小于标准m的元素放在数组前面,大于该标准的放在数组的后面。把该元素m放在两者之间。设小于标准的元素个数为j-1,如果j=k,则A(k)即为所求元素。如果j>k,则第k个元素必在区间[1,j],因此取A[1],…,A[j]为新的元素集合,然后重复上述的”部分排序”的过程。如果j

6.二分查找法

void binsearch(int b[],int x,int low,int high) {int mid;

if (low>high)

{printf("\n%d dosn't exists in the array!\n",x);

exit(0);

}

mid=(low+high)/2;

if(x==b[mid])

{printf("OK! b[%d]=%d\n",mid,x);

exit(0);

}

if(x>b[mid])

binsearch(b,x,mid+1,high);

else

binsearch(b,x,mid,high-1);

}

int main()

{int a[10]={2,3,4,5,7,10,20,40,50,100};

int key,n=10;;

printf("input a number for search:\n");

scanf("%d",&key);

binsearch(a,key,0,n-1);

return 0;

}

三、高精度数算法

1.已知P(P的个位数只能是1,3,7,9),且P×S=11...1,求S及1的个数。算法:

(1)令a=n=0,令a=a*10+1,n=n+1,直到a≥p,记下此时1的个数n;(2)输出a/p的值,令a=a % p

(3)若a≠0,则a=a*10+1,n=n+1,返回第2步,若a=0结束。

(4)输出1的个数n

程序:

#include "stdio.h"

int main( )

{

int n=0; long p,a=0;

printf("Please input p:");

do

scanf("%ld",&p);

while(p%2==0||p%5==0);

while(a

printf("s=");

while(1)

{

printf("%d",a/p);

a=a%p;

if(a==0) break;

a=a*10+1;

n++;

}

printf("\nn=%d",n);

return 0;

}

算法:

(1)从键盘上分别获取两个字符串s1,s2

(2)分别计算两个字符串的长度L1,L2,令L=L1>L2?L1:L2;

(3)将s1,s2逆序存储到足够大的数组a,数组b中

(4)for(i=0;i

(5)for(i=0;i

程序:

#include "stdio.h"

#include "string.h"

#define MAX 200

int main( )

{

int a[MAX]={0}, b[MAX]={0}, c[MAX]={0};

int i,L1,L2,L,k; char s1[MAX],s2[MAX];

printf("\nPlease input s1:"); scanf("%s",s1);

printf("\nPlease input s2:"); scanf("%s",s2);

L1=strlen(s1); L2=strlen(s2);

L=L1>L2?L1:L2;

for(i=L1-1,k=0;i>=0;i--) a[k++]=s1[i]-'0';

for(i=L2-1,k=0;i>=0;i--) b[k++]=s2[i]-'0';

for(i=0;i

for(i=0;i

{

c[i+1]=c[i+1]+c[i]/10 ; c[i]=c[i]%10;

}

printf("\ns1+s2=");

for(i=L;c[i]==0;i--); /*找出首位*/

while(i>=0)printf("%d",c[i--]);

return 0;

}

算法:

(1)从键盘上分别获取两个字符串s1,s2

(2)分别计算两个字符串的长度L1,L2,另L=L1>L2?L1:L2; (3)将s1,s2逆序存储到足够大的数组a,数组b中

(4)若s1>s2则for(i=0;i

程序:

#include "stdio.h"

#include "string.h"

#define MAX 200

int main( )

{ int a[MAX]={0},b[MAX]={0},c[MAX]={0};

int i,L1,L2,L,k; char s1[MAX],s2[MAX];

printf("\nPlease input s1:"); scanf("%s",s1);

printf("\nPlease input s2:"); scanf("%s",s2);

L1=strlen(s1); L2=strlen(s2);

L=L1>L2?L1:L2;

for(i=L1-1,k=0;i>=0;i--)a[k++]=s1[i]-'0';

for(i=L2-1,k=0;i>=0;i--)b[k++]=s2[i]-'0';

for(i=L-1,k=0;i>=0;i--) /*若k=-1表示s1

printf("\ns1-s2=");

if(k==0) for(i=0;i

else for(i=0;i

for(i=0;i

for(i=MAX-1;c[i]==0;i--);

if(k==-1)printf("-"); /*s1=0)printf("%d",c[i--]);

return 0;

}

4.高精度数乘法:已知两个超出long型范围的大数s1,s2,求s1*s2 算法:

(1)从键盘上分别获取两个字符串s1,s2

(2)分别计算两个字符串的长度L1,L2

(3)将s1,s2逆序存储到足够大的数组a,数组b中

(4)for(i=0;i

for(j=0;j

c[j+i]=c[j+i]+b[i]*a[j];

(5)for(i=0;i

程序:

#include "stdio.h"

#include "string.h"

#define MAX 200

int main( )

{ int a[MAX]={0},b[MAX]={0},c[MAX]={0};

int i,L1,L2,j,k; char s1[MAX],s2[MAX];

printf("\nPlease input s1:"); scanf("%s",s1);

printf("Please input s2:"); scanf("%s",s2);

L1=strlen(s1); L2=strlen(s2);

for(i=L1-1,k=0;i>=0;i--)a[k++]=s1[i]- '0';

for(i=L2-1,k=0;i>=0;i--)b[k++]=s2[i]- '0';

for(i=0;i

for(j=0;j

c[j+i]=c[j+i]+b[i]*a[j];

for(i=0;i

{c[i+1]=c[i+1]+c[i]/10;c[i]=c[i]%10;}

printf("s1*s2");

for(i=MAX-1;c[i]==0;i--); /*找出首位*/

while(i>=0)printf("%d",c[i--]);

return 0;

}

算法初步比较经典的教案

算法初步与框图 一、知识网络 二、考纲要求 1.算法的含义、程序框图 (1)了解算法的含义,了解算法的思想. (2)理解程序框图的三种基本逻辑结构:顺序、条件分支、循环. 2.基本算法语句 理解几种基本算法语句――输入语句、输出语句、赋值语句、条件语句、循环语句的含义. 三、复习指南 本章是新增内容,多以选择题或填空题形式考查,常与数列、函数等知识联系密切.考查的重点是算法语句与程序框图,以基础知识为主,如给出程序框图或算法语句,求输出结果或说明算法的功能;或写出程序框图的算法语句,判断框内的填空等考查题型.难度层次属中偏低. 第一节 算法与程序框图 ※知识回顾 1 2..

3. 4. 5.算法的基本特征:①明确性:算法的每一步执行什么是明确的;②顺序性:算法的“前一步”是“后一步”的前提,“后一步”是“前一步”的继续;③有限性:算法必须在有限步内完成任务,不能无限制的持续进行;④通用性:算法应能解决某一类问题. ※典例精析 例1.如图所示是一个算法的程序框图,则该程序框图所表示的功能是 解析:首先要理解各程序框的含义,输入a,b,c三个数之后,接着判断a,b的大小,若b小,则把b赋给a,否则执行下一步,即判断a与c的大小,若c小,则把c赋给a, 否则执行下一步,这样输出的a是a,b,c三个数中的最小值.所以该程序框图所表示的功能是求a,b,c三个数中的最小值. 评注: 求a,b,c三个数中的最小值的算法设计也可以用下面程序框图来表示. 例2.下列程序框图表示的算法功能是() (1)计算小于100的奇数的连乘积 (2)计算从1开始的连续奇数的连乘积 (3)计算从1开始的连续奇数的连乘积, 当乘积大于100时,计算奇数的个数 (4)计算L≥ 1×3×5××n100成立时n的最小值 解析:为了正确地理解程序框图表示的算法,可以将执行过程分解,分析每一步执行的结果.可以看出程序框图中含有当型的循环结构,故分析每一次循环的情况,列表如下: 第一次:13,5 =?=; S i 第二次:135,7 =??=; S i 第三次:1357,9 S<不成立,输出结果是7,程序框图表示的算法功能是求使=???=,此时100 S i

C语言几种常见的排序方法

C语言几种常见的排序方法 2009-04-2219:55 插入排序是这样实现的: 首先新建一个空列表,用于保存已排序的有序数列(我们称之为"有序列表")。 从原数列中取出一个数,将其插入"有序列表"中,使其仍旧保持有序状态。 重复2号步骤,直至原数列为空。 插入排序的平均时间复杂度为平方级的,效率不高,但是容易实现。它借助了"逐步扩大成果"的思想,使有序列表的长度逐渐增加,直至其长度等于原列表的长度。 冒泡排序 冒泡排序是这样实现的: 首先将所有待排序的数字放入工作列表中。 从列表的第一个数字到倒数第二个数字,逐个检查:若某一位上的数字大于他的下一位,则将它与它的下一位交换。 重复2号步骤,直至再也不能交换。 冒泡排序的平均时间复杂度与插入排序相同,也是平方级的,但也是非常容易实现的算法。 选择排序 选择排序是这样实现的: 设数组内存放了n个待排数字,数组下标从1开始,到n结束。 i=1 从数组的第i个元素开始到第n个元素,寻找最小的元素。 将上一步找到的最小元素和第i位元素交换。 如果i=n-1算法结束,否则回到第3步 选择排序的平均时间复杂度也是O(n²)的。 快速排序 现在开始,我们要接触高效排序算法了。实践证明,快速排序是所有排序算法中最高效的一种。它采用了分治的思想:先保证列表的前半部分都小于后半部分,然后分别对前半部分和后半部分排序,这样整个列表就有序了。这是一种先进的思想,也是它高效的原因。因为在排序算法中,算法的高效与否与列表中数字间的比较次数有直接的关系,而"保证列表的前半部分都小于后半部分"就使得前半部分的任何一个数从此以后都不再跟后半部分的数进行比较了,大大减少了数字间不必要的比较。但查找数据得另当别论了。 堆排序 堆排序与前面的算法都不同,它是这样的: 首先新建一个空列表,作用与插入排序中的"有序列表"相同。 找到数列中最大的数字,将其加在"有序列表"的末尾,并将其从原数列中删除。 重复2号步骤,直至原数列为空。 堆排序的平均时间复杂度为nlogn,效率高(因为有堆这种数据结构以及它奇妙的特征,使得"找到数列中最大的数字"这样的操作只需要O(1)的时间复杂度,维护需要logn的时间复杂度),但是实现相对复杂(可以说是这里7种算法中比较难实现的)。

几种排序算法分析

《几种排序算法的分析》 摘要: 排序算法是在C++中经常要用到的一种重要的算法。如何进行排序,特别是高效率的排序是是计算机应用中的一个重要课题。同一个问题可以构造不同的算法,最终选择哪一个好呢?这涉及如何评价一个算法好坏的问题,算法分析就是评估算法所消耗资源的方法。可以对同一问题的不同算法的代价加以比较,也可以由算法设计者根据算法分析判断一种算法在实现时是否会遇到资源限制的问题。排序的目的之一就是方便数据的查找。在实际生活中,应根据具体情况悬着适当的算法。一般的,对于反复使用的程序,应选取时间短的算法;对于涉及数据量较大,存储空间较小的情况则应选取节约存储空间的算法。本论文重点讨论时间复杂度。时间复杂度就是一个算法所消耗的时间。算法的效率指的是最坏情况下的算法效率。 排序分为内部排序和外部排序。本课程结业论文就内部排序算法(插入排序,选择排序,交换排序,归并排序和基数排序)的基本思想,排序步骤和实现算法等进行介绍。 本论文以较为详细的文字说明,表格对比,例子阐述等方面加以比较和总结,通过在参加数据的规模,记录说带的信息量大小,对排序稳定的要求,关键字的分布情况以及算法的时间复杂度和空间复杂度等方面进行比较,得出它们的优缺点和不足,从而加深了对它们的认识和了解,进而使自己在以后的学习和应用中能够更好的运用。

1.五种排序算法的实例: 1.1.插入排序 1.1.1.直接插入排序 思路:将数组分为无序区和有序区两个区,然后不断将无序区的第一个元素按大小顺序插入到有序区中去,最终将所有无序区元素都移动到有序区完成排序。 要点:设立哨兵,作为临时存储和判断数组边界之用。 实现: Void InsertSort(Node L[],int length) { Int i,j;//分别为有序区和无序区指针 for(i=1;i=1)//直到增量缩小为1 { Shell(L,d); d=d/2;//缩小增量 } } Void Shell(Node L[],int d) {

各种排序算法的总结和比较

各种排序算法的总结和比较 1 快速排序(QuickSort) 快速排序是一个就地排序,分而治之,大规模递归的算法。从本质上来说,它是归并排序的就地版本。快速排序可以由下面四步组成。 (1)如果不多于1个数据,直接返回。 (2)一般选择序列最左边的值作为支点数据。(3)将序列分成2部分,一部分都大于支点数据,另外一部分都小于支点数据。 (4)对两边利用递归排序数列。 快速排序比大部分排序算法都要快。尽管我们可以在某些特殊的情况下写出比快速排序快的算法,但是就通常情况而言,没有比它更快的了。快速排序是递归的,对于内存非常有限的机器来说,它不是一个好的选择。 2 归并排序(MergeSort)

归并排序先分解要排序的序列,从1分成2,2分成4,依次分解,当分解到只有1个一组的时候,就可以排序这些分组,然后依次合并回原来的序列中,这样就可以排序所有数据。合并排序比堆排序稍微快一点,但是需要比堆排序多一倍的内存空间,因为它需要一个额外的数组。 3 堆排序(HeapSort) 堆排序适合于数据量非常大的场合(百万数据)。 堆排序不需要大量的递归或者多维的暂存数组。这对于数据量非常巨大的序列是合适的。比如超过数百万条记录,因为快速排序,归并排序都使用递归来设计算法,在数据量非常大的时候,可能会发生堆栈溢出错误。 堆排序会将所有的数据建成一个堆,最大的数据在堆顶,然后将堆顶数据和序列的最后一个数据交换。接下来再次重建堆,交换数据,依次下去,就可以排序所有的数据。

Shell排序通过将数据分成不同的组,先对每一组进行排序,然后再对所有的元素进行一次插入排序,以减少数据交换和移动的次数。平均效率是O(nlogn)。其中分组的合理性会对算法产生重要的影响。现在多用D.E.Knuth的分组方法。 Shell排序比冒泡排序快5倍,比插入排序大致快2倍。Shell排序比起QuickSort,MergeSort,HeapSort慢很多。但是它相对比较简单,它适合于数据量在5000以下并且速度并不是特别重要的场合。它对于数据量较小的数列重复排序是非常好的。 5 插入排序(InsertSort) 插入排序通过把序列中的值插入一个已经排序好的序列中,直到该序列的结束。插入排序是对冒泡排序的改进。它比冒泡排序快2倍。一般不用在数据大于1000的场合下使用插入排序,或者重复排序超过200数据项的序列。

数学建模10种常用算法

数学建模10种常用算法 1、蒙特卡罗算法(该算法又称随机性模拟算法,是通过计算机仿真来解决问题的算法,同时可以通过模拟可以来检验自己模型的正确性,是比赛时必用的方法) 2、数据拟合、参数估计、插值等数据处理算法(比赛中通常会遇到大量的数据需要处理,而处理数据的关键就在于这些算法,通常使用Matlab作为工具) 3、线性规划、整数规划、多元规划、二次规划等规划类问 题(建模竞赛大多数问题属于最优化问题,很多时候这些问题可以用数学规划算法来描述,通常使用Lindo、Lingo软件实现) 4、图论算法(这类算法可以分为很多种,包括最短路、网络流、二分图等算法,涉及到图论的问题可以用这些方法解决,需要认真准备) 5、动态规划、回溯搜索、分治算法、分支定界等计算机算法(这些算法是算法设计中比较常用的方法,很多场合可以用到竞赛中) 6、最优化理论的三大非经典算法:模拟退火法、神经网络、遗传算法(这些问题是用来解决一些较困难的最优化问题的算法,对于有些问题非常有帮助,但是算法的实现比较困难,需慎重使用) 7、网格算法和穷举法(网格算法和穷举法都是暴力搜索最优点的算法,在很多竞赛题中有应用,当重点讨论模型本身而轻视算法的时候,可以使用这种暴力方案,最好使用一些高级语言作为编程工具) 8、一些连续离散化方法(很多问题都是实际来的,数据可以是连续的,而计算机只认的是离散的数据,因此将其离散化后进行差分代替微分、求和代替积分等思想是非常重要的) 9、数值分析算法(如果在比赛中采用高级语言进行

编程的话,那一些数值分析中常用的算法比如方程组 求解、矩阵运算、函数积分等算法就需要额外编写库 函数进行调用) 10、图象处理算法(赛题中有一类问题与图形有关, 即使与图形无关,论文中也应该要不乏图片的,这些 图形如何展示以及如何处理就是需要解决的问题,通 常使用Matlab进行处 参数估计 C.F. 20世纪60年代,随着电子计算机的 。参数估计有多种方法,有最小二乘法、极大似然法、极大验后法、最小风险法和极小化极大熵法等。在一定条件下,后面三个方法都与极大似然法相同。最基本的方法是最小二乘法和极大似然法. 基本介绍 参数估计(parameter 尽可能接近的参数 误差 平方和  θ,使已知数据Y 最大,这里P(Y│θ)是数据Y P(Y│θ)。在实践中这是困难的,一般可假设P(Y│θ

各种排序算法比较

排序算法 一、插入排序(Insertion Sort) 1. 基本思想: 每次将一个待排序的数据元素,插入到前面已经排好序的数列中的适当位置,使数列依然有序;直到待排序数据元素全部插入完为止。 2. 排序过程: 【示例】: [初始关键字] [49] 38 65 97 76 13 27 49 J=2(38) [38 49] 65 97 76 13 27 49 J=3(65) [38 49 65] 97 76 13 27 49 J=4(97) [38 49 65 97] 76 13 27 49 J=5(76) [38 49 65 76 97] 13 27 49 J=6(13) [13 38 49 65 76 97] 27 49 J=7(27) [13 27 38 49 65 76 97] 49 J=8(49) [13 27 38 49 49 65 76 97] Procedure InsertSort(Var R : FileType); //对R[1..N]按递增序进行插入排序, R[0]是监视哨// Begin for I := 2 To N Do //依次插入R[2],...,R[n]// begin R[0] := R[I]; J := I - 1; While R[0] < R[J] Do //查找R[I]的插入位置// begin R[J+1] := R[J]; //将大于R[I]的元素后移// J := J - 1 end R[J + 1] := R[0] ; //插入R[I] // end End; //InsertSort // 二、选择排序 1. 基本思想: 每一趟从待排序的数据元素中选出最小(或最大)的一个元素,顺序放在已排好序的数列的最后,直到全部待排序的数据元素排完。 2. 排序过程: 【示例】: 初始关键字[49 38 65 97 76 13 27 49] 第一趟排序后13 [38 65 97 76 49 27 49] 第二趟排序后13 27 [65 97 76 49 38 49] 第三趟排序后13 27 38 [97 76 49 65 49] 第四趟排序后13 27 38 49 [49 97 65 76] 第五趟排序后13 27 38 49 49 [97 97 76]

几种常见内部排序算法比较

常见内部排序算法比较 排序算法是数据结构学科经典的内容,其中内部排序现有的算法有很多种,究竟各有什么特点呢?本文力图设计实现常用内部排序算法并进行比较。分别为起泡排序,直接插入排序,简单选择排序,快速排序,堆排序,针对关键字的比较次数和移动次数进行测试比较。 问题分析和总体设计 ADT OrderableList { 数据对象:D={ai| ai∈IntegerSet,i=1,2,…,n,n≥0} 数据关系:R1={〈ai-1,ai〉|ai-1, ai∈D, i=1,2,…,n} 基本操作: InitList(n) 操作结果:构造一个长度为n,元素值依次为1,2,…,n的有序表。Randomizel(d,isInverseOrser) 操作结果:随机打乱 BubbleSort( ) 操作结果:进行起泡排序 InserSort( ) 操作结果:进行插入排序 SelectSort( ) 操作结果:进行选择排序 QuickSort( ) 操作结果:进行快速排序 HeapSort( ) 操作结果:进行堆排序 ListTraverse(visit( )) 操作结果:依次对L种的每个元素调用函数visit( ) }ADT OrderableList 待排序表的元素的关键字为整数.用正序,逆序和不同乱序程度的不同数据做测试比较,对关键字的比较次数和移动次数(关键字交换计为3次移动)进行测试比较.要求显示提示信息,用户由键盘输入待排序表的表长(100-1000)和不同测试数据的组数(8-18).每次测试完毕,要求列表现是比较结果. 要求对结果进行分析.

详细设计 1、起泡排序 算法:核心思想是扫描数据清单,寻找出现乱序的两个相邻的项目。当找到这两个项目后,交换项目的位置然后继续扫描。重复上面的操作直到所有的项目都按顺序排好。 bubblesort(struct rec r[],int n) { int i,j; struct rec w; unsigned long int compare=0,move=0; for(i=1;i<=n-1;i++) for(j=n;j>=i+1;j--) { if(r[j].key

贪心算法经典例题

贪心算法经典例题 发布日期:2009-1-8 浏览次数:1180 本资料需要注册并登录后才能下载! ·用户名密码验证码找回密码·您还未注册?请注册 您的账户余额为元,余额已不足,请充值。 您的账户余额为元。此购买将从您的账户中扣除费用0.0元。 内容介绍>> 贪心算法经典例题 在求最优解问题的过程中,依据某种贪心标准,从问题的初始状态出发,直接去求每一步的最优解,通过若干次的贪心选择,最终得出整个问题的最优解,这种求解方法就是贪心算法。 从贪心算法的定义可以看出,贪心法并不是从整体上考虑问题,它所做出的选择只是在某种意义上的局部最优解,而由问题自身的特性决定了该题运用贪心算法可以得到最优解。 我们看看下面的例子 例1 均分纸牌(NOIP2002tg) [问题描述] 有 N 堆纸牌,编号分别为 1,2,…, N。每堆上有若干张,但纸牌总数必为 N 的倍数。可以在任一堆上取若干张纸牌,然后移动。移牌规则为:在编号为 1 堆上取的纸牌,只能移到编号为 2 的堆上;在编号为 N 的堆上取的纸牌,只能移到编号为 N-1 的堆上;其他堆上取的纸牌,可以移到相邻左边或右边的堆上。现在要求找出一种移动方法,用最少的移动次数使每堆上纸牌数都一样多。例如 N=4,4 堆纸牌数分别为: ①9 ②8 ③17 ④ 6 移动3次可达到目的: 从③取 4 张牌放到④(9 8 13 10) -> 从③取 3 张牌放到②(9 11 10 10)-> 从②取 1 张牌放到①(10 10 10 10)。 [输入]:键盘输入文件名。 文件格式:N(N 堆纸牌,1 <= N <= 100) A1 A2 … An (N 堆纸牌,每堆纸牌初始数,l<= Ai <=10000) [输出]:输出至屏幕。格式为:所有堆均达到相等时的最少移动次数。 [输入输出样例] a.in: 4 9 8 17 6 屏慕显示:3 算法分析:设a[i]为第i堆纸牌的张数(0<=i<=n),v为均分后每堆纸牌的张数,s为最小移到次数。 我们用贪心法,按照从左到右的顺序移动纸牌。如第i堆(0

C语言经典算法大全

C语言经典算法大全 老掉牙 河内塔 费式数列 巴斯卡三角形 三色棋 老鼠走迷官(一) 老鼠走迷官(二) 骑士走棋盘 八个皇后 八枚银币 生命游戏 字串核对 双色、三色河内塔 背包问题(Knapsack Problem) 数、运算 蒙地卡罗法求PI Eratosthenes筛选求质数 超长整数运算(大数运算) 长PI 最大公因数、最小公倍数、因式分解 完美数 阿姆斯壮数 最大访客数 中序式转后序式(前序式) 后序式的运算 关于赌博 洗扑克牌(乱数排列) Craps赌博游戏 约瑟夫问题(Josephus Problem) 集合问题 排列组合 格雷码(Gray Code) 产生可能的集合

m元素集合的n个元素子集 数字拆解 排序 得分排行 选择、插入、气泡排序 Shell 排序法- 改良的插入排序Shaker 排序法- 改良的气泡排序Heap 排序法- 改良的选择排序快速排序法(一) 快速排序法(二) 快速排序法(三) 合并排序法 基数排序法 搜寻 循序搜寻法(使用卫兵) 二分搜寻法(搜寻原则的代表)插补搜寻法 费氏搜寻法 矩阵 稀疏矩阵 多维矩阵转一维矩阵 上三角、下三角、对称矩阵 奇数魔方阵 4N 魔方阵 2(2N+1) 魔方阵

1.河内之塔 说明河内之塔(Towers of Hanoi)是法国人M.Claus(Lucas)于1883年从泰国带至法国的,河内为越 战时北越的首都,即现在的胡志明市;1883年法国数学家Edouard Lucas曾提及这个故事,据说创世纪时Benares有一座波罗教塔,是由三支钻石棒(Pag)所支撑,开始时神在第一根棒上放置64个由上至下依由小至大排列的金盘(Disc),并命令僧侣将所有的金盘从第一根石棒移至第三根石棒,且搬运过程中遵守大盘子在小盘子之下的原则,若每日仅搬一个盘子,则当盘子全数搬运完毕之时,此塔将毁损,而也就是世界末日来临之时。 解法如果柱子标为ABC,要由A搬至C,在只有一个盘子时,就将它直接搬至C,当有两个盘 子,就将B当作辅助柱。如果盘数超过2个,将第三个以下的盘子遮起来,就很简单了,每次处理两个盘子,也就是:A->B、A ->C、B->C这三个步骤,而被遮住的部份,其实就是进入程式的递回处理。事实上,若有n个盘子,则移动完毕所需之次数为2^n - 1,所以当盘数为64时,则所需次数为:264- 1 = 18446744073709551615为5.05390248594782e+16年,也就是约5000世纪,如果对这数字没什幺概念,就假设每秒钟搬一个盘子好了,也要约5850亿年左右。 #include void hanoi(int n, char A, char B, char C) { if(n == 1) { printf("Move sheet %d from %c to %c\n", n, A, C); } else { hanoi(n-1, A, C, B); printf("Move sheet %d from %c to %c\n", n, A, C); hanoi(n-1, B, A, C); } } int main() { int n; printf("请输入盘数:"); scanf("%d", &n); hanoi(n, 'A', 'B', 'C'); return 0; }

C语言经典算法100例题目

看懂一个程序,分三步:1、流程;2、每个语句的功能;3、试数; 小程序:1、尝试编程去解决他;2、看答案;3、修改程序,不同的输出结果; 4、照答案去敲; 5、调试错误; 6、不看答案,自己把答案敲出来; 7、实在不会就背会。。。。。周而复始,反复的敲。。。。。 【程序1】 题目:有1、2、3、4个数字,能组成多少个互不相同且无重复数字的三位数?都是多少? ============================================================== 【程序2】 题目:企业发放的奖金根据利润提成。利润(I)低于或等于10万元时,奖金可提10%;利润高 于10万元,低于20万元时,低于10万元的部分按10%提成,高于10万元的部分,可可提 成7.5%;20万到40万之间时,高于20万元的部分,可提成5%;40万到60万之间时高于 40万元的部分,可提成3%;60万到100万之间时,高于60万元的部分,可提成1.5%,高于 100万元时,超过100万元的部分按1%提成,从键盘输入当月利润I,求应发放奖金总数? ============================================================== 【程序3】 题目:一个整数,它加上100后是一个完全平方数,再加上168又是一个完全平方数,请问该数是多少?============================================================== 【程序4】 题目:输入某年某月某日,判断这一天是这一年的第几天? ============================================================== 【程序5】 题目:输入三个整数x,y,z,请把这三个数由小到大输出。 ============================================================== 【程序6】 题目:用*号输出字母C的图案。 ============================================================== 【程序7】 题目:输出特殊图案,请在c环境中运行,看一看,Very Beautiful! ============================================================== 【程序8】 题目:输出9*9口诀。 ============================================================== 【程序9】 题目:要求输出国际象棋棋盘。 ============================================================== 【程序10】 题目:打印楼梯,同时在楼梯上方打印两个笑脸。 -------------------------------------------------------------------------------- 【程序11】 题目:古典问题:有一对兔子,从出生后第3个月起每个月都生一对兔子,小兔子长到第三个月 后每个月又生一对兔子,假如兔子都不死,问每个月的兔子总数为多少? ==============================================================

常见经典排序算法(C语言)1希尔排序 二分插入法 直接插入法 带哨兵的直接排序法 冒泡排序 选择排序 快速排

常见经典排序算法(C语言) 1.希尔排序 2.二分插入法 3.直接插入法 4.带哨兵的直接排序法 5.冒泡排序 6.选择排序 7.快速排序 8.堆排序 一.希尔(Shell)排序法(又称宿小增量排序,是1959年由D.L.Shell提出来的) /* Shell 排序法*/ #include void sort(int v[],int n) { int gap,i,j,temp; for(gap=n/2;gap>0;gap /= 2) /* 设置排序的步长,步长gap每次减半,直到减到1 */ { for(i=gap;i= 0) && (v[j] > v[j+gap]);j -= gap ) /* 比较相距gap远的两个元素的大小,根据排序方向决定如何调换*/ { temp=v[j]; v[j]=v[j+gap]; v[j+gap]=temp; } }

} } 二.二分插入法 /* 二分插入法*/ void HalfInsertSort(int a[], int len) { int i, j,temp; int low, high, mid; for (i=1; i temp) /* 如果中间元素比但前元素大,当前元素要插入到中间元素的左侧*/ { high = mid-1; } else /* 如果中间元素比当前元素小,但前元素要插入到中间元素的右侧*/ { low = mid+1; } } /* 找到当前元素的位置,在low和high之间*/ for (j=i-1; j>high; j--)/* 元素后移*/ { a[j+1] = a[j]; } a[high+1] = temp; /* 插入*/ } }

c语言经典算法100例

60.题目:古典问题:有一对兔子,从出生后第3个月 起每个月都生一对兔子,小兔 子长到第三个月后每个月又生一对兔子,假如兔子都不死,问每个月的兔子总 数 为多少? _________________________________________________________________ _ 程序分析:兔子的规律为数列1,1,2,3,5,8,13,21.... _________________________________________________________________ __ 程序源代码: main() { long f1,f2; int i; f1=f2=1; for(i=1;i<=20;i++) { printf("%12ld %12ld",f1,f2); if(i%2==0) printf("\n");/*控制输出,每行四个*/

f1=f1+f2;/*前两个月加起来赋值给第三个月*/ f2=f1+f2;/*前两个月加起来赋值给第三个月*/ } } 上题还可用一维数组处理,you try! 61.题目:判断101-200之间有多少个素数,并输出所有素数。 _________________________________________________________________ _ 1 程序分析:判断素数的方法:用一个数分别去除2到sqrt(这个数),如果能被 整 除,则表明此数不是素数,反之是素数。 _________________________________________________________________ __ 程序源代码: #include "math.h" main() { int m,i,k,h=0,leap=1;

十 大 经 典 排 序 算 法 总 结 超 详 细

数据挖掘十大经典算法,你都知道哪些? 当前时代大数据炙手可热,数据挖掘也是人人有所耳闻,但是关于数据挖掘更具体的算法,外行人了解的就少之甚少了。 数据挖掘主要分为分类算法,聚类算法和关联规则三大类,这三类基本上涵盖了目前商业市场对算法的所有需求。而这三类里又包含许多经典算法。而今天,小编就给大家介绍下数据挖掘中最经典的十大算法,希望它对你有所帮助。 一、分类决策树算法C4.5 C4.5,是机器学习算法中的一种分类决策树算法,它是决策树(决策树,就是做决策的节点间的组织方式像一棵倒栽树)核心算法ID3的改进算法,C4.5相比于ID3改进的地方有: 1、用信息增益率选择属性 ID3选择属性用的是子树的信息增益,这里可以用很多方法来定义信息,ID3使用的是熵(shang),一种不纯度度量准则,也就是熵的变化值,而 C4.5用的是信息增益率。区别就在于一个是信息增益,一个是信息增益率。 2、在树构造过程中进行剪枝,在构造决策树的时候,那些挂着几个元素的节点,不考虑最好,不然容易导致过拟。 3、能对非离散数据和不完整数据进行处理。 该算法适用于临床决策、生产制造、文档分析、生物信息学、空间数据建模等领域。 二、K平均算法

K平均算法(k-means algorithm)是一个聚类算法,把n个分类对象根据它们的属性分为k类(kn)。它与处理混合正态分布的最大期望算法相似,因为他们都试图找到数据中的自然聚类中心。它假设对象属性来自于空间向量,并且目标是使各个群组内部的均方误差总和最小。 从算法的表现上来说,它并不保证一定得到全局最优解,最终解的质量很大程度上取决于初始化的分组。由于该算法的速度很快,因此常用的一种方法是多次运行k平均算法,选择最优解。 k-Means 算法常用于图片分割、归类商品和分析客户。 三、支持向量机算法 支持向量机(Support Vector Machine)算法,简记为SVM,是一种监督式学习的方法,广泛用于统计分类以及回归分析中。 SVM的主要思想可以概括为两点: (1)它是针对线性可分情况进行分析,对于线性不可分的情况,通过使用非线性映射算法将低维输入空间线性不可分的样本转化为高维特征空间使其线性可分; (2)它基于结构风险最小化理论之上,在特征空间中建构最优分割超平面,使得学习器得到全局最优化,并且在整个样本空间的期望风险以某个概率满足一定上界。 四、The Apriori algorithm Apriori算法是一种最有影响的挖掘布尔关联规则频繁项集的算法,其核心是基于两阶段“频繁项集”思想的递推算法。其涉及到的关联规则在分类上属于单维、单层、布尔关联规则。在这里,所有支持度大于最小支

C语言经典算法100例(1---30)

2008-02-18 18:48 【程序1】 题目:有1、2、3、4个数字,能组成多少个互不相同且无重复数字的三位数?都是多少? 1.程序分析:可填在百位、十位、个位的数字都是1、2、3、4。组成所有的排列后再去 掉不满足条件的排列。 2.程序源代码: main() { int i,j,k; printf("\n"); for(i=1;i<5;i++) /*以下为三重循环*/ for(j=1;j<5;j++) for (k=1;k<5;k++) { if (i!=k&&i!=j&&j!=k) /*确保i、j、k三位互不相同*/ printf("%d,%d,%d\n",i,j,k); } } ============================================================== 【程序2】 题目:企业发放的奖金根据利润提成。利润(I)低于或等于10万元时,奖金可提10%;利润高 于10万元,低于20万元时,低于10万元的部分按10%提成,高于10万元的部分,可可提 成7.5%;20万到40万之间时,高于20万元的部分,可提成5%;40万到60万之间时高于 40万元的部分,可提成3%;60万到100万之间时,高于60万元的部分,可提成1.5%,高于 100万元时,超过100万元的部分按1%提成,从键盘输入当月利润I,求应发放奖金总数? 1.程序分析:请利用数轴来分界,定位。注意定义时需把奖金定义成长整型。 2.程序源代码: main() { long int i; int bonus1,bonus2,bonus4,bonus6,bonus10,bonus; scanf("%ld",&i); bonus1=100000*0.1;bonus2=bonus1+100000*0.75; bonus4=bonus2+200000*0.5; bonus6=bonus4+200000*0.3; bonus10=bonus6+400000*0.15; if(i<=100000)

C#实现所有经典排序算法

C#实现所有经典排序算法 //选择排序 class SelectionSorter { private int min; public void Sort(int[] arr) { for (int i = 0; i < arr.Length - 1; ++i) { min = i; for (int j = i + 1; j < arr.Length; ++j) { if (arr[j] < arr[min]) min = j; } int t = arr[min]; arr[min] = arr[i]; arr[i] = t; } } static void Main(string[] args) { int[] array = new int[] { 1, 5, 3, 6, 10, 55, 9, 2, 87, 12, 34, 75, 33, 47 }; SelectionSorter s = new SelectionSorter(); s.Sort(array); foreach (int m in array) Console.WriteLine("{0}", m); } } //冒泡排序 class EbullitionSorter { public void Sort(int[] arr) { int i, j, temp; bool done = false; j = 1; while ((j < arr.Length) && (!done))//判断长度 { done = true; for (i = 0; i < arr.Length - j; i++) { if (arr[i] > arr[i + 1]) {

C常用经典算法及其实现

C常用经典算法及其实 现 集团企业公司编码:(LL3698-KKI1269-TM2483-LUI12689-ITT289-

常用算法经典代码(C++版) 一、快速排序 voidqsort(intx,inty)//待排序的数据存放在a[1]..a[n]数组中 {inth=x,r=y; intm=a[(x+y)>>1];//取中间的那个位置的值 while(hm)r--;//比中间那个位置的值大,循环直到找一个比中间那个值小的 if(h<=r) {inttemp=a[h];//如果此时h<=r,交换a[h]和a[r] a[h]=a[r]; a[r]=temp; h++;r--;//这两句必不可少哦 } } if(r>x)qsort(x,r);//注意此处,尾指针跑到前半部分了

if(h=1;j--)//相邻的两两比较 if(a[j]

各种排序算法总结

各种排序算法总结 排序算法有很多,所以在特定情景中使用哪一种算法很重要。为了选择合适的算法,可以按照建议的顺序考虑以下标准: ()执行时间 ()存储空间 ()编程工作 对于数据量较小的情形,()()差别不大,主要考虑();而对于数据量大的,()为首要。主要排序法有: 一、冒泡()排序——相邻交换 二、选择排序——每次最小大排在相应的位置 三、插入排序——将下一个插入已排好的序列中 四、壳()排序——缩小增量 五、归并排序 六、快速排序 七、堆排序 八、拓扑排序 九、锦标赛排序 十、基数排序 一、冒泡()排序 从小到大排序个数 () { ( <) { ( <) { ([]>[])比较交换相邻元素 { ; []; [][]; []; } } } } 效率(2),适用于排序小列表。 二、选择排序 从小到大排序个数

{ ; ( <) { ; ( <)每次扫描选择最小项 ([]<[]) ; ()找到最小项交换,即将这一项移到列表中的正确位置 { ; []; [][]; []; } } } 效率(2),适用于排序小的列表。 三、插入排序 从小到大排序个数 () { ( <)循环从第二个数组元素开始,因为[]作为最初已排序部分 { []标记为未排序第一个元素 ; (> []>)*将与已排序元素从小到大比较,寻找应插入的位置* { [][]; ; } []; } } 最佳效率();最糟效率(2)与冒泡、选择相同,适用于排序小列表若列表基本有序,则插入排序比冒泡、选择更有效率。 四、壳()排序——缩小增量排序 从小到大排序个数

{ ( <)增量递减 { ( <())重复分成的每个子列表 { ( <)对每个子列表应用插入排序 { []; ; (>[]>) { [][]; ; } []; } } } } 适用于排序小列表。 效率估计(^)(^),取决于增量值的最初大小。建议使用质数作为增量值,因为如果增量值是的幂,则在下一个通道中会再次比较相同的元素。 壳()排序改进了插入排序,减少了比较的次数。是不稳定的排序,因为排序过程中元素可能会前后跳跃。 五、归并排序 从小到大排序 ( ) { (>) 每个子列表中剩下一个元素时停止 ()*将列表划分成相等的两个子列表,若有奇数个元素,则在左边子列表大于右侧子列表* ()子列表进一步划分 (); [] []新建一个数组,用于存放归并的元素 ( < <)*两个子列表进行排序归并,直到两个子列表中的一个结束* { ([]<[];) { [][];

十大经典排序算法

.1 算法分类 十种常见排序算法可以分为两大类: ?比较类排序:通过比较来决定元素间的相对次序,由于其时间复杂度不能突破O(nlogn),因此也称为非线性时间比较类排序。 ?非比较类排序:不通过比较来决定元素间的相对次序,它可以突破基于比较排序的时间下界,以线性时间运行,因此也称为线性时间非比较类排序。 0.2 算法复杂度

0.3 相关概念 ?稳定:如果a原本在b前面,而a=b,排序之后a仍然在b的前面。 ?不稳定:如果a原本在b的前面,而a=b,排序之后a 可能会出现在b 的后面。 ?时间复杂度:对排序数据的总的操作次数。反映当n变化时,操作次数呈现什么规律。 ?空间复杂度:是指算法在计算机 内执行时所需存储空间的度量,它也是数据规模n的函数。 1、冒泡排序(Bubble Sort) 冒泡排序是一种简单的排序算法。它重复地走访过要排序的数列,一次比较两个元素,如果它们的顺序错误就把它们交换过来。走访数列的工作是重复地进行直到没有再需要交换,也就是说该数列已经排序完成。这个算法的名字由来是因为越小的元素会经由交换慢慢“浮”到数列的顶端。

1.1 算法描述 ?比较相邻的元素。如果第一个比第二个大,就交换它们两个; ?对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对,这样在最后的元素应该会是最大的数; ?针对所有的元素重复以上的步骤,除了最后一个; ?重复步骤1~3,直到排序完成。 1.2 动图演示 1.3 代码实现 ?

2、选择排序(Selection Sort) 选择排序(Selection-sort)是一种简单直观的排序算法。它的工作原理:首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾。以此类推,直到所有元素均排序完毕。 2.1 算法描述 n个记录的直接选择排序可经过n-1趟直接选择排序得到有序结果。具体算法描述如下: ?初始状态:无序区为R[1..n],有序区为空; ?第i趟排序(i=1,2,3…n-1)开始时,当前有序区和无序区分别为R[1..i-1]和R(i..n)。该趟排序从当前无序区中-选出关键字最小的记录R[k],将它与无序区的第1个记录R交换,使R[1..i]和R[i+1..n)分别变为记录个数增加1个的新有序区和记录个数减少1个的新无序区; ?n-1趟结束,数组有序化了。 2.2 动图演示 2.3 代码实现 ?

相关主题
文本预览
相关文档 最新文档