非线性一元方程求解——弦截法、牛顿迭代法(C++)

一元方程

对于一元方程,如果要求f(x)=0的解,其过程大致包括如下三个问题:

  1. 根的存在性:是否有根,如果有,有几个?
  2. 根的分布:根分布区间;
  3. 求根的公式:如何从根的近似值逐步精确化?

求解方法

首先,对于某些特殊放入函数 f ( x ) f(x) f(x),如果满足以下定理: f ( x ) f(x) f(x)在区间 [ a , b ] [a,b] [a,b]上连续, f ( a ) ⋅ f ( b ) < 0 f(a)\cdot f(b)<0 f(a)f(b)<0,则 f ( x ) = 0 f(x)=0 f(x)=0在区间 [ a , b ] [a,b] [a,b]上至少存在一个根。
然后就需要确定根的分布,此处有两种方法:

  1. 搜索法:将区间 [ a , b ] [a,b] [a,b]分割为n个子区间,子区间长度为 Δ x \Delta x Δx,如果 f ( x i ) ⋅ f ( x i + 1 ) < 0 ( i = 0 , ⋯ , n ) f(x_i)\cdot f(x_{i+1})<0(i=0,\cdots,n) f(xi)f(xi+1)<0(i=0,,n),则根在区间 [ x i , x i + 1 ] [x_i,x_{i+1}] [xi,xi+1]内。
  2. 对分法:将区间 [ a , b ] [a,b] [a,b]从中点分开,分别计算 f ( a ) ⋅ f ( ( a + b ) / 2 ) f(a)\cdot f((a+b)/2) f(a)f((a+b)/2) f ( b ) ⋅ f ( ( a + b ) / 2 ) f(b)\cdot f((a+b)/2) f(b)f((a+b)/2),判断哪个乘积小于0,继续对分对应区间,直至满足精度。

