草庐IT

一元稀疏多项式计算器

<vince> 2023-04-10 原文

别把自己太当回事,要把你做的事当回事!💓💓💓

文章目录

1、问题描述

问题描述】设计一个一元稀疏多项式得简单计算器。
基本要求】采用带头结点的单链表存储一元稀疏多项式,设计实现简单计算器以下基本功能。
(1)输入并且建立多项式。
(2)一元多项式的相加运算。
(3)一元多项式的相减运算。
(4)一元多项式的相乘运算。
(5)一元多项式的求导运算。

👇👇👇
💘💘💘知识连线时刻(直接点击即可)

  🎉🎉🎉复习回顾🎉🎉🎉
    链表复习回顾

测试数据
输入数据:第一个多项式A,共5项,分别是(6,0)(2,1)(8,7)(4,15)(3,18);第二个多项式B,共3项,分别是(7,1)(2,6)(-8,7)。
输出数据:
A多项式的结果:6 + 2X + 8X ^ 7 + 4X ^ 15 + 3X ^ 18
B多项式的结果:7X + 2X ^ 6 - 8X ^ 7
多项式A + B的结果:6 + 9X + 2X ^ 6 + 4X ^ 15 + 3X ^ 18
多项式A - B的结果:6 - 5X - 2X ^ 6 + 16X ^ 7 + 4X ^ 15 + 3X ^ 18
多项式A * B的结果:42X + 14X ^ 2 + 12X ^ 6 - 44X ^ 7 + 40X ^ 8 + 16X ^ 13 - 64X ^ 14 + 28X ^ 16 + 21X ^ 19 + 8X ^ 21 - 32X ^ 22 + 6X ^ 24 - 24X ^ 25
多项式A的导数结果:2 + 56X ^ 6 + 60X ^ 14 + 54X ^ 17

2、问题分析

这里小编提供一种思路哈:
1、首先我们要清楚这是需要建立带哨兵位头节点的单链表;每个节点里面存两个数据,一个是指数,一个是系数。
2、其次就是需要注意多项式输出的时候需要注意的各种情况,这里需要考虑完整,例如:系数小于0时,指数为1时等等。
3、接下来就到了,多项式计算环节,这里就需要用到数学上计算功底了,这里面需要注意。

3、源代码图解分析

