C# 盛金公式 求解一元三次方程

    class dd2
    {
        private static readonly double dsr3 = Math.Sqrt(3);
        // <summary>
        /// 盛金公式求解一元三次方程,ax^3+bx^2+cx+d=0。
        /// </summary>
        /// <param name="a">三次系数</param>
        /// <param name="b">二次系数</param>
        /// <param name="c">一次系数</param>
        /// <param name="d">常系数</param>
        /// <param name="x1">结果1,为null则没有实根</param>
        /// <param name="x2">结果2</param>
        /// <param name="x3">结果3</param>
        public static void solveEquations(double a, double b, double c, double d, out double? x1, out double? x2, out double? x3,double dTol = 1e-9)
        {
            x1 = null;
            x2 = null;
            x3 = null;

            double A = b * b - 3 * a * c;
            double B = b * c - 9 * a * d;
            double C = c * c - 3 * b * d;
            double delta = B * B - 4 * A * C;

            if (IsZero(A, dTol) && IsZero(B, dTol))
            {
                //盛金公式1
                //方程有一个三重实根
                if (!IsZero(a, dTol))
                {
                    double r1 = -b / (3 * a);
                    x1 = x2 = x3 = r1;
                }
                else if (!IsZero(b, dTol))
                {
                    double r2 = -c / b;
                    x1 = x2 = x3 = r2;
                }
                else if (!IsZero(c, dTol))
                {
                    double r3 = -3 * d / c;
                    x1 = x2 = x3 = r3;
                }
                else
                {
                    //无解
                }
            }
            else if (IsZero(delta, dTol))
            {
                //盛金公式3
                //方程有三个实根,其中有一个二重根。
                //A != 0
                double K = B / A;
                x1 = -b / a + K;
                x2 = x3 = -K / 2;
            }
            else if (delta > 0)
            {
                //盛金公式2
                //方程有一个实根和一对共轭复根
                double dsrdelta = Math.Sqrt(delta);
                double yp = -B + dsrdelta;
                double yn = -B - dsrdelta;
                double y1 = A * b + 3 * a / 2 * yp;
                double y2 = A * b + 3 * a / 2 * yn;
                const double dZhiShu = 1d / 3;
                double y1cr = Math.Pow(y1, dZhiShu);
                double y2cr = Math.Pow(y2, dZhiShu);
                double dAm3 = 3 * a;
                x1 = -b - (y1cr + y2cr);
                x1 /= dAm3;
            }
            else
            {
                //盛金公式4:
                //方程有三个不相等的实根。
                //(A>0,-1<T<1)
                double T = (2 * A * b - 3 * a * B) / (2 * Math.Sqrt(Math.Pow(A, 3)));
                double theta = Math.Acos(T);
                double thetaOf1d3 = theta / 3;
                double cosThetaOf1d3 = Math.Cos(thetaOf1d3), sinTheta_1d3 = Math.Sin(thetaOf1d3);
                double dx2part = cosThetaOf1d3 + dsr3 * sinTheta_1d3;
                double dx3part = cosThetaOf1d3 - dsr3 * sinTheta_1d3;
                double dsrA = Math.Sqrt(A);
                double dAm3 = 3 * a;
                x1 = -b - 2 * dsrA * cosThetaOf1d3;
                x2 = -b + dsrA * dx2part;
                x3 = -b + dsrA * dx3part;
                x1 /= dAm3;
                x2 /= dAm3;
                x3 /= dAm3;
            }
        }
        /// <summary>
        /// num是否为0.(由于double本身的精度问题)
        /// </summary>
        /// <param name="num"></param>
        /// <param name="dTol"></param>
        /// <returns></returns>
        public static bool IsZero(double num,double dTol)
        {
            return Math.Abs(num) <= dTol;
        }
    }

参考资料:

盛金公式(盛金公式_百度百科

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值