GSL中的错误 - 根发现

时间:2016-06-28 15:07:32

标签: c gsl

大家好,我有问题让我的程序与GSL一起工作 - 找根。我试图找到解决方案的方法。我正在寻找64行数据的解决方案,但在某些特定行中,程序无法继续,可能是因为不存在一个好的解决方案。但我希望程序只是在他找不到解决方案时跳过。但我的程序有时会停止并出现此消息: gsl:brent.c:74:错误:端点不跨越y = 0 调用passou1passou2passou3Default GSL错误处理程序。 中止陷阱:6

所以,我做了一些打印来检查我的程序到底在哪里,我发现它位于 gsl_root_fsolver_set(s,& F,x_lo,x_hi),但我没有找到如何打印此值或此功能给我的内容。

我的节目在这里,谢谢大家!

#include <stdio.h>
#include <math.h>
#include <gsl/gsl_errno.h>
#include <gsl/gsl_math.h>
#include <gsl/gsl_complex_math.h>
#include <gsl/gsl_roots.h>
#include "demo_fn.h"
#include "demo_fn.c"

int
main (void)
{

double NT, c;
c = 64.0/2.0;

char url[]="charm.txt";
double corr, core, a[64][3];
int nt, i = 0;
FILE *arq;
arq = fopen(url, "r");
if(arq == NULL)
    printf("Erro, nao foi possivel abrir o arquivo\n");
else
    while( (fscanf(arq,"%lf %lf %i\n", &corr, &core, &nt))!=EOF ) {
        a[i][0]=corr;
        a[i][1]=core;
        a[i][2]=nt;
        i++;
        //printf("%lf, %lf, %i\n",corr, core, nt);
    }

fclose(arq);




for (i= 0; i < 64; i++)
{
    int status;
    int iter = 0, max_iter = 200;
    const gsl_root_fsolver_type *T;
    gsl_root_fsolver *s;
    double r = 0, r_expected = 4.0;
    double x_lo = 0.0001, x_hi = 4.0;
    double ratio1, ratio2;

    ratio1 = a[i][0]/a[i+1][0];
    ratio2 = a[i+1][0]/a[i+2][0];
    printf ("ratio1: %lf, ratio2: %lf", ratio1, ratio2);
    printf ("\n");

 //   if (ratio1*ratio2 > 0)
 //      {

    printf("C(n_t) : %.15lf -- loop index : %i ----- ratio: %lf \n", a[i][0],i, ratio1);

    gsl_function F;
    struct quadratic_params params = {a[i][0], i, c, i+1, a[i+1][0]};
    F.function = &quadratic;
           printf ("passou1");
    F.params = &params;
    T = gsl_root_fsolver_brent;
           printf ("passou2");
    //T = gsl_root_fsolver_bisection;
    s = gsl_root_fsolver_alloc (T);
        printf ("passou3");
    gsl_root_fsolver_set (s, &F, x_lo, x_hi);
        printf ("passou4");
    printf ("using %s method\n", gsl_root_fsolver_name (s));
    printf ("%5s [%9s, %9s] %9s %10s %9s\n", "iter", "lower", "upper", "root", "err", "err(est)");


    do
    {
        iter++;
        status = gsl_root_fsolver_iterate (s);
        r = gsl_root_fsolver_root (s);
        x_lo = gsl_root_fsolver_x_lower (s);
        x_hi = gsl_root_fsolver_x_upper (s);
        status = gsl_root_test_interval (x_lo, x_hi,0, 0.001);
        if (status == GSL_SUCCESS)
        {
            printf ("Converged:\n");
        }
        printf ("%5d [%.7lf, %.7lf] %.7lf %+.7lf %.7lf\n", iter, x_lo, x_hi, r, r - r_expected, x_hi - x_lo);


    }
    while (status == GSL_CONTINUE && iter < max_iter);

    gsl_root_fsolver_free (s);

 //     }

    printf("\n");
}



return 0;

}

1 个答案:

答案 0 :(得分:2)

加布里埃拉。输出已经告诉你为什么编程错了。 端点不跨越y = 0。

  

本节中描述的根包围算法要求初始间隔保证包含根 - 如果a和b是间隔的端点,则f(a)必须与f(b)的符号不同。

以上内容来自manual of gsl,因此如果结束点具有相同的符号,程序将停止并告诉您此错误。

您是否曾尝试过gsl中的错误处理程序。在本手册的第3章中,如果您将此函数放在gsl_root_fsolver_set(s,&F, x_lo, x_hi)之前,它们会提供一个名为 gsl_set_error_handler_off()的函数,您可以将此函数分配给int类型变量,让我们说status,我们可以从手册中看到它(gsl_root_fsolver_set(s,&F, x_lo, x_hi))是int function,然后如果你打印出{{1}的值并使用gsl_errno.h文件检查它,您将知道该值的含义。

gsl_set_error_handler_off()可以解除执行中的堕胎。

对于您的代码,您应该执行以下操作:

  1. 添加status
  2. #include <gsl/gsl_errno.h>
  3. 之前添加gsl_set_error_handler_off()
  4. 使用status=gsl_root_fsolver_set(s,&F, x_lo, x_hi)的值,就像你程序的以下循环一样,对你的终点进行少许更改,扩展或缩小或翻译你的间隔,当他们满足初始条件时,程序将再次运行< / LI>