具体细节操作还得看源码:(小编这里先拿图片对各个分文件里面重要环节做详解,之后再附上完整源代码

首先看头文件中的代码:(Many.h)

其次来看看源文件:(Many.cpp)

输出打印多项式函数

插多项式入函数:

A+B函数:

A-B函数:

A*B函数:

A的导数函数:

最后是主函数源代码(test.cpp)

4、完整源代码

上面的图解分析,小编是对该程序分文件操作,然后小编对其每个文件中重点函数进行分析展现,这里为了大家方便看源代码,小编也将其放在一个源文件中并且依然能正确运行:


#include <stdio.h>
#include <assert.h>
#include <stdlib.h>

typedef int SLDataType;

typedef struct List
{
	SLDataType data;//存系数
	SLDataType val;//存指数
	struct List* next;
}LS, LN;

LS* ListInit();//链表初始化(实际建立哨兵位头节点)

void ListPushBack(LS* Phead, SLDataType x, SLDataType y);//尾插构建链表

void PrintPloy(LS* Phead);//输出多项式

void AddPloy(LS* Pheada, LS* Pheadb, LS* AddPhead);//A+B

void SubPloy(LS* Pheada, LS* Pheadb, LS* SubPhead);//A-B

void MulPloy(LS* Pheada, LS* Pheadb, LS* MulPhead);//A*B

void DerPloy(LS* Pheada, LS* DerPhead);//A的导数

//创建结点
LS* BuyNode(SLDataType x, SLDataType y)
{
	LS* newnode = (LS*)malloc(sizeof(LS));
	if (newnode == NULL)
	{
		printf("malloc fail\n");
		exit(-1);
	}
	newnode->data = x;//存系数
	newnode->val = y;//存指数
	newnode->next = NULL;
	return newnode;
}

//初始化建立哨兵位头节点
LS* ListInit()
{
	LS* Phead = BuyNode(0, 0);
	Phead->next = Phead;
	return Phead;
}
//尾插构建链表
void ListPushBack(LS* Phead, SLDataType x, SLDataType y)
{
	assert(Phead);
	LS* newnode = BuyNode(x, y);
	//单向循环,因此找尾部
	LS* tail = Phead->next;
	while (tail->next != Phead)
	{
		tail = tail->next;
	}
	tail->next = newnode;
	newnode->next = Phead;
}
//输出打印多项式
void PrintPloy(LS* Phead)
{
	int count = 1;
	LS* tail = Phead->next;
	assert(Phead);
	while (tail != Phead)
	{
		if (tail->data == 0)//系数为0
		{
			tail = tail->next;
			continue;
		}
		if (tail->val == 0 )//指数为0
		{
			printf("%d", tail->data);
		}
		if (tail->data > 0 && tail->val > 0)//系数大于0输出
		{
			if (count != 1)
				printf(" + ");
			if (tail->val == 1)//指数为1时输出
			{
				if (tail->data == 1.0)
					printf("X");//系数为1时输出
				else
					printf("%dX", tail->data);
			}
			else//指数大于1的时候输出
			{
				if (tail->data == 1.0)
					printf("X^%d", tail->val);
				else
					printf("%dX^%d", tail->data, tail->val);
			}
		}
		if (tail->data < 0 && tail->val > 0)//系数小于0输出
		{
			if (count != 1)
				printf(" - ");
			if (tail->val == 1)//指数为1时输出
			{
				if (tail->data == 1.0)
					printf("X");//系数为1时输出
				else
					printf("%dX", tail->data * (-1));
			}
			else//指数大于1的时候输出
			{
				if (tail->data == 1.0)
					printf("X^%d", tail->val);
				else
					printf("%dX^%d", tail->data * (-1), tail->val);
			}
		}
		count = 0;
		tail = tail->next;
	}
}



int cmp(SLDataType a, SLDataType b)
{
	if (a > b)
		return 1;
	else if (a == b)
		return 0;
	else
		return -1;
}
//将运算后的结果插入
void Insert(LS* AddPhead, SLDataType x, SLDataType y)//x是指数,y是系数
{
	LS* cur = AddPhead;
	assert(AddPhead);
	while (cur->next != AddPhead && cmp(cur->next->val, x) < 0)
	{
		cur = cur->next;
	}
	if (cur->next != AddPhead && cmp(cur->next->val, x) == 0)
	{
		cur->next->data += y;
	}
	else
	{
		LS* newnode = (LS*)malloc(sizeof(LS));
		if (newnode == NULL)
		{
			printf("malloc fail\n");
			exit(-1);
		}
		newnode->data = y;
		newnode->val = x;
		newnode->next = cur->next;
		cur->next = newnode;
	}
}
//实现A+B
void AddPloy(LS* Pheada, LS* Pheadb, LS* AddPhead)
{
	LS* str = Pheadb->next;
	LS* cur = Pheadb->next;
	assert(Pheada && Pheadb && AddPhead);
	for (cur = Pheada->next; cur != Pheada; cur = cur->next)
	{
		Insert(AddPhead, cur->val, cur->data);
	}
	for (str = Pheadb->next; str != Pheadb; str = str->next)
	{
		Insert(AddPhead, str->val, str->data);
	}
}

//A-B
void SubPloy(LS* Pheada, LS* Pheadb, LS* SubPhead)
{
	LS* str = Pheadb->next;
	LS* cur = Pheada->next;
	assert(Pheada && Pheadb && SubPhead);
	for (cur = Pheada->next; cur != Pheada; cur = cur->next)//将A插入
	{
		Insert(SubPhead, cur->val, cur->data);
	}
	for (str = Pheadb->next; str != Pheadb; str = str->next)//将B插入
	{
		str->data = -(str->data);//将B的系数变为负值
		Insert(SubPhead, str->val, str->data);
		str->data = -(str->data);//将B的系数变为原值
	}
}
//A*B
void MulPloy(LS* Pheada, LS* Pheadb, LS* MulPhead)
{
	int a = 0, b = 0;
	LS* cur = Pheada->next;
	LS* str = Pheadb->next;
	LS* prev = Pheada;
	assert(Pheada && Pheadb && MulPhead);
	for (cur = Pheada->next; cur != Pheada; cur = cur->next)
	{
		for (str = Pheadb->next; str != Pheadb; str = str->next)
		{
			a = (cur->data) * (str->data);//系数相乘
			b = (cur->val) + (str->val);//指数相加
			Insert(MulPhead, b, a);//将运算后的结果插入
		}
	}
}
//A的导数
void DerPloy(LS* Pheada, LS* DerPhead)
{
	LS* cur = Pheada->next;
	assert(Pheada);
	for (cur = Pheada->next; cur != Pheada; cur = cur->next)
	{
		if (cur->val == 0)//判断指数是否为0
		{
			cur->data = 0;
		}
		else
		{
			cur->data *= cur->val;//系数与指数相乘
			cur->val = cur->val - 1;
		}
		Insert(DerPhead, cur->val, cur->data);//将结果插入
	}
}

int main()
{
	LS* Pa = ListInit();//存放A多项式
	LS* Pb = ListInit();//存放B多项式
	LS* Add = ListInit();//存放A+B多项式
	LS* Sub = ListInit();//存放A-B多项式
	LS* Mul = ListInit();//存放A*B多项式
	LS* Der = ListInit();//存放A的导数
	size_t n = 0, m = 0;
	int a = 0, b = 0, c = 0, d = 0;
	printf("A多项式的项数:");
	scanf("%d", &n);//A的项数
	for (size_t i = 1; i <= n; i++)
	{
		printf("A多项式的第%d项系数和指数:", i);
		scanf("%d %d", &a, &b);
		ListPushBack(Pa, a, b);
	}
	printf("A多项式的输出:");
	PrintPloy(Pa);
	printf("\n");
	printf("B多项式的项数:");
	scanf("%d", &m);
	for (size_t i = 1; i <= m; i++)
	{
		printf("B多项式的第%d项系数和指数:", i);
		scanf("%d %d", &c, &d);
		ListPushBack(Pb, c, d);
	}
	printf("B多项式的输出:");
	PrintPloy(Pb);
	printf("\n");
	AddPloy(Pa, Pb, Add);
	printf("多项式A+B的输出:");
	PrintPloy(Add);
	printf("\n");
	printf("多项式A-B的输出:");
	SubPloy(Pa, Pb, Sub);
	PrintPloy(Sub);
	printf("\n");
	printf("多项式A*B的输出:");
	MulPloy(Pa, Pb, Mul);
	PrintPloy(Mul);
	printf("\n");
	printf("多项式A的导数输出:");
	DerPloy(Pa, Der);
	PrintPloy(Der);
	printf("\n");
	return 0;
}

运行截图:

结语

这次也是一个小小的实验,是一元稀疏多项式计算器的实现,能够实现对两个多项式进行简单的计算处理即可。以上只是vince提供的一种方法,当然也还有很多不足之处,也望大家批评指正。

🎉🎉🎉以上代码均可运行,所用编译环境为 vs2019 ,运行时注意加上编译头文件#define _CRT_SECURE_NO_WARNINGS 1

有关一元稀疏多项式计算器的更多相关文章

  1. ruby-on-rails - 使用一系列等级计算字母等级 - 2

    这里是Ruby新手。完成一些练习后碰壁了。练习:计算一系列成绩的字母等级创建一个方法get_grade来接受测试分数数组。数组中的每个分数应介于0和100之间,其中100是最大分数。计算平均分并将字母等级作为字符串返回,即“A”、“B”、“C”、“D”、“E”或“F”。我一直返回错误:avg.rb:1:syntaxerror,unexpectedtLBRACK,expecting')'defget_grade([100,90,80])^avg.rb:1:syntaxerror,unexpected')',expecting$end这是我目前所拥有的。我想坚持使用下面的方法或.join,

  2. 计算机毕业设计ssm+vue基本微信小程序的小学生兴趣延时班预约小程序 - 2

    项目介绍随着我国经济迅速发展,人们对手机的需求越来越大,各种手机软件也都在被广泛应用,但是对于手机进行数据信息管理,对于手机的各种软件也是备受用户的喜爱小学生兴趣延时班预约小程序的设计与开发被用户普遍使用,为方便用户能够可以随时进行小学生兴趣延时班预约小程序的设计与开发的数据信息管理,特开发了小程序的设计与开发的管理系统。小学生兴趣延时班预约小程序的设计与开发的开发利用现有的成熟技术参考,以源代码为模板,分析功能调整与小学生兴趣延时班预约小程序的设计与开发的实际需求相结合,讨论了小学生兴趣延时班预约小程序的设计与开发的使用。开发环境开发说明:前端使用微信微信小程序开发工具:后端使用ssm:VU

  3. ruby - 如何计算 Liquid 中的变量 +1 - 2

    我对如何计算通过{%assignvar=0%}赋值的变量加一完全感到困惑。这应该是最简单的任务。到目前为止,这是我尝试过的:{%assignamount=0%}{%forvariantinproduct.variants%}{%assignamount=amount+1%}{%endfor%}Amount:{{amount}}结果总是0。也许我忽略了一些明显的东西。也许有更好的方法。我想要存档的只是获取运行的迭代次数。 最佳答案 因为{{incrementamount}}将输出您的变量值并且不会影响{%assign%}定义的变量,我

  4. ruby - 使用 Ruby,计算 n x m 数组的每一列中有多少个 true 的简单方法是什么? - 2

    给定一个nxmbool数组:[[true,true,false],[false,true,true],[false,true,true]]有什么简单的方法可以返回“该列中有多少个true?”结果应该是[1,3,2] 最佳答案 使用转置得到一个数组,其中每个子数组代表一列,然后将每一列映射到其中的true数:arr.transpose.map{|subarr|subarr.count(true)}这是一个带有inject的版本,应该在1.8.6上运行,没有任何依赖:arr.transpose.map{|subarr|subarr.in

  5. arrays - 计算数组中的匹配元素 - 2

    给定两个大小相等的数组,如何找到不考虑位置的匹配元素的数量?例如:[0,0,5]和[0,5,5]将返回2的匹配项,因为有一个0和一个5共同;[1,0,0,3]和[0,0,1,4]将返回3的匹配项,因为0有两场,1有一场;[1,2,2,3]和[1,2,3,4]将返回3的匹配项。我尝试了很多想法,但它们都变得相当粗糙和令人费解。我猜想有一些不错的Ruby习惯用法,或者可能是一个正则表达式,可以很好地回答这个解决方案。 最佳答案 您可以使用count完成它:a.count{|e|index=b.index(e)andb.delete_at

  6. ruby-on-rails - 如何计算 Ruby/Rails 中 JSON 对象的数量 - 2

    Ruby中如何“一般地”计算以下格式(有根、无根)的JSON对象的数量?一般来说,我的意思是元素可能不同(例如“标题”被称为其他东西)。没有根:{[{"title":"Post1","body":"Hello!"},{"title":"Post2","body":"Goodbye!"}]}根包裹:{"posts":[{"title":"Post1","body":"Hello!"},{"title":"Post2","body":"Goodbye!"}]} 最佳答案 首先,withoutroot代码不是有效的json格式。它将没有包

  7. ruby - 如何计算自 Ruby 中给定日期以来的周数? - 2

    目标我正在尝试计算自给定日期以来周的距离,而无需跳过任何步骤。我更喜欢用普通的Ruby来做,但ActiveSupport无疑是一个可以接受的选择。我的代码我写了以下内容,这似乎可行,但对我来说似乎还有很长的路要走。require'date'DAYS_IN_WEEK=7.0defweeks_sincedate_stringdate=Date.parsedate_stringdays=Date.today-dateweeks=days/DAYS_IN_WEEKweeks.round2endweeks_since'2015-06-15'#=>32.57ActiveSupport的#weeks

  8. 最新版人脸识别小程序 图片识别 生成二维码签到 地图上选点进行位置签到 计算签到距离 课程会议活动打卡日常考勤 上课签到打卡考勤口令签到 - 2

    技术选型1,前端小程序原生MINA框架cssJavaScriptWxml2,管理后台云开发Cms内容管理系统web网页3,数据后台小程序云开发云函数云开发数据库(基于MongoDB)云存储4,人脸识别算法基于百度智能云实现人脸识别一,用户端效果图预览老规矩我们先来看效果图,如果效果图符合你的需求,就继续往下看,如果不符合你的需求,可以跳过。1-1,登录注册页可以看到登录页有注册入口,注册页如下我们的注册,需要管理员审核,审核通过后才可以正常登录使用小程序1-2,个人中心页登录成功以后,我们会进入个人中心页我们在个人中心页可以注册人脸,因为我们做人脸识别签到,需要先注册人脸才可以进行人脸比对,进

  9. ruby - 如何计算两个字符串共有的字符数? - 2

    如何计算两个字符串之间的字符交集?例如(假设我们有一个名为String.intersection的方法):"abc".intersection("ab")=2"hello".intersection("hallo")=4好的,男孩女孩们,感谢你们的大量反馈。更多示例:"aaa".intersection("a")=1"foo".intersection("bar")=0"abc".intersection("bc")=2"abc".intersection("ac")=2"abba".intersection("aa")=2一些补充说明:维基百科定义intersection如下:Int

  10. python - 如何计算文件中唯一字符的数量? - 2

    给定一个包含各种语言字符的UTF-8文件,我如何计算它包含的唯一字符的数量,同时排除选定数量的符号(例如:“!”、“@”、"#",".")从这个算起? 最佳答案 这是一个bash解决方案。:)bash$perl-CSD-ne'BEGIN{$s{$_}++forsplit//,q(!@#.)}$s{$_}++||$c++forsplit//;END{print"$c\n"}'*.utf8 关于python-如何计算文件中唯一字符的数量?,我们在StackOverflow上找到一个类似的问题

随机推荐