中南民族大学软工算法考试
#2019年软工算法考试
整理:北忘山
关注公众号获取更多资料
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-5dN70Ppv-1578581536946)(./1578578869033.png)]
估计题目范围:期中+习题
####首先是期中考试题目:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gCIbpVgk-1578581536949)(./1578570652724.png)]
####还有不知道哪里来的重点
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-i5xWKCg8-1578581536950)(./1578578790339.png)]
###非递归分析的五步骤
(1)决定用哪个(哪些)参数表示输入规模
(2)找出算法的基本操作
(3)检查基本操作的执行次数是否只依赖于输入规模,如果它还依赖于一些其他的特性,则最差效率,平均效率以及最优效率需要分别研究
(4)建立一个算法基本操作执行次数的求和表达式
(5)利用求和运算的标准公式和法则来建立一个操作次数的闭合公式,或者至少确定它的增长次数。
###递归分析的五个步骤
(1)决定用哪个(哪些)参数表示输入规模的度量标准
(2)找出算法的基本操作
(3)检查一下,对于相同的规模的不同输入,基本操作的执行次数是否可能不同。如果有这种可能,则必须对最差效率、平均效率以及最优效率做单独研究
(4)对于算法基本操作的执行次数,建立一个递推关系以及相应的初始条件。
(5)解这个递推式,或者至少确定它的解的增长次数。
##计算a 的 n次方及时间T(n)O(nlogn)
###//蛮力法求a的n次方
int Power1(int a,int n)
{
int ans=1;
for(int i=0;i<n;i++)
ans*=a;
return ans;
}
###//分治法求a的n次方
int Power2(int a,int n)
{
int ans=1;
if (n==0) ans=1;
else if(n==1) ans=a;
else
ans=Power2(a,n/2)*Power2(a,(n+1)/2);
return ans;
}
###//减治法求a的n次方
int Power3(int a,int n)
{
int ans=1;
if (n==0) ans=1;
else if(n==1) ans=a;
else
{
ans=Power3(a,n/2);
if(n%2==0)
ans=ans*ans;//当n为偶数
else
ans=ans*ans*a;//当n为奇数
}
return ans;
}
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fPfP2qgT-1578581536951)(./1578572372813.png)]
###汉诺塔问题
####解法
解法的基本思想是递归。假设有A、B、C三个塔,A塔有N块盘,目标是把这些盘全部移到C塔。那么先把A塔顶部的N-1块盘移动到B塔,再把A塔剩下的大盘移到C,最后把B塔的N-1块盘移到C。 每次移动多于一块盘时,则再次使用上述算法来移动。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ozXU0Jqw-1578581536952)(./1578572918697.png)]
####程序源码
#include<cstdio>
#include<iostream>
#include<algorithm>
using namespace std;
int n;
char x,y,z;
void mov(int n,char a,char c,char b)
{
if(n==0) return;
mov(n-1,a,b,c);
cout<<a<<"-->"<<n<<"-->"<<c<<endl;
mov(n-1,b,c,a);
}
int main()
{
cin>>n;
cin>>x>>y>>z;
mov(n,x,y,z);
return 0;
}
###螺钉螺母的匹配问题 (快速排序的变形)nlogn
####算法分析
螺母我就用nut表示,螺钉我就用bolt表示。
我是假设只有唯一一个匹配的,即nut数组与bolt一一对应。否则算法还需要有更大的更改。
1.在nut数组拿一个,可以把bolt数组分为比那个小的,比那个大的,还有一个匹配的3个部分。
2.在bolt中小的那堆那一个可以把nut分成比那个小的,比那个大的,还有一个匹配的3个部分。
3.这样可以发现,现在数组产生两对比配的螺母和螺钉和一队小的螺钉和螺母,一队大的螺钉和螺母。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gtO7gW4W-1578581536953)(./1578573180160.png)]
令数组内部的排列也是这样。一次调用后前两个是匹配的螺母和螺钉。后面给一个坐标,左边是小的,右边是较大的。
这样对较小的螺母和螺钉再调用一个次函数,即又可以产生两对匹配,和较小小和较大大的
####代码实现
//分类函数
//n和b为两个数组
//left为左索引,right为右索引
void Fix(int *n, int *b, int left, int right)
{
if (left < right)
{
int tmp = n[left];
int i = left, j = right;
while (i < j)
{
while (i < j&&b[i] < tmp)
{
i++;
}
while (i < j&&b[j] > tmp)
{
j--;
}
if (i < j)
{
swap(b[i], b[j]);
}
}
b[i] = tmp;
swap(b[left], b[i]);
cout << "n+b:" << endl; Print(n); Print(b); cout << endl;
//一趟下来,i=j的tmp的位置。以tmp为界限,左右分别是小于和大于它的元素
tmp = b[left + 1];
i = left + 1, j = right;
while (i < j)
{
while (i < j&&n[i] < tmp)
{
i++;
}
while (i < j&&n[j] > tmp)
{
j--;
}
if (i < j)
{
swap(n[i], n[j]);
}
}
n[i] = tmp;
swap(n[left + 1], n[i]);
cout << "n+b:" << endl; Print(n); Print(b); cout << endl;
Fix(n, b, left + 2, i);
Fix(n, b, i + 1, right);
}
}
###两个数组的公共元素问题
时间复杂度是 O(maxn(m,n))
#include<stdio.h>
#include<stdlib.h>
int elements(int a[],int b[],int c[],int sizea,int sizeb)
{
//a 和 b分别是两个比较数组
//c是用来存放公共元素的数组
//sizea sizeb 则是两个数组的大小
//在这个地方可以用:a.length表示,而不用传入数组的大小
int pa=0; //指向a数组的指针
int pb=0; //指向b数组的指针
int count=0; //统计数量
while(pa<sizea && pb<sizeb) //进入的必要条件
{
if (a[pa]>b[pb])
pb++;
else if(a[pa]<b[pb])
pa++;
else
{ c[count++]=a[pa];
pa++;
pb++;
}
}
return (count);
}
int main(void)
{
int a[]={1,2,3,4,5,6,7,8,9};
int b[]={4,5,6};
int c[10];
int i,count;
count = elements( a,b,c ,9,3);
printf("相同的元素的个数是:%d\n",count);
for(i=0;i<count;i++)
printf("%4d\n",c[i]);
system("pause");
return 0;
}
##最小生成树-Prim算法
普里姆算法(Prim算法)
####算法简单描述
1).输入:一个加权连通图,其中顶点集合为V,边集合为E;
2).初始化:Vnew = {x},其中x为集合V中的任一节点(起始点),Enew = {},为空;
3).重复下列操作,直到Vnew = V:
a.在集合E中选取权值最小的边<u, v>,其中u为集合Vnew中的元素,而v不在Vnew集合当中,并且v∈V(如果存在有多条满足前述条件即具有相同权值的边,则可任意选取其中之一);
b.将v加入集合Vnew中,将<u, v>边加入集合Enew中;
4).输出:使用集合Vnew和Enew来描述所得到的最小生成树。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-xnkZI2pu-1578581536956)(./1578574240722.png)][外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-JBJkNGBw-1578581536957)(./1578574246477.png)]
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-05cbNxbt-1578581536958)(./1578574256627.png)][外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-9Wd0IUNV-1578581536959)(./1578574260253.png)]
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-g9dEwjsJ-1578581536960)(./1578574265109.png)][外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-eh2HIb54-1578581536961)(./1578574268240.png)]
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-MRte4c8C-1578581536962)(./1578574272946.png)][外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-4sDvaUo1-1578581536963)(./1578574276646.png)]
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-C9dShjfd-1578581536964)(./1578574282115.png)][外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-tUpHPMYX-1578581536965)(./1578574285320.png)]
####简单证明prim算法
反证法:假设prim生成的不是最小生成树
1).设prim生成的树为G0
2).假设存在Gmin使得cost(Gmin)<cost(G0) 则在Gmin中存在<u,v>不属于G0
3).将<u,v>加入G0中可得一个环,且<u,v>不是该环的最长边(这是因为<u,v>∈Gmin)
4).这与prim每次生成最短边矛盾
5).故假设不成立,命题得证.
####代码
#include<iostream>
using namespace std;
#define MAX 100
#define MAXCOST 0x7fffffff
int graph[MAX][MAX];
int prim(int graph[][MAX], int n)
{
int lowcost[MAX];//i-max的距离
int mst[MAX];//表示i的起点是谁
int i, j, min, minid, sum = 0;
for (i = 2; i <= n; i++)
{
lowcost[i] = graph[1][i];
mst[i] = 1;
}//默认起点是1
mst[1] = 0;
for (i = 2; i <= n; i++)
{
min = MAXCOST;
minid = 0;
for (j = 2; j <= n; j++)
{
if (lowcost[j] < min && lowcost[j] != 0)
{
min = lowcost[j];
minid = j;
}
}//找出距离1最近的点
cout << "V" << mst[minid] << "-V" << minid << "=" << min << endl;
sum += min;
lowcost[minid] = 0;
for (j = 2; j <= n; j++)
{
if (graph[minid][j] < lowcost[j])
{
lowcost[j] = graph[minid][j];
mst[j] = minid;
}
}//相当于利用mind作为新的起点来更新lowcost[];
}
return sum;
}
int main()
{
int i, j, k, m, n;
int x, y, cost;
cin >> m >> n;//m=顶点的个数,n=边的个数
//初始化图G
for (i = 1; i <= m; i++)
{
for (j = 1; j <= m; j++)
{
graph[i][j] = MAXCOST;
}
}
//构建图G
for (k = 1; k <= n; k++)
{
cin >> i >> j >> cost;
graph[i][j] = cost;
graph[j][i] = cost;
}
//求解最小生成树
cost = prim(graph, m);
//输出最小权值和
cout << "最小权值和=" << cost << endl;
return 0;
}
####实例输入
6 10
1 2 6
1 3 1
1 4 5
2 3 5
2 5 3
3 4 5
3 5 6
3 6 4
4 6 2
5 6 6
####实例输出
V1-V3=1
V3-V6=4
V6-V4=2
V3-V2=5
V2-V5=3
最小权值和=15
###warshall算法 最短路径
算法推荐博客
####定义概览
Floyd-Warshall算法(Floyd-Warshall algorithm)是解决任意两点间的最短路径的一种算法,可以正确处理有向图或负权的最短路径问题,同时也被用于计算有向图的传递闭包。Floyd-Warshall算法的时间复杂度为O(N3),空间复杂度为O(N2)。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-demICG12-1578581536966)(./1578574890960.png)]
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-WdVOVcKi-1578581536971)(./1578574896401.png)]
####代码
#include <stdio.h>
int main()
{
int e[10][10],k,i,j,n,m,t1,t2,t3;
int inf=99999999; //用inf(infinity的缩写)存储一个我们认为的正无穷值
//读入n和m,n表示顶点个数,m表示边的条数
scanf("%d %d",&n,&m);
//初始化
for(i=1;i<=n;i++)
for(j=1;j<=n;j++)
if(i==j) e[i][j]=0;
else e[i][j]=inf;
//读入边
for(i=1;i<=m;i++)
{
scanf("%d %d %d",&t1,&t2,&t3);
e[t1][t2]=t3;
}
//Floyd-Warshall算法核心语句
for(k=1;k<=n;k++)
for(i=1;i<=n;i++)
for(j=1;j<=n;j++)
if(e[i][j]>e[i][k]+e[k][j] )
e[i][j]=e[i][k]+e[k][j];
//输出最终的结果
for(i=1;i<=n;i++)
{
for(j=1;j<=n;j++)
{
printf("%10d",e[i][j]);
}
printf("\n");
}
return 0;
}
###主定理的运用
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-harG2NZU-1578581536973)(./1578576929018.png)]
###增长次数比较
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gmqmADsQ-1578581536976)(./1578577477553.png)]
###递归斐波那契
public static long method(int n) {
if(n<2) {
return 1;
}else {
long a=method(n-1) + method(n-2);
return a;
}
}
###非递归斐波那契
private static int method2(int n) {
// TODO Auto-generated method stub
//a,b,c
int a=1;
int b=1;
int c = 0;
for(int i=1;i<n;i++) {
c=a+b;
a=b;
b=c;
}
return c;
}
###Floyd算法
#include
int main()
{
int e[10][10],k,i,j,n,m,t1,t2,t3;
int inf=99999999; //用inf(infinity的缩写)存储一个我们认为的正无穷值
//读入n和m,n表示顶点个数,m表示边的条数
scanf("%d %d",&n,&m);
//初始化
for(i=1;i<=n;i++)
for(j=1;j<=n;j++)
if(i==j) e[i][j]=0;
else e[i][j]=inf;
//读入边
for(i=1;i<=m;i++)
{
scanf("%d %d %d",&t1,&t2,&t3);
e[t1][t2]=t3;
}
//Floyd-Warshall算法核心语句
for(k=1;k<=n;k++)
for(i=1;i<=n;i++)
for(j=1;j<=n;j++)
if(e[i][j]>e[i][k]+e[k][j] )
e[i][j]=e[i][k]+e[k][j];
//输出最终的结果
for(i=1;i<=n;i++)
{
for(j=1;j<=n;j++)
{
printf("%10d",e[i][j]);
}
printf("\n");
}
return 0;
}
###排序算法
排序算法总结文章连接:点击即可
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-9Sj9gF6v-1578581536977)(./1578579539734.png)]
####冒泡排序(BubbleSort)
####基本思想:
两个数比较大小,较大的数下沉,较小的数冒起来。
####过程:
比较相邻的两个数据,如果第二个数小,就交换位置。
从后向前两两比较,一直到比较最前两个数据。最终最小数被交换到起始的位置,这样第一个最小数的位置就排好了。
继续重复上述过程,依次将第2.3…n-1个最小数排好位置。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-JNRrd4fe-1578581536978)(./1578579654701.png)]
冒泡排序 平均时间复杂度:O(n2)
public static void BubbleSort(int [] arr){
int temp;//临时变量
for(int i=0; i<arr.length-1; i++){ //表示趟数,一共arr.length-1次。
for(int j=arr.length-1; j>i; j--){
if(arr[j] < arr[j-1]){
temp = arr[j];
arr[j] = arr[j-1];
arr[j-1] = temp;
}
}
}
}
####选择排序(SelctionSort)
####基本思想:
在长度为N的无序数组中,第一次遍历n-1个数,找到最小的数值与第一个元素交换;
第二次遍历n-2个数,找到最小的数值与第二个元素交换;
。。。
第n-1次遍历,找到最小的数值与第n-1个元素交换,排序完成。
####过程:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-FC8ESCXf-1578581536979)(./1578579712012.png)]
选择排序 平均时间复杂度:O(n2)
public static void select_sort(int array[],int lenth){
for(int i=0;i<lenth-1;i++){
int minIndex = i;
for(int j=i+1;j<lenth;j++){
if(array[j]<array[minIndex]){
minIndex = j;
}
}
if(minIndex != i){
int temp = array[i];
array[i] = array[minIndex];
array[minIndex] = temp;
}
}
}
####插入排序(Insertion Sort)
####基本思想:
在要排序的一组数中,假定前n-1个数已经排好序,现在将第n个数插到前面的有序数列中,使得这n个数也是排好顺序的。如此反复循环,直到全部排好顺序。
####过程:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-VHHpo9WR-1578581536980)(./1578579783948.png)]
插入排序
平均时间复杂度:O(n2)
public static void insert_sort(int array[],int lenth){
int temp;
for(int i=0;i<lenth-1;i++){
for(int j=i+1;j>0;j--){
if(array[j] < array[j-1]){
temp = array[j-1];
array[j-1] = array[j];
array[j] = temp;
}else{ //不需要交换
break;
}
}
}
}
####快速排序(Quicksort)
####基本思想:(分治)
先从数列中取出一个数作为key值;
将比这个数小的数全部放在它的左边,大于或等于它的数全部放在它的右边;
对左右两个小数列重复第二步,直至各区间只有1个数。
####辅助理解:挖坑填数
初始时 i = 0; j = 9; key=72
由于已经将a[0]中的数保存到key中,可以理解成在数组a[0]上挖了个坑,可以将其它数据填充到这来。
从j开始向前找一个比key小的数。当j=8,符合条件,a[0] = a[8] ; i++ ; 将a[8]挖出再填到上一个坑a[0]中。
这样一个坑a[0]就被搞定了,但又形成了一个新坑a[8],这怎么办了?简单,再找数字来填a[8]这个坑。
这次从i开始向后找一个大于key的数,当i=3,符合条件,a[8] = a[3] ; j-- ; 将a[3]挖出再填到上一个坑中。
数组:72 - 6 - 57 - 88 - 60 - 42 - 83 - 73 - 48 - 85
0 1 2 3 4 5 6 7 8 9
此时 i = 3; j = 7; key=72
再重复上面的步骤,先从后向前找,再从前向后找。
从j开始向前找,当j=5,符合条件,将a[5]挖出填到上一个坑中,a[3] = a[5]; i++;
从i开始向后找,当i=5时,由于i==j退出。
此时,i = j = 5,而a[5]刚好又是上次挖的坑,因此将key填入a[5]。
数组:48 - 6 - 57 - 88 - 60 - 42 - 83 - 73 - 88 - 85
0 1 2 3 4 5 6 7 8 9
可以看出a[5]前面的数字都小于它,a[5]后面的数字都大于它。因此再对a[0…4]和a[6…9]这二个子区间重复上述步骤就可以了。
<数组:48 - 6 - 57 - 42 - 60 - 72 - 83 - 73 - 88 - 85
0 1 2 3 4 5 6 7 8 9
平均时间复杂度:O(N*logN)
public static void quickSort(int a[],int l,int r){
if(l>=r)
return;
int i = l; int j = r; int key = a[l];//选择第一个数为key
while(i<j){
while(i<j && a[j]>=key)//从右向左找第一个小于key的值
j--;
if(i<j){
a[i] = a[j];
i++;
}
while(i<j && a[i]<key)//从左向右找第一个大于key的值
i++;
if(i<j){
a[j] = a[i];
j--;
}
}
//i == j
a[i] = key;
quickSort(a, l, i-1);//递归调用
quickSort(a, i+1, r);//递归调用
}
###利用选择排序 E,X,A,M,P,L,E ``` import java.util.Arrays;
public class T {
public static void main(String[] args) {
char[] a = { ‘e’, ‘x’, ‘a’, ‘m’, ‘p’, ‘l’, ‘e’ };
quickSort(a);
System.out.println(Arrays.toString(a));
}
private static void quickSort(char[] a) {
quickSort(a, 0, a.length - 1);
}
private static void quickSort(char[] a, int low, int high) {
if (low < high) {
int pivotPosition = partition(a, low, high);
quickSort(a, low, pivotPosition - 1);
quickSort(a, pivotPosition + 1, high);
}
}
private static int partition(char[] a, int low, int high) {
char pivot = a[low];
while (low < high) {
while (low < high && a[high] >= pivot) {
high–;
}
if (low < high) {
a[low] = a[high];
low++;
}
while (low < high && a[low] <= pivot) {
low++;
}
if (low < high) {
a[high] = a[low];
high–;
}
}
a[low] = pivot;
return low;
}
}
###冒泡排序
![Alt text](./1578580238097.png)
![Alt text](./1578580300851.png)
![Alt text](./1578580310413.png)
#include <stdio.h>
#define NUM 5
void arrsort(int[],int);
void arrout(int[],int);
main(){
int a[NUM]={16,25,9,90,23};
arrout(a,NUM);//输出a数组中原始数据
arrsort(a,NUM);//对a数组中的数进行排序
arrout(a,NUM);//输出排序后a数组中的数据
}
void arrsort(int a[],int n){
for(int i=0;i<n-1;i++){
for(int j=0;j<n-1-i;j++){
if(a[j]>a[j+1]){
int temp =a[j+1];
a[j+1] = a[j];
a[j] = temp;
}
}
}
}
void arrout(int a[],int n){
int i;
for(i=0;i<n;i++){
printf("%3d",a[i]);
}
printf("\n");
}
<br>
###背包问题
[动态规划](https://www.cnblogs.com/yun-an/p/11037618.html)
[贪心算法](https://blog.csdn.net/practical_sharp/article/details/102257213)
###估计题目
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-q0PksZcr-1578581536981)(./1578581387910.png)]
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-kcBdb7KU-1578581536982)(./1578581394276.png)]
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-m7ubgl2X-1578581536983)(./1578581400006.png)]