常用算法案例之动态规划(C语言)

1.最长公共子序列

// 动态规划之最长公共子序列.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"

/***最长公共子序列***/
/*动态规划*/

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<ctype.h>

#ifndef size_c
#define size_c 200
#endif // 预定义字符串的长度

#define EQUAL   1   //EQUAL表示c[i][j]是由c[i-1][j-1]+1来的=此时两个序列有相同的字符
#define UP      2   //UP表示c[i][j]是由c[i-1][j]来的========此时两个序列没有相同的字符
#define LEFT    3   //LEFT表示c[i][j]是由[ci][j-1]来的======此时两个序列没有相同的字符

//int char1[size_c][size_c];  //定义两个二维数组存放字符串
//int char2[size_c][size_c];  //1存放位置,2存放路径

//int max(int m, int n, int i, int j);
//int print(int i, int j);

/***函数一、判断LCS长度***/
int Lcs_len(char *str1, char *str2, int **char1, int **char2)
{
	//int char1[size_c][size_c] = {0};
	//int char2[size_c][size_c] = {0};

	int m = strlen(str1);
	int n = strlen(str2);  //求出两个数组的边界长度

	int i, j;
	for (i = 0; i <= m; i++)
	{
		char1[i][0] = 0;
	}

	for (j = 0; j <= n; j++)  //初始化边界条件
	{
		char1[0][j] = 0;
	}

	for (i = 1; i <= m; i++)
	{
		for (j = 1; j <= n; j++)
		{
			if (str1[i - 1] == str2[j - 1])
				// 这里使用i-1以及j-1是由于数组的下标从0开始
				//另一种实现方式是逆序实现,对于路径的确定更方便
			{
				char1[i][j] = char1[i - 1][j - 1] + 1;
				char2[i][j] = EQUAL;
			}
			else if (char1[i - 1][j] >= char1[i][j - 1])//在j循环时若字符串不等
			{                                // 则只用判断char中的元素
				char1[i][j] = char1[i - 1][j];
				char2[i][j] = UP;
			}
			else
			{
				char1[i][j] = char1[i][j - 1];
				char2[i][j] = LEFT;
			}
		}


	}
	return char1[m][n];  //递归的最终位存储的数字就是LCS长度
}

/***函数二、输出LCS***/

void Print_Lcs(char *str, int **b, int i, int j)
{
	if (i == 0 || j == 0)
		return;    //递归至边界则扫描完毕
	if (b[i][j] == EQUAL)
	{                     //对于相等的元素,其路径为左上方对角移动
		Print_Lcs(str, b, i - 1, j - 1);
		printf("%c ", str[i - 1]);  //相等的话,原字符序列向前递归一位并打印出字符
	}
	else if (b[i][j] == UP)  //不相等时判断方向:向上则数组向上位移
		Print_Lcs(str, b, i - 1, j);
	else
		Print_Lcs(str, b, i, j - 1); //否则数组下标向左位移一位

}

/***函数三、整合LCS函数***/
void Find_Lcs(char *str1, char *str2)
{
	int i, j, length;
	int len1 = strlen(str1),
		len2 = strlen(str2);
	//申请二维数组
	int **c = (int **)malloc(sizeof(int*) * (len1 + 1));
	int **b = (int **)malloc(sizeof(int*) * (len1 + 1));
	for (i = 0; i <= len1; i++)  这个等号之前没加,导致内存泄漏
	{
		c[i] = (int *)malloc(sizeof(int) * (len2 + 1));
		b[i] = (int *)malloc(sizeof(int) * (len2 + 1));
	}

	//将c[len1][len2]和b[len1][len2]初始化为0
	for (i = 0; i <= len1; i++)
		for (j = 0; j <= len2; j++)
		{
			c[i][j] = 0;
			b[i][j] = 0;
		}

	//计算LCS的长度
	length = Lcs_len(str1, str2, c, b);
	printf("The number of the Longest-Common-Subsequence is %d\n", length);

	//利用数组b输出最长子序列
	printf("The Longest-Common-Subsequence is: ");
	Print_Lcs(str1, b, len1, len2);
	printf("\n");

	//动态内存释放
	for (i = 0; i <= len1; i++)
	{
		free(c[i]);
		free(b[i]);
	}
	free(c);
	free(b);
}

/***LCS测试输出***/

