3

次の定義があります。

class PartitioningMethod {
public:
  virtual void addConstraints(ConstraintManager& cm) = 0;
  virtual bool hasMoreConstraints() = 0;
  virtual void setQuery(const Query& q) = 0;
  virtual ~PartitioningMethod(){ }
};


class Random : public PartitioningMethod {
private:
  vector< ref<Expr> > constraints;
  vector< ref<Expr> >::iterator it;
  vector< ref<Expr> >::iterator end;
  int numConstraints;
  RNG theRNG;

public:
  void setQuery(const Query& q) { 

    constraints.clear();

    //Set random number
    //srand ( unsigned ( time (NULL) ) * theRNG.getInt32() );
    srand ( theRNG.getInt32() );

    //Copy constraints    
    copy(q.constraints.begin(),q.constraints.end(),std::back_inserter(constraints));

    //Shuffle Randomly
    std::random_shuffle(constraints.begin(),constraints.end(), p_myrandom);

    it = constraints.begin();
    end = constraints.end();
    numConstraints = constraints.size();
  }

  void addConstraints(ConstraintManager& cm) {
    int step = rand() % numConstraints + 1;
    while(step != 0) {
      cm.addConstraint(*it);
      ++it;
      --step;
      --numConstraints;
    }   
  }

  bool hasMoreConstraints() {
    return it != end;
  }
};


bool PartitioningSolver::computeInitialValues(const Query& query,
                            const std::vector<const Array*> &objects,
                            std::vector< std::vector<unsigned char> > &values,
                            bool &hasSolution) {

  fprintf(stderr,"INIT\n");  
  // If there are no constraints in the query
  if(query.constraints.size() == 0 || query.constraints.size() == 1)
    return solver->impl->computeInitialValues(query, objects, values, hasSolution);

  // If the number constraints in the query are > 0 
  method->setQuery(query);

  ConstraintManager cm;
  ref<Expr> expr = query.expr;

  fprintf(stderr,"Begin partitioning\n");
  fprintf(stderr,"---------------------\n");  

  while(method->hasMoreConstraints()){
    fprintf(stderr, "HERE");
    //Add Constraints
    method->addConstraints(cm);

    //Construct a query
    Query temp_query(cm,expr);

     ExprPPrinter::printQuery(std::cerr,temp_query.constraints,temp_query.expr); 
     fprintf(stderr,"---------------------\n");

    //Query STP to check if satisfiable
    values.clear(); 

    if(!solver->impl->computeInitialValues(temp_query, objects, values, hasSolution))
      return false;

    //If not, return immediately (a win!) 
    if(!hasSolution)
      return true; 

    //If a solution is returned, check if the solution satisfies the entire set of constraints
    vector<const Array*> obj = objects;
    Assignment solution(obj, values);
    bool satisfiesAll = checkSolution(solution, query.constraints);

    //  fprintf(stderr,"Satisfies all: %i\n", satisfiesAll);

    // If it is successful, return the solution (a win again!), 
    if(satisfiesAll)
      return true;

    // If not add more constraints (if there is more) and repeat
  }
  return true;
}

パーティショニング ソルバー クラスの部分的な定義:

class PartitioningSolver : public SolverImpl {
private:
  Solver*             solver;
  PartitioningMethod* method;
  bool checkSolution(Assignment& solution,  const ConstraintManager& constraints);  
public:
  PartitioningSolver(Solver *s,  PartitioningMethod* pm) : solver(s), method(pm) { }
  ~PartitioningSolver() { delete solver; delete method; }
};

このような長いコード スニペットを貼り付けて申し訳ありませんが、私は何時間も作業しており、エラーが発生し続けています。

pure virtual method called
terminate called without an active exception

何が悪いのかわからない。が配置されているcomputeInitialValues関数で失敗するようfprintf(stderr,"Begin partitioning\n");です。最後の手段として印刷ステートメントを追加しようとしましたが、何も印刷しません..どんなアイデアでも大歓迎です。

編集:
わかりましたので、名前を Random から Ran に変更すると、機能し始めました。私はこのクラスのインスタンスをその場で new Random() の引数として作成していましたが、別のコンストラクターまたは私が知らない何かと混同していたと思います..

4

2 に答える 2

9

別の種類のバグがあり、このエラー メッセージが出力される可能性があります。

オブジェクトを削除し、後でそれを呼び出そうとしています。これは未定義の動作であり、一部のコンパイラでは、運が良ければ、それが表示されます。コードを valgrind で実行してみてください。

http://tombarta.wordpress.com/2008/07/10/gcc-pure-virtual-method-called/

于 2012-08-20T21:01:03.180 に答える
2

含まれていないコードで、コンストラクターから純粋仮想関数を呼び出しています。

基本クラスのコンストラクターがこのオブジェクトで仮想関数を呼び出すと、派生クラスによるその仮想関数のオーバーライドが呼び出されないのはなぜですか?

于 2012-08-20T20:54:02.763 に答える