Advertisement
Guest User

Untitled

a guest
Feb 23rd, 2020
111
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
C++ 10.94 KB | None | 0 0
  1. #include <iostream>
  2. #include <math.h>
  3. #include <cmath>
  4. #include <vector>
  5. #include <Windows.h>
  6. #include <stdlib.h>
  7. #include <fstream>
  8. #include <string.h>
  9. #include <time.h>
  10. #include <iomanip>
  11. using namespace std;
  12.  
  13. const double PI = 3.141592653589793238463;
  14.  
  15. //Для хранения сетки
  16. typedef struct Node
  17. {
  18.     double x, y;
  19. } Node;
  20.  
  21. //Задание функции
  22. typedef double(*functiontype)(double x);
  23. double Myfunc(double x)
  24. {
  25.     return x*x;
  26. }
  27. functiontype Func = &Myfunc;
  28.  
  29. //Приблизительное значение 2 производной
  30. double approximate_derivative_2(functiontype* f, double x) {
  31.     double delta = 1e-5;
  32.     return (((*f)(x) - 2 * (*f)(x + delta) + (*f)(x + 2 * delta)) / (delta*delta));
  33. }
  34.  
  35. //Приблизительное значение 2 производной
  36. double approximate_derivative_1(functiontype* f, double x) {
  37.     double delta = 1e-5;
  38.     return (  ((*f)(x + delta) - (*f)(x)) / delta );
  39. }
  40.  
  41. //Возвращает значение сплайна в точке
  42. double Spline(double x, int i, double *Array_steps, double *x_massiv, Node* Array)
  43. {
  44.     double h2, g1, g2, x1, x2, y1, y2;
  45.    
  46.     h2 = Array_steps[i + 1]; //этот массив с  индексацией от 1, размер - CountSegments, т.е. кол-в точек - 1
  47.     g1 = x_massiv[i];       g2 = x_massiv[i + 1]; //тут обращение в конце к n+1, а в степс к n при том же индексе, всё норм
  48.     x1 = Array[i].x;        x2 = Array[i + 1].x;
  49.     y1 = Array[i].y;        y2 = Array[i + 1].y;
  50.    
  51.     return (
  52.     (x*x*x)*( (g2-g1)/(6*h2) ) +
  53.     (x*x)*( (g1*x2-g2*x1)/(2*h2) ) +
  54.     (x)*( (g2*x1*x1-g1*x2*x2)/(2*h2) + (g1*h2-g2*h2)/6 + (y2-y1)/h2 ) +
  55.     ( (g1*x2*x2*x2-g2*x1*x1*x1)/(6*h2) + (g2*h2*x1-g1*h2*x2)/6 + (y1*x2-y2*x1)/h2 )
  56.     );
  57.  
  58. }
  59.  
  60. //Равномерная сетка
  61. void ValueUniformTable(functiontype* f, Node* Array, double Initial, double End, int CountSegments)
  62. {
  63.     int i;
  64.     double h;
  65.  
  66.     Array[0].x = Initial; Array[0].y = (*f)(Initial);
  67.     h = (End-Initial) / CountSegments;
  68.  
  69.     cout << "(" << Array[0].x << ":" << Array[0].y << ")" << endl;
  70.     for (int i = 1; i <= CountSegments; i++)
  71.     {
  72.         Array[i].x = Array[i - 1].x + h;
  73.         Array[i].y = (*f)(Array[i].x);
  74.         cout << "(" << Array[i].x << ":" << Array[i].y << ")" << endl;
  75.     }
  76.     cout << "(" << Array[CountSegments].x << ":" << Array[CountSegments].y << ")" << endl;
  77. }
  78.  
  79. //Неравномерная сетка
  80. void ValueIrregularTable(functiontype* f, Node* Array, double Initial, double End, int CountSegments, double *Array_steps)
  81. {
  82.     int i;
  83.     double alpha, h;
  84.  
  85.     Array[0].x = Initial;           Array[CountSegments].x = End;
  86.     Array[0].y = (*f)(Array[0].x);  Array[CountSegments].y = (*f)(Array[CountSegments].x);
  87.     h = (Array[CountSegments].x - Array[0].x) / CountSegments;
  88.     Array_steps[0] = 0;
  89.     alpha = 2 * PI / CountSegments;
  90.  
  91.     for (i = 1; i < CountSegments; i++) {
  92.         Array_steps[i] = h + (2*h/3)*cos(alpha*i);     //hi = h_i-h_i-1, то есть шаг назад
  93.         Array[i].x = Array[i - 1].x + Array_steps[i];
  94.         Array[i].y = (*f)(Array[i].x);
  95.     }
  96.     Array_steps[CountSegments] = Array[CountSegments].x - Array[CountSegments - 1].x;
  97.  
  98.     //вывод в консоль
  99.     for (i=0; i<= CountSegments; i++)
  100.         cout << "(" << Array[i].x << ":" << Array[i].y << ")" << endl;
  101.  
  102. }
  103.  
  104. /* Принимает на вход коэффициенты СЛАУ в виде массива matrix_coeffs - массив вида [[a1,b1,c1,d1],[a2,b2,c2,d2],...,[an,bn,cn,dn].
  105.    Заполняет Двумерный массив прогоночных коэффициентов coeffs_massiv (0 - Кси, 1 - Эта).
  106.    Находит решение в виде массива x_massiv
  107.  */
  108. void tridiagonal_matrix_algorithm(double** matrix_coeffs, int matrix_size, double* x_massiv, double B) {
  109.     int i;
  110.     double denominator;
  111.     double K, E, K_prev = 0, E_prev = 0; //Прогоночные коэффициенты
  112.     double = y1,h2,h2;
  113.  
  114.     double** coeffs_massiv;
  115.     coeffs_massiv = new double*[matrix_size + 1];
  116.     for (i = 0; i < matrix_size+1; i++)
  117.         coeffs_massiv[i] = new double[2];
  118.  
  119.     coeffs_massiv[0][0] = 0; coeffs_massiv[0][1] = 0; //K1,E1 = 0
  120. //  matrix_coeffs[0][0] = 0; matrix_coeffs[matrix_size - 1][2] = 0; //а1=сn=0 - убрать если работает без этого, написано в др функции
  121.  
  122.     //Прямой ход
  123.     for (i = 1; i <= matrix_size; i++) { //Ищем K_i+1 и E_i+1 на iом шаге
  124.         denominator = (matrix_coeffs[i - 1][0] * K_prev + matrix_coeffs[i - 1][1]);
  125.         K = -(matrix_coeffs[i - 1][2]) / denominator;
  126.         E = (matrix_coeffs[i - 1][3] - matrix_coeffs[i - 1][0] * E_prev) / denominator;
  127.         K_prev = K;
  128.         E_prev = E;
  129.         coeffs_massiv[i][0] = K; coeffs_massiv[i][1] = E; //лежат со сдвигом из за индексации
  130.     }
  131.  
  132.     //Обратный ход
  133.     h2 = Array_steps[matrix_size]; //вроде так, h0 условно существует и h1 лежит на 1 инд, h_n-1 na matrix size
  134.     y1 = Array[matrix_size-1].y;        
  135.     y2 = Array[matrix_size].y; //по индексам хз
  136. //  x_massiv[0] = A; //g0
  137.     x_massiv[matrix_size - 1] = coeffs_massiv[matrix_size][1]; //это gn-1. [g0,...gn] - n+1 значение, перепишу в соот-вии с этим
  138.     x_massiv[matrix_size] = 6*(y2-y1-B*h2)/(h2*h2) - 2*x_massiv[matrix_size - 1] //gn
  139.    
  140.     for (i = matrix_size - 2; i >= 1; i--) {
  141.         x_massiv[i] = coeffs_massiv[i + 1][0] * x_massiv[i + 1] + coeffs_massiv[i + 1][1];
  142.     }
  143.  
  144.     cout << endl << "Размер матрицы = " << matrix_size << ", Гаммы (1 и посл известны изначально):" << endl;
  145.     for (i = 0; i <= matrix_size; i++)
  146.         cout << x_massiv[i] << endl;
  147.  
  148. }
  149.  
  150. //Составляем СЛАУ с учётом нач данных и закидываем это всё в один массив
  151. void get_matrix_coeffs(double Initial, double End, int matrix_size, functiontype* f, double** matrix_coeffs, Node* Array, double *Array_steps, double B) {
  152.     int i;
  153.     double h1, h2, h_prev, d2, g2;
  154.     double x1, x2, y1, y2, y3, denominator;
  155.  
  156.     for (i = 1; i <= matrix_size; i++) { //в системе i=1,..n-1, g0,gn - известны
  157.         h1 = Array_steps[i]; //i
  158.         h2 = Array_steps[i + 1];; //i+1
  159.         h_prev = h2;
  160.         matrix_coeffs[i - 1][0] = h1;
  161.         matrix_coeffs[i - 1][1] = 2 * (h1 + h2);
  162.         matrix_coeffs[i - 1][2] = h2;
  163.         matrix_coeffs[i - 1][3] = 6 * (  (Array[i + 1].y - Array[i].y) / h2 - (Array[i].y - Array[i - 1].y) / h1  );
  164.     }
  165.    
  166.     //Переопределние последнего уравнения для моего случая:
  167.     h1 = Array_steps[matrix_size];
  168.     y1 = Array[matrix_size - 1].y;
  169.     y2 = Array[matrix_size].y;
  170.    
  171.     matrix_coeffs[matrix_size - 1][0] = h1;
  172.     matrix_coeffs[matrix_size - 1][1] = 2 * h1;
  173.     matrix_coeffs[matrix_size - 1][2] = 0; //cn=0
  174.     matrix_coeffs[matrix_size - 1][3] = 6 * (B - (y2 - y1) / h1);
  175.     matrix_coeffs[0][0] = 0; //а1=0
  176.  
  177.     for (i = 0; i < matrix_size; i++) {
  178.         cout << "(" << matrix_coeffs[i][0] << ")" << "  " << "(" << matrix_coeffs[i][1] << ")" << "  " << "(" << matrix_coeffs[i][2] << ")" << "  " << endl;
  179.     }
  180.  
  181. }
  182.  
  183. /*
  184. Строит график функции на всём интервале с заданным количеством точек Count_dots
  185. */
  186. void orig_table_in_file(functiontype* f, int Count_dots, double Initial, double End) { /
  187.     int k;
  188.     double x_value, step;
  189.  
  190.     step = (End - Initial) / (Count_dots - 1);
  191.     x_value = Initial - step;
  192.     ofstream fout("D:/original_graphic.txt");
  193.  
  194.     for (i=0; i < Count_dots; i++) {
  195.         x_value += step;
  196.         fout << x_value << " ";
  197.         fout << (*f)(x_value) << endl;
  198.     }
  199.     fout.close();
  200. }
  201.  
  202. //Используя функцию Spline записываем в файл таблицу значений спалйна на всём интервале в Count_dots точках
  203. void spline_table_in_file(Node* Array, int Count_dots, int Count_Segments, double Initial, double End, double *Array_steps, double *x_massiv) { // Функция берет таблицу иксов и игриков, количество точек в которых считаем знач полинома (мб убрать) и коэфф полинома,
  204.                                                  
  205.     int i = 0;
  206.     double step = (End - Initial) / (Count_dots - 1), y_value, x_value;
  207.  
  208.     ofstream fout("D:/spline_graphic.txt");
  209.     x_value = Initial;
  210.     for (i = 0; i < Count_Segments; i++) {
  211.         while (x_value < Array[i + 1].x) {
  212.             fout << x_value << " ";
  213.             y_value = Spline(x_value, i, Array_steps, x_massiv, Array);
  214.             fout << y_value << endl;
  215.             x_value += step;
  216.         }
  217.     }
  218.     fout << End << " ";
  219.     fout << Spline(End, Count_Segments-1, Array_steps, x_massiv, Array) << endl;
  220.  
  221.     fout.close();
  222. }
  223.  
  224. int main()
  225. {
  226.     setlocale(LC_ALL, "RUS");
  227.     functiontype Func = &Myfunc;
  228.     double Initial = 0, End =9, B = 5;
  229.     int CountSegments, i, Countdots = 5000;
  230.     cout << "Введите N: ";
  231.     //cin >> CountSegments;
  232.     CountSegments = 10;
  233.     cout << "Точек: " << CountSegments + 1 << endl << endl;
  234.     double *Array_steps = new double[CountSegments];
  235.  
  236.     //Построение сетки
  237.     cout << "Равномерная Сетка: " << endl;
  238.     Node* ArrayUniformNodes = new Node[CountSegments + 1];
  239.     ValueUniformTable(&Func, ArrayUniformNodes, Initial, End, CountSegments);
  240.  
  241.     cout << endl << "Неравномерная Сетка: " << endl;
  242.     Node* ArrayIrregularNodes = new Node[CountSegments + 1];
  243.     ValueIrregularTable(&Func, ArrayIrregularNodes, Initial, End, CountSegments, Array_steps);
  244.     cout << endl;
  245.  
  246.     //Заполнение массива коэффициентами матрицы
  247.     double** matrix_coeffs;
  248.     matrix_coeffs = new double*[CountSegments - 1];
  249.     for (i = 0; i < CountSegments - 1; i++)
  250.         matrix_coeffs[i] = new double[3];
  251.  
  252.     cout << "Полученная матрица: " << endl;
  253.     get_matrix_coeffs(Initial, End, CountSegments - 1, &Func, matrix_coeffs, ArrayIrregularNodes, Array_steps, B);
  254.  
  255.  
  256.     //Получение ответа в x_massiv
  257.     double* x_massiv = new double[CountSegments + 1];
  258.     x_massiv[0] = A; //g0
  259.     tridiagonal_matrix_algorithm(matrix_coeffs, CountSegments, x_massiv); //ПОч CountSegments - 1 было? счиьаю ошибкой
  260.  
  261.     //Вывод интерполируемой функции
  262.     orig_table_in_file(ArrayIrregularNodes, &Func, Countdots, Initial, End);
  263.  
  264.     //Вывод сплайна на экран
  265.     spline_table_in_file(ArrayIrregularNodes, Countdots, CountSegments, Initial, End, Array_steps, x_massiv);
  266.  
  267.     //Приближение производных на кроях
  268.     cout << endl;
  269.     cout << "1 производная на левом конце: " << approximate_derivative_1(&Func, Initial) << endl;
  270.     cout << "2 производная на левом конце: " << approximate_derivative_1(&Func, Initial) << endl;
  271.     cout << "1 производная на правом конце: " << approximate_derivative_2(&Func, End) << endl;
  272.     cout << "2 производная на правом конце: " << approximate_derivative_2(&Func, End) << endl;
  273.  
  274.  
  275.     cout << endl;
  276.     system("pause");
  277.     return 0;
  278. }
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement