سؤال

أقوم حاليًا بتنفيذ Tracer Ray Simple في C ++. لدي فصل يدعى OrthonorMalBasis ، والذي يولد ثلاثة متجهات وحدة متعامدة من ناقل أو اثنين محدد ، على سبيل المثال:

void
OrthonormalBasis::init_from_u ( const Vector& u )
{
    Vector n(1,0,0);
    Vector m(0,1,0);
    u_ = unify(u);
    v_ = cross(u_,n);
    if ( v_.length() < ONB_EPSILON )
        v_ = cross(u_,m);
    w_ = cross(u_,v_);
}

أقوم باختبار جميع أساليب مع إطار عمل غير unittest ++. تكمن المشكلة في أن هناك أكثر من حل واحد ممكن لوضع تقويمه الصحيح. على سبيل المثال هذا الاختبار:

TEST ( orthonormalbasis__should_init_from_u )
{
    Vector u(1,0,0);
    OrthonormalBasis onb;

    onb.init_from_u(u);

    CHECK_EQUAL( Vector( 1, 0, 0 ), onb.u() );
    CHECK_EQUAL( Vector( 0, 0, 1 ), onb.v() );
    CHECK_EQUAL( Vector( 0, 1, 0 ), onb.w() );
}

في بعض الأحيان ينجح ، في بعض الأحيان يفشل ، لأن المتجهات V و W يمكن أن يكون لها أيضًا 1 سلبي ، ولا تزال تمثل أساسًا تقويمًا صحيحًا. هل هناك طريقة لتحديد قيم متعددة متوقعة؟ أو هل تعرف طريقة أخرى للقيام بذلك؟

من المهم ، أن أحصل على القيم الفعلية والمتوقعة المطبوعة على stdout ، من أجل تصحيح الأساليب حتى لا يقوم هذا الحل بالمهمة:

TEST ( orthonormalbasis__should_init_from_u )
{
    Vector u(1,0,0);
    OrthonormalBasis onb;

    onb.init_from_u(u);

    CHECK_EQUAL( Vector( 1, 0, 0 ), onb.u() );
    CHECK(
        Vector( 0, 0, 1 ) == onb.v() ||
        Vector( 0, 0,-1 ) == onb.v() );
    CHECK(
        Vector( 0, 1, 0 ) == onb.w() ||
        Vector( 0,-1, 0 ) == onb.w() );
}
هل كانت مفيدة؟

المحلول

بالتأكيد إذا كان كل ما تختبره هو ما إذا كان أساسك هو عظام ، فهذا ما تحتاج إلى اختباره؟

// check orthogonality

CHECK_EQUAL( 0, dot(onb.u(), onb.v));
CHECK_EQUAL( 0, dot(onb.u(), onb.w));
CHECK_EQUAL( 0, dot(onb.v(), onb.w));

// check normality

CHECK_EQUAL( 1, dot(onb.u(), onb.u));
CHECK_EQUAL( 1, dot(onb.v(), onb.v));
CHECK_EQUAL( 1, dot(onb.w(), onb.w));

نصائح أخرى

أحد الاحتمالات هو إنشاء وظيفة check_multi الخاصة بك:

void CHECK_MULTI(TYPE actual, vector<TYPE> expected, const char* message)
{
  for (element in expected) {
    if (element == actual) {
      // there's a test here so the test count is correct
      CHECK(actual, element);
      return;   
    }
  }
  CHECK(actual, expected);
}

سأستخدم وظيفة أو فئة فائدة حتى تتمكن من فعل شيء مثل هذا:

CHECK_EQUAL(VectorList(0,0,1)(0,0,-1), onb.v());

بالنظر إلى ، فإن هذا التفسير للمساواة غريب إلى حد ما ، ولكن يجب أن يطبع لك كل القيم التي تريد رؤيتها دون الحاجة إلى تقديم ماكرو مخصص.
إذا كنت قلقا بشأن EQUAL في هذا السياق ، مثل ماكرو مخصص CHECK_CONTAINS() لا ينبغي أن يكون من الصعب القيام به.

VectorList سيتم بناؤها على أنها مؤقتة و operator() يتم استخدامها لإدراج القيم في القائمة الموجودة Vectorق ، على غرار Boost.assign.

النهج الأساسي:

class VectorList {
    std::vector<Vector> data_;
public:
    VectorList(double a, double b, double c) {
        data_.push_back(Vector(a,b,c));
    }
    VectorList& operator()(double a, double b, double c) {
        data_.push_back(Vector(a,b,c));
        return *this;
    }
    bool operator==(const Vector& rhs) const {
        return std::find(data_.begin(), data_.end(), rhs) != data_.end();
    }
};
مرخصة بموجب: CC-BY-SA مع الإسناد
لا تنتمي إلى StackOverflow
scroll top