[作业] 三次方程根的二分求解问题

正文:

题目:

问题拆解:

题目约定,存在三个根,且,,,.

三个根既函数的零点,显然,要通过二分法找到零点,需要确定三个根分别在的单调区间,所以求其极值点(驻点):

,(这是因为原函数有三个根,所以函数必定有拐点).

极值点1:.

极值点2:.

然后使用二分法在分别找函数零点即可.

关于这三个区间的单调性:

,则函数上单调递减,在其余两个区间上单调递增;

,则函数上单调递增,在其余两个区间上单调递减.

代码:

cpp 复制代码
#include <iostream>
#include <iomanip>
#include <ctime>
#include <cmath>
#include <algorithm>
using namespace std;
double a,b,c,d;
void res(){
	scanf("%lf%lf%lf%lf",&a,&b,&c,&d);
}
double func(double x){
	return a*x*x*x+b*x*x+c*x+d;
}
double findx(double x,double l,double r,double eps,bool mono){//mono1+0-
	if(r-l<=eps){
		return l;
	}
	double mid=l+(r-l)/2;
	if(mono){
		if(x<func(mid)){
			return findx(x,l,mid,eps,mono);
		}else{
			return findx(x,mid,r,eps,mono);
		}
		
	}else{
		if(x<func(mid)){
			return findx(x,mid,r,eps,mono);
		}else{
			return findx(x,l,mid,eps,mono);
		}
	}
	
}
int main(){
	res();
	double x1=0,x2=0;
	//f'=3ax2+2bx+c=0
	double delta=(2*b)*(2*b)-4*(3*a)*c;
	if(delta<0){
		printf("%lf\n",findx(0,-100,x1,0.0001,func(-100)<func(100)));
	}else{
		x1=(-2*b-sqrt(delta))/(2*3*a);
		x2=(-2*b+sqrt(delta))/(2*3*a);
		bool f=(func(x1)>func(x2));
		cout<<fixed<<setprecision(2)<<findx(0,-100,x1,0.00001,f)<<' '<<findx(0,x1,x2,0.00001,!f)<<' '<<findx(0,x2,100,0.00001,f)<<endl;
	}
	
	return 0;
}

*注:在40行,为了防止Δ<0,特别做了判定,因为函数整体单调,所以只需要进行一次查找即可.

相关推荐
大熊背2 小时前
双目拼接竖缝消除(ISP 分区锐化实操方案) 优化方案
人工智能·算法·双目拼接
_日拱一卒3 小时前
LeetCode:105从前序与中序遍历序列构造二叉树
算法·leetcode·职场和发展
MicroTech20253 小时前
微算法科技(NASDAQ :MLGO)发布基于NEQR技术的新型量子视频处理算法,重构智能视觉底层逻辑
科技·算法·音视频
techdashen3 小时前
Async Rust 近况补课:从 `async-trait` 到原生 async trait
网络·算法·rust
一行代码一行诗++3 小时前
循环的嵌套
数据结构·算法
计算机安禾3 小时前
【c++面向对象编程】第44篇:typename与class的区别,依赖类型名与template消除歧义
java·jvm·c++
Hua-Jay3 小时前
OpenCV联合C++/Qt 学习笔记(二十五)----监督学习聚类及K均值聚类
c++·笔记·opencv·学习·计算机视觉·聚类
玖釉-3 小时前
C++ 中的矩阵介绍:以二维矩阵查找为例
c++·windows·算法·矩阵
ECT-OS-JiuHuaShan3 小时前
存在是微分张量积,标量是参数但不可能是本质。还原论泛化,是语义劫持和以偏概全的逻辑谋杀伪科学庞氏骗局
数据库·人工智能·算法·机器学习·数学建模
CQU_JIAKE3 小时前
5.22【A】
算法