int main(int *argc, int *argv[])
{

	char X[size_c] = "asdfghjkt";
	char Y[size_c] = "yyydfooo";
	int len;
	//printf("please enter your characters:");
	//scanf("%s", X);

	while (strlen(X) > 200) //规定字符串序列的最大长度,此处为200
	{
		printf("what you input is too long, please try again");
	//	scanf("%s\n", X);//超出限制时提醒并重新输入
	}
	//printf("please enter your characters:");
	//scanf("%s", Y);

	while (strlen(Y) > 200)  //长度限制同上
	{
		printf("what you input is too long, please try again");
	//	scanf("%s", Y);
	}

	Find_Lcs(X, Y);  //使用LCS函数输出长度与子序列
	system("pause");
}

 

  • 1
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 《常用算法程序集(C语言描述)第三版》是由美国纽约大学计算机科学系的Andrew Binstock和John Rex所著的一本著名的计算机算法参考手册,该书以C语言描述了我们在日常编程中会用到的一些经典算法和数据结构。该书的内容十分详细且系统,覆盖了排序、查找、哈希、串、树、图等多个领域,并提供了各种算法的代码实现和解释,适合各种开发人员阅读和参考。 该书的主要特点有: 1. 深入浅出的讲解方式:作者通过清晰的语言和可执行的实例来解释复杂算法的工作原理和步骤,为读者提供了深入了解算法底层机理的机会。 2. 多种数据结构和算法的覆盖:该书包括了众多数据结构和算法,如线性表,栈,队列,树,图等等,为读者提供了全面的参考,从而可以解决许多复杂的开发问题。 3. C语言描述:该书使用C语言来描述算法和数据结构,让读者更容易理解和解决编程问题,同时也提高了代码的可读性和可维护性。 总之,《常用算法程序集(C语言描述)第三版》是一本值得阅读和参考的计算机算法及数据结构经典书籍。其内容涵盖广泛、语言简明易懂,不仅可供初学者学习,也可供从事编程工作的专业人士参考。无论是想提高编程能力还是解决编程难题,都可以从该书中获得帮助。 ### 回答2: 《常用算法程序集(C语言描述)第三版》是一本经典的书籍,主要介绍了各种常用算法和数据结构,并且用C语言进行了详细描述。这本书的作者是严蔚敏和吴伟民,是计算机专业的学生和工程师所必读的一本书。 本书主要包括了10个部分,分别是基本的算法、数据结构、数字处理、字符串处理、排序和查找、图形处理、加密与解密、计算几何、动态规划和高级数据结构。每个部分都有详细的讲解和相应的案例,方便读者理解和应用。 此外,本书还详细描述了C语言的语法和常用的函数库,让读者对C语言的使用更加熟练。所有算法和数据结构的代码都是用C语言编写的,方便读者的学习和使用。 本书的亮点是示例代码和习题解答,这些都能够帮助读者更好地理解和掌握学习内容。同时,本书也是一本既适合初学者,又能为专业人员提供不同层次的挑战的优秀教材。 总之,《常用算法程序集(C语言描述)第三版》是一本经典的计算机书籍,一直被广泛应用于计算机科学和工程领域,并且在各种练习和竞赛中都有广泛的应用。无论是学生,还是从事计算机编程的人员都应该把本书放在重要的书单之中。 ### 回答3: 《常用算法程序集(C语言描述)第三版》是一本涉及到常用算法的书籍,作者是王润基。本书主要介绍了常用的排序算法、查找算法、字符串匹配算法等,并给出了相应的C语言实现。本书不仅适合C语言初学者,也适合对算法感兴趣的程序员阅读。 本书的主要内容包括: 1.排序算法。本书介绍的排序算法有冒泡排序、选择排序、插入排序、希尔排序、快速排序、归并排序等,每种算法都给出了详细的C语言实现代码,还给出了对比不同算法时的时间复杂度和空间复杂度。 2.查找算法。本书介绍的查找算法有顺序查找、二分查找、插值查找、斐波那契查找、哈希查找等,同样也给出了具体的C语言实现,介绍了查找算法的时间复杂度和空间复杂度。 3.字符串匹配算法。本书介绍的字符串匹配算法有暴力匹配、KMP算法、Boyer-Moore算法、Sunday算法等,同样也给出了精简的C语言实现代码,仔细分析了每种算法的优缺点以及时间复杂度。 总之,《常用算法程序集(C语言描述)第三版》不仅介绍了常用算法,还通过具体的代码实现,让读者更加深入地理解算法的本质和运行机制。如果您想提升自己的算法能力,这本书是非常不错的选择。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值