如何处理初始化类的大量属性

How do I handle a large number of properties that initialize a class?

本文关键字:属性 初始化 何处理 处理      更新时间:2023-10-16

我有一个类需要大量属性进行初始化:

Hand hand = new Hand();
hand->skinColor(Color(0.5, 0.5, 0.2));
hand->indexFingerLength(0.5);
hand->middleFingerLength(0.6);
hand->ringFingerLength(0.55);
...
hand->init(); // Builds the hand model

这些属性只能在调用init()之前进行修改,init()使用这些属性来构建类。问题是,在调用init()后,这些属性仍然可以被修改,但没有效果,这个接口可能会误导用户认为它们会产生效果。

除了将所有属性移动到init()的参数列表中之外,还有更好的方法来重构它吗?

struct structHandParms
{
  param1;
  param2;
  param3;
  ...
};
structHandParms stParams;
Hand hand = new Hand(stParams)

您可以添加一个传递给构造函数的"Hand_Init"结构:

Hand_Init hand_init;
hand_init.skinColor = Color(0.5, 0.5, 0.2);
hand_init.indexFingerLength = 0.5;
hand_init.middleFingerLength = 0.6;
hand_init.ringFingerLength = 0.55;
Hand hand = new Hand(hand_init);

这样一来,一旦构造函数调用完成,手就被初始化了,就不会有一堆可能导致混乱的匿名参数。

如果要创建多个Hand,那么可以使用类似于Builder Pattern的东西。

// set properties
HandBuilder hb;
hb.skinColor(Color(0.5, 0.5, 0.2));
hb.indexFingerLength(0.5);
hb.middleFingerLength(0.6);
hb.ringFingerLength(0.55);
// create object
Hand hand = hb.buildHand();

这完全要求在构造函数中包含逻辑,而不是后面跟着init的单独方法。然后,很明显(并由编译器强制执行)以后不能更改它们。如果有很多参数,请将它们全部分组到一个结构中,该结构将被填充并传递到构造函数中,而不是25个单独的参数。

"对init()的调用,它使用它们来构建类"真的构建了一个类,还是初始化了这个类的实例?

为什么不在ctor中进行整个初始化,并将所需的值作为参数传递?

我认为这是为您的项目维护文档和用户手册的一个很好的理由,或者您可以更改为向类发送描述符。然后,创建描述符,用配置填充它,并将所有信息一次性发送给构造函数。

您可以通过抛出异常来解决后init()修改的问题。这将防止用户调用其中一个设置函数并期望它执行某些操作。

作为替代方案,您可以创建一个HandParameter类/结构。该对象将负责收集有关Hand对象所需的所有数据。接下来,您可以使Hand对象的init()函数将HandParameter作为其唯一的参数。

struct HandParam
{
    // Whatever settings you need
};
class Hand
{
    bool initialized;
    // other private data
    public:
        Hand();
        Hand( const HandParam& params ); // Calls the init() funciton with params
        void init( const HandParam& params ); // sets up hand and sets initialized to true
    // Whatever else you need in your class
};

祝你好运!