Факультет вычислительной математики и кибернетики
ОТЧЕТ
по лабораторной работе № 4
«Аналитические преобразования полиномов
от нескольких переменных»
Выполнил: студент группы 8210
__________________Баранов А.М. Проверил:
__________________Сидоров С.В.
Нижний Новгород
2011 г.
Оглавление
Аннотация. 3
Введение. 3
Постановка задачи. 3
Руководство пользователя. 4
Руководство программиста. 6
Описание структуры хранения данных. 6
Описание программы.. 7
Описание алгоритма. 7
Заключение. 10
Литература. 11
Приложение (исходный код программы) 12
Аннотация
В данной работе рассматриваются вопросы реализации некоторых базовых аналитических преобразований и операций над полиномами от нескольких переменных.
Для хранения полиномов используются однонаправленные динамические списки.
Введение
Списки являются чрезвычайно гибкой структурой, так как их легко сделать большими или меньшими, и их элементы доступны для вставки или удаления в любой позиции списка. Списки также можно объединять или разбивать на меньшие списки.
Списки регулярно используются в приложениях, например в программах информационного поиска, трансляторах программных языков или при моделировании различных процессов. Методы управления памятью в современных вычислительных системах широко используют технику обработки списков.
Постановка задачи
Лабораторная работа предполагает программную реализацию системы аналитических преобразований полиномов от трех переменных. Требуется реализовать следующие операции в аналитическом представлении исходных данных и результатов:
· ввод полиномов;
· вывод полиномов;
· удаление полиномов;
· приведение полиномов
· сложение полиномов;
· Вычитание полиномов;
· Производная от полинома;
· Умножение полиномов
Например, пусть имеется два полинома: P(x,y,z)=x7y2z+3x2z-6y2-3z9 и Q(x,y,z)=-7x2z+6y2+5. В результате выполнения операции сложения должен быть получен полином R(x,y,z)=x7y2z-4x2z-3z9+5.
Программа должна обеспечивать диалоговый интерфейс с пользователем обеспечивающий ввод исходных данных из заранее подготовленного файла с заданным именем.
Руководство пользователя
После запуска исполняемого файла программы вы увидите главное окно программы (рис. 1).
Рис. 1. Главное окно программы.
Основные операции.
Операция сложения/вычитания/умножения полиномов:
· Выбирается пункт меню 1/2/4;
· Вводится имя файла с первым полиномом;
· Выводится исходный неприведенный полином;
· Выводится приведенный полином;
· Вводится имя файла со вторым полиномом;
· Выводится исходный неприведенный полином;
· Выводится приведенный полином;
· Выводится результат в приведенном виде.
Рис. 2. Выполнение операции сложения.
Рис.3. Результат операции вычитания.
Рис.4. Результат операции умножения.
Операция вычисления производной:
· Выбирается пункт меню 3;
· Вводится имя файла с полиномом;
· Выводится исходный неприведенный полином;
· Выводится приведенный полином;
· Выводится результат в приведенном виде.
Рис.5.Результат вычисления производной.
Руководство программиста
Описание структуры хранения данных
При манипуляции разреженными полиномами эффективной структурой хранения являются списки.
В качестве структуры хранения данных полинома был выбран однонаправленный динамический список. Каждый узел содержит два элемента содержащие атрибуты монома
elem - коэфффициент монома
pol – свертка монома
под сверткой монома имеется в виду вычисляемое целое число без знака вычисляемое в соответствии со следующими правилами
· фиксируется старшинство переменных.
· Будем считать, что x - самая старшая переменная, затем следует y, затем z.
· Для каждого монома определим его "свернутую степень" (индекс).
· Для монома xAyBzC+n. индекс равен A*1000+B*100+C*10+n (по условию задачи A, B и C не выше 9).
· Старшим считается моном с большей свернутой степенью.
Свертка монома служит в дальнейшем ключом для идентификации монома
Кроме этого в каждый узел входит ссылка на следующий элемент списка
Для хранения промежуточного результата синтаксического разбора исходной строки используется массив Mpars
unsigned int Mpars[500];
node *Bfirst, *Blast;
node *BRes; //Указатель на найденное звено списка.
node *Afirst, *Alast;
node *ARes; //Указатель на найденное звено списка.
struct node {
int elem;
unsigned int pol;
node *next;
};
Для хранения полинома класс Spisok
class Spisok
{
private:
int N_monom;
public:
Spisok() {phead = new (node); (*phead).next=NULL;} //Конструктор.
~Spisok() { delete phead; } //Деструктор.
void Initial(node **first, node **last);
int Empty(node *first);
void Add(node **last,int elem, unsigned int pol);
void Show(node *first);
int Poisk_Eqv (node *first,unsigned int pol,node **last);
int Poisk_Less (node *first,int el,node **last);
void Izm (node *Res, int el,unsigned int pol);
void DelAll(node *first, node **last);
void AddLess (node **Res, int el,unsigned int pol);
int Next (node *first,int el,node **last);
void Del1 ();
};
Описание программы
Программа использует класс Spisok
class Spisok
{
private:
node *phead; //указатель на начало списка.
node *pend;
node *cursosr;
node *Res; //указатель на найденное звено.
int N_monom;
public:
Spisok() {phead = new (node); (*phead).next=NULL;} //конструктор
~Spisok() { delete phead; } //деструктор.
void Initial(node **first, node **last);
int Empty(node *first);
void Add(node **last,int elem, unsigned int pol);
void Show(node *first);
int Poisk_Eqv (node *first,unsigned int pol,node **last);
int Poisk_Less (node *first,int el,node **last);
void Izm (node *Res, int el,unsigned int pol);
void DelAll(node *first, node **last);
void AddLess (node **Res, int el,unsigned int pol);
int Next (node *first,int el,node **last);
void Del1 ();
};
Описание алгоритма
Работа программы начинается с вывода основного меню. Для начала работы выбирается пункт
"1.Операции"
При этом осуществляется запрос имени файла, где хранится описание полинома.
Ввод полинома. Ввод полинома реализуется как текстовая строка с формульным выражением, описывающим вводимый полином. Форма задания полинома производится образом, аналогичным примеру:
P='x7y2z1+3x2z1-6y2-3z9 '
Q='-7x2z1+6y2+5'
Коэффициенты мономов имеют целочисленный тип. При вводе полинома производится синтаксический разбор строки и приведение монома к стандартному виду – коэффициент со знаком и свертка.
Вывод полинома. Вывод полинома осуществляется обходом списка и печатью коэффициента и степеней переменных каждого монома. Следует отметить, что голова полинома играет роль барьера при обходе списка: цикл завершается, когда текущим звеном окажется головное звено.
Рекомендуемая форма печати полинома состоит в представлении полинома в виде текстовой строки с формульным выражением, описывающим полином. Например, полином P(x,y,z) из контрольного примера может быть представлен как строка:
P(x,y,z)='x7y2z1-4x2z1-3x9+5'
Реализация алгоритмов обработки полиномов организуется путем следующих шагов
1. Производится считывание исходной строки полинома с stdin ()
2. Выполняется анализ исходной строки на наличие запрещенных символов
3. Организуется однонаправленный список мономов многочлена каждый элемент которого содержит коэффициент монома, свертку степеней переменных и ссылку на следующий элемент. При создании списка производится нормализация полинома по совпадающим сверткам и его реорганизация в соответствии с убыванием модуля сверток
4. Если свернутые степени мономов совпадают, то осуществляется приведение подобных членов: к коэффициенту монома полинома P прибавляется коэффициент монома полинома Q. Если при этом сумма коэффициентов оказывается равной нулю, то соответствующий моном полинома P исключается из списка.
Для полиномов реализованы следующие функции:
· Сложение и вычитание
· Умножение
· Взятие производной. Для сложения полиномов P и Q производится одновременный обход обоих списков, начиная с головных звеньев, по правилу слияния упорядоченных массивов. Сумма формируется на месте первого слагаемого (P). При этом, если свернутая степень текущего монома полинома Q больше свернутой степени текущего монома полинома P, то копия монома полинома Q вставляется в полином P; в полиноме Q осуществляется переход на следующее звено. Если свернутая степень текущего монома полинома P больше свернутой степени текущего монома полинома Q, то в полиноме P осуществляется переход на следующее звено. Если свернутые степени текущих мономов полиномов P и Q совпадают, то осуществляется приведение подобных членов: к коэффициенту монома полинома P прибавляется коэффициент монома полинома Q. Если при этом сумма коэффициентов оказывается равной нулю, то соответствующий моном полинома P исключается из списка. В обоих списках осуществляется переход к следующему звену. Для реализации операций вставки и удаления звеньев в односвязных списках необходимы указатели как текущего, так и предыдущего звеньев.
· Удаление полинома. При удалении полинома предлагается немедленно возвращать освободившуюся память в кучу. В этом случае не образуется мусор из освободившихся звеньев. При удалении полинома осуществляется обход списка аналогично предыдущей операции, и каждое звено возвращается в кучу процедурой Dispose(q).
Заключение
Таким образом, все задачи, поставленные во введении, с успехом достигнуты, а именно: была реализована структура хранения список для прикладной задачи произведение операций над полиномами от трех переменных: сложение, вычитание, умножение полиномов, вычисление производной. Были получены необходимые навыки практического использования и реализации структуры список.
Литература
1. Топп У., Форд У. Структуры данных в C++: Пер. с англ. – М.: ЗАО «Издательство БИНОМ», 1999. – 816 с.: ил.
2. Шилдт Г. Полный справочник по C++:Пер. с англ. – М.: изд. Вильямс, 2007. – 800 с.
3. Гергель В.П., Курс лекций по предмету «Языки программирования».
Приложение (исходный код программы)
Stdafx.h
#ifndef ST_REAL_H
#define ST_REAL_H
#pragma once
#include <stdlib.h >
#include <conio.h>
#include <stdarg.h>
#include <math.h>
#include <windows.h>
#include <process.h>
#include <string.h>
#include <stdio.h>
#include <fstream> // средства для управляемой пользователем обработки файлов.
#include <iostream> // объекты cin, cout, cerr и clog,
#include <iomanip> // управления форматируемым вводом/выводом с помощью параметризованных манипуляторов потока.
using namespace std;
using namespace System;
using std::cin;
using std::cout;
int parse();
void inttoname(int nom,unsigned int pol,char buffer[]);
int Svernut(char s[]);
void AntiPars(int Anom,unsigned int Mpars[]);
void CreatSpis(int Anom);
int getfile();
struct node {
int elem;
unsigned int pol;
node *next;
};
void Proizvodnay(node *first);
class Spisok
{
private:
node *phead; //указатель на начало списка.
node *pend;
node *cursosr;
node *Res; //указатель на найденное звено.
int N_monom;
public:
Spisok() {phead = new (node); (*phead).next=NULL;} //конструктор
~Spisok() { delete phead; } //деструктор.
void Initial(node **first, node **last);
int Empty(node *first);
void Add(node **last,int elem, unsigned int pol);
void Show(node *first);
int Poisk_Eqv (node *first,unsigned int pol,node **last);
int Poisk_Less (node *first,int el,node **last);
void Izm (node *Res, int el,unsigned int pol);
void DelAll(node *first, node **last);
void AddLess (node **Res, int el,unsigned int pol);
int Next (node *first,int el,node **last);
void Del1 ();
};
#endif
Spis06.cpp
#include "stdafx.h"
Spisok A,B;
unsigned int Mpars[500];
//unsigned int Mpars1[500];
node *Bfirst, *Blast;
node *BRes; //Указатель на найденное звено списка.
node *Afirst, *Alast;
node *ARes; //Указатель на найденное звено списка.
void Spisok::Del1 ()
//Удаление звена, на которое указывает ссылка Res.
{
node *q,*q1,*q2;
//Определяем местоположение следующего за удаляемым элемента.
q = (*ARes).next;
if (q!=NULL)//Если удаляемое звено не является последним
{
(*ARes).elem = (*q).elem;
(*ARes).pol = (*q).pol;
(*ARes).next = (*q).next; // "Переписываем" следующий элемент в удаляемый
delete q;
}
else
{
q1 = Afirst;
q2 = (*q1).next;
while (q2!=ARes)
{
q1 = q2;
q2 = (*q2).next;
}
(*q1).next = NULL;
q2 = NULL;
delete ARes;
}
}
void Spisok::Initial(node **first, node **last)
{
*first=new node;
(*first)->next=NULL;
*last=*first;
N_monom=0;
}
int Spisok::Empty(node *first)
{
if (first->next==NULL)
return 1;
else
return 0;
}
void Spisok::Add(node **last,int elem,unsigned int pol)
{
node *tmp=new node;
tmp->elem=elem;
tmp->pol=pol;
tmp->next=NULL;
(*last)->next=tmp;
*last=tmp;
N_monom++;
}
void Spisok::Izm (node *Res, int el,unsigned int pol)
// изменение содержимого найденого звена на el
{
// cout << endl<<"izmres="<< Res->elem<< Res->pol;
cout << endl;
Res->elem = el;
Res->pol = pol;
// (*Res).pol = pol;
}
void Spisok::DelAll(node *first, node **last)
{
node *tmp;
while(first->next!=NULL)
{
tmp=first->next;
first->next=tmp->next;
delete tmp;
}
*last=first;
}
void Spisok::Show(node *first)
{
node *tmp=first->next;
cout <<endl;
while(tmp!=NULL)
{
cout << tmp->elem << " "<< tmp->pol << ", ";
tmp=tmp->next;
}
cout <<endl;
}
int Spisok::Poisk_Eqv (node *first,unsigned int pol,node **last)
//Поиск звена с элементом равным el в списке, заданном указателем
// *first. В случае успешного поиска в Res заносится 1 а в last находится адрес
//звена списка, содержащего элемент el, в случае неуспеха
//в Res помещается 0.
{
unsigned int r;
int rez=0;
node *tmp=first->next;
while(tmp!=NULL)
{
r=tmp->pol;
if(pol==r)
{
// cout << r << " ";
rez=1;
break;
}
tmp=tmp->next;
}
*last=tmp;
return(rez);
}
int Spisok::Poisk_Less (node *first,int sv,node **last)
//Поиск звена с элементом меньше el в списке, заданном указателем
// *first. В случае успешного поиска в Res заносится 1 а в last находится адрес
//звена списка, содержащего элемент меньше el, в случае неуспеха
//в Res помещается 0.
{
int rez=0;
int r;
node *tmp=first->next;
while(tmp!=NULL)
{
r=tmp->pol;
if(sv>r)
{
/// cout << r << " ";
rez=1;
break;
}
tmp=tmp->next;
}
*last=tmp;
return(rez);
}
void Spisok::AddLess (node **Res, int el,unsigned int pol)
//Включение звена с информационным полем el
//перед звеном, на которое указывает *Res.
{
node *q;
// формируем новое звено и перекидываем в него содержимое
q = new (node);
(*q).elem = (**Res).elem;
(*q).pol = (**Res).pol;
(*q).next = (**Res).next;
// заменяем значения в старом звене
(**Res).elem = el;
(**Res).pol = pol;
(**Res).next = q;
N_monom++;
}
int Spisok::Next (node *first,int el,node **last)
//Поиск звена с элементом el в списке, заданном указателем
//phead. В случае успешного поиска в Res находится адрес
//звена списка, содержащего элемент el, в случае неуспеха
//в Res помещается NULL.
{
int r;
node *tmp=first->next;
if (tmp==NULL) return (1);
r=tmp->elem;
*last=tmp;
return(0);
}
void Show(node *first)
{
node *tmp=first->next;
int el;
unsigned int pol;
char buf[256];
while(tmp!=NULL)
{
el =tmp->elem;
pol =tmp->pol;
inttoname(el,pol, buf);
tmp=tmp->next;
}
}
void CreatSpisA(int Anom)
{
int dig=0;
unsigned int sv=0;
int dig0=0;
for(int i=0;i<(Anom/2);i++)
{
dig0=0;
dig=Mpars[i*2];// результаты разбора
sv=Mpars[i*2+1];
A.Poisk_Eqv (Afirst, sv,&ARes);
if(ARes!=NULL)
{
dig0=ARes->elem; // такой уже есть- складываем коэффициенты и не добавляем
dig0=dig0+dig;
/// cout << endl<<"poisk pol = "<< ARes->pol<< endl;
if(dig0!=0)
{
A.Izm (ARes, dig0, sv); // складываем коэффициенты и не добавляем
continue;
}
if(dig0==0) // не добавляем и удаляем
{
A.Del1 ();
continue;
}
}
if(A.Poisk_Less (Afirst, sv,&ARes)== 1) {
A.AddLess (&ARes, dig, sv);
}
else {
A.Add(&Alast,dig,sv);
}
}
cout << endl<<"== Приведенный полином A" << endl;
Show(Afirst);
}
void CreatSpisB(int Anom)
{
int dig=0;
unsigned int sv=0;
int dig0=0;
for(int i=0;i<(Anom/2);i++)
{
dig0=0;
dig=Mpars[i*2];// результаты разбора
sv=Mpars[i*2+1];
B.Poisk_Eqv (Bfirst, sv,&BRes);
if(BRes!=NULL)
{
dig0=BRes->elem; // такой уже есть- складываем коэффициенты и не добавляем
dig0=dig0+dig;
if(dig0!=0)
{
B.Izm (BRes, dig0, sv); // складываем коэффициенты и не добавляем
continue;
}
if(dig0==0) // не добавляем и удаляем
{
B.Del1 ();
continue;
}
}
if(B.Poisk_Less (Bfirst, sv,&BRes)== 1) {
B.AddLess (&BRes, dig, sv);
}
else {
B.Add(&Blast,dig,sv);
}
}
cout << endl<<"== Приведенный полином B" << endl;
Show(Bfirst);
}
void Proizvodnay(node *first)
{
node *tmp=first->next;
int el;
unsigned int pol;
char buf[256];
int x,y,z,i;
cout <<endl;
while(tmp!=NULL)
{
el =tmp->elem;
pol =tmp->pol;
x=pol/1000;
y=(pol%1000)/100;
z=((pol%1000)%100)/10;
if(x>1)
{
el=el*x;
x=x-1;
}
pol=x*1000+y*100+z*10;
if(x!=0) B.Add(&Blast,el,pol);
// поскольку исходный полином приведен добавляем все подряд
tmp=tmp->next;
}
cout<<endl<< "производная"<<endl;
Show(Bfirst);
}
void Proizvodnay1(node *Afirst,node *Bfirst,int Anom)
{
node *Atmp=Afirst->next;
int el=5;
int el1;
int rel;
unsigned int pol, pol1, rpol;
char buf[256];
int x,y,z,i;
int x1,y1,z1;
int rx,ry,rz;
int dig=0;
int imonom=0;
unsigned int MMult[500];
int Mi=0;
Mi=0;
imonom=0;
while(Atmp!=NULL)
{
node *Btmp=Bfirst->next;
el =Atmp->elem;
pol =Atmp->pol;
x=pol/1000;
y=(pol%1000)/100;
z=((pol%1000)%100)/10;
// умножаем коэфиициент на степень x
pol=pol*x;
y=0;
z=0;
if (x>1) x=x-1;
// вычитаем из степени x
// формируем свертку
pol=x*1000+y*100+z*10;
if(x!=0) B.Add(&Blast,el,pol);
// поскольку исходный полином приведен добавляем все подряд
Atmp=Atmp->next;
} // конец цикла по первому полиному
AntiPars(imonom, Mpars);
// нормализуем результат
cout << endl << "Результат "<< endl;
Show(Bfirst); // антипарсер для полинома
getch();
cout<<endl<< "производная"<<endl;
Show(Bfirst);
}
void Mult(node *Afirst,node *Bfirst,int Anom)
{
node *Atmp=Afirst->next;
int el=5;
int el1;
int rel;
unsigned int pol, pol1, rpol;
char buf[256];
int x,y,z,i;
int x1,y1,z1;
int rx,ry,rz;
int dig=0;
int imonom=0;
unsigned int MMult[500];
int Mi=0;
Mi=0;
imonom=0;
while(Atmp!=NULL)
{
node *Btmp=Bfirst->next;
el =Atmp->elem;
pol =Atmp->pol;
x=pol/1000;
y=(pol%1000)/100;
z=((pol%1000)%100)/10;
while(Btmp!=NULL)
{
el1 =Btmp->elem;
pol =Btmp->pol;
x1=pol/1000;
y1=(pol%1000)/100;
z1=((pol%1000)%100)/10;
// перемножаем элементы
rel=el*el1;
// складывавем степени при x y z
rx=x+x1;
ry=y+y1;
rz=z+z1;
// записываем коэффициент и формируем свертку
Mpars[imonom]=rel;// запоминаем результаты разбора
Mpars[imonom+1]=rx*1000+ry*100+rz*10;
imonom=imonom+2;
Btmp=Btmp->next;
} // конец цикла по второму полиному
Atmp=Atmp->next;
} // конец цикла по первому полиному
AntiPars(imonom, Mpars);
// нормализуем результат
A.DelAll(Afirst,&Alast); // чистим список
CreatSpisA(imonom); //создаем список из массива с доп проверками
cout << endl << "Результат "<< endl;
Show(Afirst); // антипарсер для полинома
getch();
}
int main()
{
A.Initial(&Afirst,&Alast);
B.Initial(&Bfirst,&Blast);
char otv;
int nom;
unsigned int svertka;
char buf[300];
int dig;
int Anom=0;
unsigned int sv;
FILE *in;
do
{
cout // << endl << "1. Отладка" << endl
// << "2. Разбор и свертка полинома" << endl
//<< "3. Развертка и вывод полинома" << endl
///<< "4. Создание списка мономов" << endl
//<< "5. Прогон " << endl
//<< "6. Сдвиг полинома" << endl
<< "1. Операция" << endl
//<< "8. " << endl
//<< "9. " << endl
<< "0. Выход" << endl;
cout << '>';
cin >> otv;
switch(otv)
{
case '88': //+++ отладка
break;
case '2': //
//Anom= getfile(); break;
case '3'://+++
// AntiPars(Anom,Mpars); // разворачиваем полином из буфера
break;
case '4'://+++
// CreatSpis(Anom); //создаем список из массива с доп проверками
break;
case '5'://+++
//Show(Afirst); // антипарсер для полинома
break;
case '6'://+++ // сдвиг полинома
break;
case '1'://+++ операция
cout<<endl<< "--1-сложить";
cout<<endl<< "--2-вычесть";
cout<<endl<< "--3-производная";
cout<<endl<< "--4-умножить"<< endl;
cin >> otv;
//прочитали первый и второй полином для операций с двумя операндами
if(otv=='1') //сложить
{
A.DelAll(Afirst,&Alast); // чистим список
Anom= getfile();
CreatSpisA(Anom); //создаем список из массива с доп проверками
B.DelAll(Bfirst,&Blast); // чистим список
Anom= getfile();
CreatSpisB(Anom); //создаем список из массива с доп проверками
CreatSpisA(Anom); //добавляем в список из массива с доп проверками
getch();
}
if(otv=='2') //вычесть
{
A.DelAll(Afirst,&Alast); // чистим список
Anom= getfile();
CreatSpisA(Anom); //создаем список из массива с доп проверками
B.DelAll(Bfirst,&Blast); // чистим список
Anom= getfile();
CreatSpisB(Anom); //создаем список из массива с доп проверками
AntiPars(Anom, Mpars);
// меняем знак
for(int i=0;i<Anom/2;i++)
{
Mpars[i*2]=(Mpars[i*2])*(-1);// результаты разбора
//dig=Mpars[i*2];
//sv=Mpars[i*2+1];
//inttoname(dig,sv, buf);
}
AntiPars(Anom, Mpars);
CreatSpisA(Anom); //добавляем в список из массива с доп проверками
getch();
}
if(otv=='3') //призводная нужен только один операнд
{
A.DelAll(Afirst,&Alast); // чистим список
Anom= getfile();
CreatSpisA(Anom); //создаем список из массива с доп проверками
B.DelAll(Bfirst,&Blast); // чистим список
Proizvodnay1(Afirst,Bfirst, Anom);
_getch();
}
if(otv=='4') //умножить
{
A.DelAll(Afirst,&Alast); // чистим список
Anom= getfile();
CreatSpisA(Anom); //создаем список из массива с доп проверками
B.DelAll(Bfirst,&Blast); // чистим список
Anom= getfile();
CreatSpisB(Anom); //создаем список из массива с доп проверками
Mult(Afirst,Bfirst,Anom);
_getch();
}
break;
case '0':// Выход
A.DelAll(Afirst,&Alast);
break;
default:
cout << endl << "Ошибка" << endl;
break;
}
}while(otv!='0');
}
Pars6.cpp
#include "stdafx.h"
extern unsigned int Mpars[500];
void AntiPars(int Anom, unsigned int Mpars[])
// разворачивание полинома
// печать полинома из массива в форме xyz
{
int i=0;
int dig;
unsigned int sv;
char buf[256];
printf("\n");
for(int i=0;i<Anom/2;i++)
{
dig=Mpars[i*2];// результаты разбора
sv=Mpars[i*2+1];
inttoname(dig,sv, buf);
}
printf("\n");
}
void inttoname(int elem,unsigned int pol,char buffer[])
//разворачивание монома
// вход коэффициент. свертка монома. буфер - строка развернутого монома
{
int j=0;
int x,y,z,i;
i=pol%1000;
x=pol/1000;
y=(pol%1000)/100;
z=((pol%1000)%100)/10;
j = sprintf(buffer," "); // C4996
if(elem!=1) j += sprintf(buffer+j,"%+d",elem); // C4996
if(elem==1) j += sprintf(buffer+j,"+"); // C4996
if(elem==-1) j += sprintf(buffer+j,"-"); // C4996
if(x>1) j += sprintf(buffer+j,"x%d",x); // C4996
if(y>1) j += sprintf(buffer+j,"y%d",y); // C4996
if(z>1) j += sprintf(buffer+j,"z%d",z); // C4996
if(x==1) j += sprintf(buffer+j,"x"); // C4996
if(y==1) j += sprintf(buffer+j,"y"); // C4996
if(z==1) j += sprintf(buffer+j,"z"); // C4996
printf("%s",buffer);
}
int getfile()
{
FILE *in;
char name[200];
char buf[300];
int inom=0;
cout << endl<< "Введите имя файла"<< endl<< "=";
cin >> name;
// printf("\n %s \n",name);
if((in = fopen(name, "r")) == NULL) // C4996
{
printf("\n Файл не открыт %s\n",name);
_getch();
exit(0);
}
else
{
// printf("\n Файл открыт %s\n",name);
fgets(buf,200,in);
fclose(in);
inom= Svernut (buf);
}
return(inom);
}
//============================================================
/*
фиксируется старшинство переменных.
Будем считать, что x - самая старшая переменная, затем следует y, затем z.
Для каждого монома определим его "свернутую степень" (индекс).
Для монома xAyBzC+n. индекс равен A*1000+B*100+C*10+n (по условию задачи A, B и C не выше 9).
Старшим считается моном с большей свернутой степенью.
Например, x3y старше xy7z6, так как 310 больше 176.
Например, пусть имеется два полинома:
P(x,y,z)=x7y2z+3x2z-6y2-3z9 и
Q(x,y,z)=-7x2z+6y2+5.
В результате выполнения операции сложения должен быть получен полином R(x,y,z)=x7y2z-4x2z-3z9+5.
вход - исходная строка
выход - свернутый полином в Mpars
выход - количество мономов
*/
int Svernut(char buf[])
{
// int x,y,z;
int i,j,k,l;
char c;
unsigned int sv=0; // 0 to 4,294,967,295
//char buf[128];
char nom[50];
char monom[50];
int pr=0; // признак начала числа
int dig=0;
int imonom=0;
i=0;
cout << endl<< "==Исходный полином: "<< endl;
cout << buf;
for(;;)
{
j=0;
if(i>strlen(buf)) break;
c=buf[i];
if(isdigit(c))
{
monom[j]='+';
j++;
}
if(pr==0)
{
if(c=='+'|| c=='-')
{
monom[j]=c;
j++;
i++;
pr=1;
}
if(isdigit(c))
{
monom[j]='+';
j++;
pr=1;
}
} //конец первого монома
for(;;)
{
pr=0;
monom[j]=0x00;
c=buf[i];
if(i>strlen(buf)) break;
if(c=='+'|| c=='-') break;
monom[j]=c;
j++;
i++;
monom[j]=0x00;
} //моном
// разделяем моном на коэффициент и степени
k=0;
for(;;)
{
if(k>=j) break;
c=monom[k];
if(c=='x'|| c=='y'|| c=='z') break;
nom[k]=c;
k++;
}
nom[k]=0x00;
// printf("\n nom=%s",nom); // нашли коффициент
// ищем свертку
for(;;)
{
if(k>=j) break;
c=monom[k];
l=1;
if(k+1<j) // есть что-то дальше
{
if(isdigit(monom[k+1]))
{
l=monom[k+1]-'0';
}
}
if(c=='x')
{
sv=sv+l*1000;
if (l>1) k++; // пропускаем степень
}
if(c=='y')
{
sv=sv+l*100;
if (l>1) k++; // пропускаем степень
}
if(c=='z')
{
sv=sv+l*10;
if (l>1) k++; // пропускаем степень
}
k++;
}
nom[k]=0x00;
dig=atoi(nom);
if(strlen(nom)==1) // это для мономов без коэффициента
{
if(nom[0]=='-') dig=-1;
if(nom[0]=='+')dig=1;
}
if(dig==0) dig=1;
///printf("\n Коэффициент монома=%s %d",nom,dig); // нашли коффициент strtoi
if(sv==0) sv=1; // это завершающее число
///printf("\n Свертка монома sv=%d",sv); // нашли свертку
Mpars[imonom]=dig;// запоминаем результаты разбора
Mpars[imonom+1]=sv;
imonom=imonom+2;
sv=0; // пошли наследующий моном
j=0;
} // конец буфера
return(imonom);// количество мономов
}