对于一般的情况,对分法的搜索速度较快,但是如果存在根与某个区间端点非常靠近的情况,搜索法反而速度更快。对分法还有一个问题,若区间内部存在多个根,可能会出现遗漏和算法报错等原因,因此需要根据实际需要选取对应的方法确定根的分布。
当初始区间较大时,对于上述两种方法其收敛速度都不太理想,因此引入下面的方法:

  1. 牛顿迭代法:将原方程进行泰勒展开,可以发现 f ( x ) = f ( x 0 ) + f ′ ( x 0 ) ( x − x 0 ) + f ′ ′ ( ξ ) 2 ! ( x − x 0 ) 2 f(x)=f(x_0)+f'(x_0)(x-x_0)+\frac{f''(\xi)}{2!}(x-x_0)^2 f(x)=f(x0)+f(x0)(xx0)+2!f′′(ξ)(xx0)2,因此有 f ( x ) ≈ f ( x 0 ) + f ′ ( x 0 ) ( x − x 0 ) f(x)\approx f(x_0)+f'(x_0)(x-x_0) f(x)f(x0)+f(x0)(xx0),由此得到牛顿迭代式: x k + 1 = x k − f ( x k ) f ′ ( x k ) x_{k+1}=x_k-\frac{f(x_k)}{f'(x_k)} xk+1=xkf(xk)f(xk)。可是牛顿迭代法的收敛性取决于 x 0 x_0 x0的选取,因此使用牛顿迭代法之前还需要满足如下条件:
    • f ( a ) ⋅ f ( b ) < 0 f(a)\cdot f(b)<0 f(a)f(b)<0;
    • f ′ ( x ) ≠ 0 f'(x)\neq 0 f(x)=0;
    • f ′ ′ ( x ) f''(x) f′′(x)存在且不变号;
    • x 0 ∈ [ a , b ] x_0\in [a,b] x0[a,b],使得 f ′ ′ ( x 0 ) ⋅ f ( x 0 ) > 0 f''(x_0)\cdot f(x_0)> 0 f′′(x0)f(x0)>0;
  2. 弦截法:相比于牛顿迭代法,弦截法不需要使用导数,使用差商进行计算,如果 f ( x ) f(x) f(x)比较复杂,即使是近似计算导数也非常不方便。根据这个思路可以得到弦截法的迭代公式: x k + 1 = x k − ( x k − x k − 1 ) f ( x k ) f ( x k ) − f ( x k − 1 ) x_{k+1}=x_k-(x_k-x_{k-1})\frac{f(x_k)}{f(x_k)-f(x_{k-1})} xk+1=xk(xkxk1)f(xk)f(xk1)f(xk)。弦截法的使用条件与牛顿迭代法相同,收敛速度也较牛顿法慢,并且需要两个初值。

代码实现

首先是牛顿法的实现:

#include<iostream>
#include<cmath>
#include <chrono>//计时模块
#include <iomanip>
#define DerivativeIntervalAccuracy 1e-6
//近似求导,传递函数进来
double f_derivative(double (*fx)(double),double x)
{return (fx(x + DerivativeIntervalAccuracy) - fx(x))/DerivativeIntervalAccuracy;
}//指定函数
double fx(double x)
{//计算一个复杂函数return pow(x,4)-11.7*log10(x+3)+exp(-x)-5;
}
double Newton_iterative(double (*fx)(double x),double x_0,double error)
{double x_1=0.0,x_2=x_0;//x_1表示x_k-1,x_2表示x_kdouble d_fx;int i = 0;do{x_1=x_2;d_fx=f_derivative(fx, x_1);x_2=x_1-fx(x_1)/d_fx;if(i++>10000){std::cout<<"迭代次数过多"<<std::endl;return 0.0;}//std::cout<<"x(k)="<<x_1<<" "<<"x(k+1)="<<x_2<<" "<<std::endl;}while(fabs(x_2-x_1)>error);//std::cout<<"迭代次数"<<i<<std::endl;return x_2;
}
int  main()
{double error=1e-6;double x;double x_0=100.0;//初始值double d_fx = 0.0;//导数auto start = std::chrono::steady_clock::now();x = Newton_iterative(fx, x_0, error);// 结束计时auto end = std::chrono::steady_clock::now();// 计算时间差auto duration = std::chrono::duration_cast<std::chrono::microseconds>(end - start);// 输出时间差(以毫秒为单位)std::cout << "Elapsed time: " << duration.count() << " microseconds" << std::endl;std::cout<<"x="<<std::setprecision(10)<<x<<std::endl;return 0;
}

其次是弦截法的实现:

#include<iostream>
#include<cmath>
#include <iomanip>
#include <chrono>//计时模块
using namespace std;
//指定函数
double fx(double x)
{//计算一个复杂函数return pow(x,4)-11.7*log10(x+3)+exp(-x)-5;
}
double chord_truncation(double (*fx)(double x),double x_0,double x_1,double error)
{int i = 0;double x_2,x_3=x_0,x_4=x_1;do{x_2 = x_3;x_3 = x_4;x_4=x_3-(x_3-x_2)*(fx(x_3)/(fx(x_3)-fx(x_2)));if(i++>10000){std::cout<<"迭代次数过多"<<std::endl;return 0.0;}cout<<"x(k-1)="<<x_2<<" "<<"x(k)="<<x_3<<" "<<"x(k+1)="<<x_4<<endl;}while(fabs(x_3-x_2)>error);//std::cout<<"迭代次数"<<i<<std::endl;return x_4;
}
int  main()
{double error=1e-6;double x_1=10.0,x_2=0.0;//x_1表示x_k-1,x_2表示x_kauto start = std::chrono::steady_clock::now();double x=chord_truncation(fx,x_1,x_2,error);// 结束计时auto end = std::chrono::steady_clock::now();// 计算时间差auto duration = std::chrono::duration_cast<std::chrono::microseconds>(end - start);// 输出时间差(以毫秒为单位)std::cout << "Elapsed time: " << duration.count() << " microseconds" << std::endl;cout<<"x="<<std::setprecision(15)<<x<<endl;return 0;
}

结果分析

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
对于牛顿法,为了避免cout输出的输出影响测试,第一次测试了算法的执行时间,耗时9ms,总共迭代了18次,最终结果在matlab中可以看出非常接近0,使用牛顿迭代法时需要注意初始点的选择,这决定了算法的迭代速度。
在这里插入图片描述
在这里插入图片描述

对于弦截法,如果将初始的 x 0 、 x 1 x_0、x_1 x0x1换为10和100,运行结果如下:
在这里插入图片描述

对于弦截法,其迭代次数与两个初始点选择都相关。在与matlab中测试结果发现,其结果精度更高,更加接近0。
在这里插入图片描述
另外地,如果错误地选择某些点,迭代结果会出现错误,例如在牛顿迭代法和弦截法中初始点包括0,就会出现下面报错:
在这里插入图片描述

这是因为在迭代过程中产生了某个时刻的x<-3,而函数里面包括了 log ⁡ 10 ( x + 3 ) \log_{10}(x+3) log10(x+3),但是将初始点换为-2时,又能求出此函数的另外一个解:
在这里插入图片描述

那为什么会迭代发散呢?使用matlab将函数图像绘制出来:
在这里插入图片描述

结合上面提到的牛顿迭代法的使用条件,当初始值为0时,显然不满足第四条。
综上所述可以得出如下结论:无论是牛顿迭代法还是弦截法都需要注意初始点的选择,这不仅关系到算法速度,还关系到能否得到正确的结果。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/686850.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

成本效能FinOps: Crane 部署

目录 一、实验 1.环境 2.安装kind 3.安装Crane 二、问题 1.脚本安装prometheus报错 2.查看集群信息失败 3.Helm添加grafana 报错 4.查看crane资源失败 5.prometheus部署时kube-state-metrics 拉取镜像显示ImagePullBackOff 6.Crane 功能与架构 一、实验 1.环境 &a…

记录 | 验证pytorch-cuda是否安装成功

检测程序如下&#xff1a; import torchprint(torch.__version__) print(torch.cuda.is_available()) 或者用终端 Shell&#xff0c;运行情况如下

单片机移植Lua(STM32H743移植Lua-5.4.6)

文章目录 目的移植演示示例链接更多说明合理设置内存大小按需加载标准库实现系统和IO接口设置引用路径 总结 目的 通常单片机都是使用C/C来开发的&#xff0c;任何修改都需要重新编译固件然后下载运行。在一些需要灵活性更强的场合中可以内嵌Lua解释器实现动态更新应用程序的功…

【Linux】管道文件 打包压缩 文本编辑器nano 进度条

目录 什么是管道文件&#xff1f; 打包和压缩 文本编辑器 nano的安装 nano的使用 退出nano编辑&#xff0c;ctrlx 普通用户无法sudo&#xff0c;该怎么解决 Linux小程序-进度条 预备知识 1.回车换行 2.缓冲区 准备工作 代码实现 1.processBar.h代码编写 2.main.c代…

@ControllerAdvice 的介绍及三种用法

ControllerAdvice 的介绍及三种用法 浅析ControllerAdvice 首先&#xff0c;ControllerAdvice本质上是一个Component&#xff0c;因此也会被当成组建扫描&#xff0c;一视同仁&#xff0c;扫扫扫。 然后&#xff0c;我们来看一下此类的注释&#xff1a; 这个类是为那些声明了&…

订餐|网上订餐系统|基于springboot的网上订餐系统设计与实现(源码+数据库+文档)

网上订餐系统目录 目录 基于springboot的网上订餐系统设计与实现 一、前言 二、系统功能设计 三、系统实现 1、用户功能模块的实现 &#xff08;1&#xff09;用户注册界面 &#xff08;2&#xff09;用户登录界面 &#xff08;3&#xff09;菜品详情界面 &#xff08…

Codeforces Round 486 (Div. 3)

目录 A. Diverse Team B. Substrings Sort C. Equal Sums D. Points and Powers of Two E. Divisibility by 25 F. Rain and Umbrellas A. Diverse Team 找出不重复的同时存下下标即可&#xff0c;依次遍历map判断重复最后判断数量即可 void solve(){cin>>n>>…

AI专题:5G-A扬帆风正劲,踏AI增长新浪潮

今天分享的是AI系列深度研究报告&#xff1a;《AI专题&#xff1a;5G-A扬帆风正劲&#xff0c;踏AI增长新浪潮》。 &#xff08;报告出品方&#xff1a;开源证券&#xff09; 报告共计&#xff1a;22页 足立连接&#xff0c;拓展算力&#xff0c;双曲线稳步发力 中兴通讯拥…

【论文精读】DALL·E

摘要 本文利用从互联网上收集的2.5亿个图像/文本对数据&#xff0c;训练了一个120亿参数的自回归transformer&#xff0c;进而得到一个可以通过自然语言/图像控制生成的高保真图像生成模型。在大多数数据集上的表现超越以往的方法。 框架 本文的目标为通过训练一个自回归trans…

java+SSM+mysql 开放式实验管理系统78512-计算机毕业设计项目选题推荐(免费领源码)

摘 要 我国高校开放式实验管理普遍存在实验设备使用率较低、管理制度不完善,实验设备共享程度不高等诸多问题。要在更大范围推行开放式实验管理,就必须在开放式实验教学管理流程中,通过引入信息化管理加大信息技术在其中的应用,才能真正发挥这种教学模式的开放性优势。 本系统…

文生视频:Sora模型报告总结

作为世界模拟器的视频生成模型 我们探索视频数据生成模型的大规模训练。具体来说&#xff0c;我们在可变持续时间、分辨率和宽高比的视频和图像上联合训练文本条件扩散模型。我们利用对视频和图像潜在代码的时空补丁进行操作的变压器架构。我们最大的模型 Sora 能够生成一分钟…

小苯的数组切分 ---- 牛客月赛

题目描述 qionghuaqionghuaqionghua 给了小苯一个长度为 n 的数组 a&#xff0c;希望小苯将数组 aaa 分为恰好非空的三段。即&#xff1a;[1,l−1],[l,r],[r1,n]这三段&#xff0c;其中 1< l≤r<n。接着&#xff1a; ∙ 第一段的所有数字做 ⊕&#xff08;按位异或&…

【C语言相关问题】C语言中关于大小写字母转换的问题

大家好&#xff0c;这里是争做图书馆扫地僧的小白。非常感谢各位的支持&#xff0c;也期待着您的关注。 目前博主有着C语言、C、linux以及数据结构的专栏&#xff0c;内容正在逐步的更新。 希望对各位朋友有所帮助同时也期望可以得到各位的支持&#xff0c;有任何问题欢迎私信与…

每日OJ题_算法_递归④力扣24. 两两交换链表中的节点

目录 ④力扣24. 两两交换链表中的节点 解析代码 ④力扣24. 两两交换链表中的节点 24. 两两交换链表中的节点 难度 中等 给你一个链表&#xff0c;两两交换其中相邻的节点&#xff0c;并返回交换后链表的头节点。你必须在不修改节点内部的值的情况下完成本题&#xff08;即…

【大厂AI课学习笔记】【2.1 人工智能项目开发规划与目标】(2)项目开发周期

我们来学习项目开发的周期。 再次声明&#xff0c;本文来自腾讯AI课的学习笔记&#xff0c;图片和文字&#xff0c;仅用于大家学习&#xff0c;想了解更多知识&#xff0c;请访问腾讯云相关章节。如果争议&#xff0c;请联系作者。 今天&#xff0c;我们来学习AI项目的周期。 主…

17-k8s控制器资源-job控制

job控制器&#xff1a;就是一次性任务的pod控制器&#xff0c;pod完成作业后不会重启&#xff0c;其重启策略是&#xff1a;Never 1&#xff0c;job控制器案例描述 启动一个pod&#xff0c;执行完成一个事件&#xff0c;然后pod关闭&#xff1b; 事件&#xff1a;计算π的值&a…

AI大模型学习笔记之五:监督学习--数据如何驱动决策

监督学习&#xff0c;又称为监督式机器学习&#xff0c;是机器学习和人工智能领域的一个重要分支。 其基本原理是利用带有标签的数据集来训练算法&#xff0c;以实现精确分类数据或预测结果的目标。 在监督学习中&#xff0c;通过将数据输入模型&#xff0c;并不断调整数据权…

软件工程师,为什么不喜欢关电脑

概述 你是否注意到&#xff0c;软件工程师们似乎从不关电脑&#xff0c;也不喜欢关电脑&#xff1f;别以为他们是电脑“上瘾”&#xff0c;或是沉迷于电脑&#xff0c;这一现象背后蕴含着多种实际原因。 1、代码保存与恢复。 在编写代码过程中&#xff0c;遇到问题时可能会暂时…

【JavaScript】面试手写题精讲之数组(下)

引入 这章主要讲的是数组的排序篇&#xff0c;我们知道面试的时候&#xff0c;数组的排序是经常出现的题目。所以这块还是有必要进行一下讲解的。笔者观察了下前端这块的常用算法排序题&#xff0c;大概可以分为如下 冒泡排–> 稳定排序插入排序–> 稳定排序选择排序–…

Swing程序设计(10)列表框,文本框,文本域,密码框

文章目录 前言一、列表框二、文本框&#xff08;域&#xff09; 1.文本框2.文本域三、密码框总结 前言 该篇文章简单介绍了Java中Swing组件里的列表框、文本框、密码框。 一、列表框 列表框&#xff08;JList&#xff09;相比下拉框&#xff0c;自身只是在窗体上占据固定的大小…