1. ホーム
  2. c++

[解決済み] このスコープでは 'foo' が宣言されていない c++

2022-03-03 11:19:31

質問

c++を勉強しているところです(数年前に1週間のサマーキャンプを受けたとき以来、初めて見ました)。

Javaで作っているプログラムをC++に変換していました。

#ifndef ADD_H
#define ADD_H
#define _USE_MATH_DEFINES
#include <iostream>
#include <math.h>

using namespace std;

class Evaluatable {
public:
  virtual double evaluate(double x);
};

class SkewNormalEvalutatable : Evaluatable{
public:
  SkewNormalEvalutatable();
  double evaluate(double x){
    return 1 / sqrt(2 * M_PI) * pow(2.71828182845904523536, -x * x / 2);
  }
};

SkewNormalEvalutatable::SkewNormalEvalutatable()
{
}

double getSkewNormal(double skewValue, double x)
{
  SkewNormalEvalutatable e ();
  return 2 / sqrt(2 * M_PI) * pow(2.71828182845904523536, -x * x / 2) * integrate(-1000, skewValue * x, 10000, e);
}

// double normalDist(double x){
//   return 1 / Math.sqrt(2 * Math.PI) * Math.pow(Math.E, -x * x / 2);
// }

double integrate (double start, double stop,
                                     int numSteps, 
                                     Evaluatable evalObj)
{
  double stepSize = (stop - start) / (double)numSteps;
  start = start + stepSize / 2.0;
  return (stepSize * sum(start, stop, stepSize, evalObj));
}

double sum (double start, double stop,
                               double stepSize,
                               Evaluatable evalObj)
{
  double sum = 0.0, current = start;
  while (current <= stop) {
    sum += evalObj.evaluate(current);
    current += stepSize;
  }
  return(sum);
}

// int main()
// {
//   cout << getSkewNormal(10.0, 0) << endl;
//   return 0;
// }
#endif

というエラーが出ました。

SkewNormal.h: In function 'double getSkewNormal(double, double)' :
SkewNormal.h: 29: error: 'integrate' was not declared in this scope
SkewNormal.h: In function 'double integrate(double, double, int, Evaluatable)':
SkewNormal.h:41: error: 'sum' was not declared in this scope

Integrateとsumは両方とも関数であることが前提です。

以下はJavaのコードですが、ほぼ同じです。

public static double negativelySkewed(double skew, int min, int max){
    return randomSkew(skew) * (max - min) + min;
}

public static double randomSkew(final double skew){
    final double xVal = Math.random();
    return 2 * normalDist(xVal) * Integral.integrate(-500, skew * xVal, 100000, new Evaluatable() {

        @Override
        public double evaluate(double value) {
            return normalDist(value);
        }
    });
}

public static double normalDist(double x){
    return 1 / Math.sqrt(2 * Math.PI) * Math.pow(Math.E, -x * x / 2);
}

/** A class to calculate summations and numeric integrals. The
 *  integral is calculated according to the midpoint rule.
 *
 *  Taken from Core Web Programming from 
 *  Prentice Hall and Sun Microsystems Press,
 *  http://www.corewebprogramming.com/.
 *  &copy; 2001 Marty Hall and Larry Brown;
 *  may be freely used or adapted. 
 */

public static class Integral {
  /** Returns the sum of f(x) from x=start to x=stop, where the
   *  function f is defined by the evaluate method of the 
   *  Evaluatable object.
   */

  public static double sum(double start, double stop,
                           double stepSize,
                           Evaluatable evalObj) {
    double sum = 0.0, current = start;
    while (current <= stop) {
      sum += evalObj.evaluate(current);
      current += stepSize;
    }
    return(sum);
  }

  /** Returns an approximation of the integral of f(x) from 
   *  start to stop, using the midpoint rule. The function f is
   *  defined by the evaluate method of the Evaluatable object.
   */

  public static double integrate(double start, double stop,
                                 int numSteps, 
                                 Evaluatable evalObj) {
    double stepSize = (stop - start) / (double)numSteps;
    start = start + stepSize / 2.0;
    return(stepSize * sum(start, stop, stepSize, evalObj));
  }
}

/** An interface for evaluating functions y = f(x) at a specific
 *  value. Both x and y are double-precision floating-point 
 *  numbers.
 *
 *  Taken from Core Web Programming from 
 *  Prentice Hall and Sun Microsystems Press,
 *  http://www.corewebprogramming.com/.
 *  &copy; 2001 Marty Hall and Larry Brown;
 *  may be freely used or adapted. 
 */
public static interface Evaluatable {
      public double evaluate(double value);
}

きっと、とてもシンプルなことなんだろうけど......。

また、どのように

getSkewNormal(double skewValue, double x)

SkewNormal.h 以外のファイルから?

解決方法は?

C++では、関数を使用する前に宣言することになっています。あなたのコードでは integrate を最初に呼び出した時点の前に宣言されていない。 integrate . 以下も同様です。 sum . したがって、エラーになります。関数定義がその関数の最初の呼び出しの前に来るように定義を並べ替えるか、各関数に [forward] 非定義宣言を導入してください。

また、C++では、ヘッダーファイルで外部のノンインライン関数を定義することは禁じられています。ヘッダーファイルの SkewNormalEvalutatable::SkewNormalEvalutatable , getSkewNormal , integrate などは、ヘッダーファイルには関係ありません。

また SkewNormalEvalutatable e(); C++の宣言は、関数 e オブジェクトではなく e を想定しているようです。単純な SkewNormalEvalutatable e; は、デフォルトのコンストラクタで初期化されたオブジェクトを宣言します。

また、最後のパラメータを受け取る integrate (そして sum ) 値で のオブジェクトとして Evaluatable 型になります。つまり SkewNormalEvalutatable の最後の引数として integrate は、結果的に SkewNormalEvalutatable にスライスされる。 Evaluatable . そのためポリモーフィズムは機能しません。もしポリモーフィックな動作をさせたいなら、このパラメータを参照かポインタで受け取り、値で受け取らないようにしなければなりません。