6

次の例を検討してください。

struct MyStruct {
    int a;
    int b;
};

マクロを使用して、構造体のインスタンスからメンバーを設定するには、次のようにします。

#define setVar(x,y) instance.x = y

次に、どの関数でも次のように言えます。

setVar(a, 4)

を文字列としてマクロに送信するにはどうすればよいですか? それも可能ですか?

setVar("a", 4)

編集:すべて double 型のメンバーを持つ事前定義された構造体がたくさんあります。渡された XML 構成ファイルで使用している構造体しかわかりません。解析後、設定する必要があるすべてのデータ メンバーと値のリストである一連の文字列が得られます。このリストを使用して、構造体の各メンバーの値を設定する必要があります。

4

3 に答える 3

9

たとえば、次のようなマクロを使用して構造体自体を定義する場合にのみ可能です。

#define MY_STRUCT_STRUCTURE FIELD(a) FIELD(b) FIELD(d) FIELD(e) FIELD(f)

struct MyStruct {

# define FIELD(name) int name;
    MY_STRUCT_STRUCTURE
# undef FIELD

  bool setVar(char* fieldname, int val)
  {
#   define FIELD(name) if(strcmp(#name,fieldname)==0){name=val; return true;};
    MY_STRUCT_STRUCTURE
#   undef FIELD
    return false; // name not found
  }
};


int main()
{
  MyStruct s;
  s.setVar("a",1);
  s.setVar("b",2);
  s.setVar("f",100);
}
于 2012-05-04T16:21:05.357 に答える
0

簡単で汚いコードをいくつかコーディングしましたが、いくつかのアイデアを提供できれば幸いです。ここでの主なトリックは、共用体を使いすぎることです。

struct MyStruct
{
int a;
double b;

MyStruct() 
    : a(0), b(0) {}
};

MyStruct instance;

union value 
{
    long value_a;
    double value_d;
} myvalue;


void blah_a(value v)
{
    instance.a = v.value_a;
}

void blah_b(value v)
{
    instance.b = v.value_d;
}

struct 
{
    (void)(*fn)(value);
    const char* key;
}

lookup_table[] = 
{
    { &blah_a, "a" },
    { &blah_b, "b" }
};

void setVar(const char* c, value v)
{
     for (int i = 0; lookup_table[i].fn; i++)
          if (c == lookup_table[i].key)
               (*(lookup_table[i].fn))(v);
}

int main(int argc, char* argv[])
{
    value v;
    v.value_a = 6;
    setVar("a", v);
    return 0;
}
于 2012-05-04T16:32:23.553 に答える
0

あなたが探しているものではないかもしれませんが、マクロなどの代替ソリューションは、カプセル化とオブジェクト指向の設計だけです。後で Field クラスをテンプレートに変更でき、基本的には何でも表現できます。

クラスを作成できます

class Field
{
public:
    Field(const std::string& name, const std::string& type);
    virtual ~Field(void);
    std::string toString() const;
    std::string getName() const;
    int getValue() const { return value };
private:
    std::string name;
    std::string type;
    int value;
};

そして、構造体クラス

#pragma once
#include <boost/ptr_container/ptr_deque.hpp>
#include <string>

class Field;

class MyStructure
{
public:
    typedef boost::ptr_deque<Field> FieldList;
    typedef FieldList::iterator FieldListIter;
    typedef FieldList::auto_type AutoField;

    MyStructure(void);
    MyStructure(const std::string& name);
    virtual ~MyStructure(void);

    void setName(const std::string& name);
    std::string getName() const;
    void addField( std::auto_ptr<Field> field );
    std::string getFieldValue( const std::string& name ) const;
    MyStructure::AutoField removeField( const std::string& name );
    std::string toString(void) const;

private:
    std::string struct_name;
    FieldList fields;
};

そして、それを使用するには:

auto_ptr<MySructure> struct_a(new MySructure("StructName1",0) );
struct_a->addField( auto_ptr<Field> ( new Field( "Field1",    1 ) ) );
struct_a->addField( auto_ptr<Field> ( new Field( var_str1,    2) ) );
struct_a->addField( auto_ptr<Field> ( new Field( getName(),   getVal() ) ) );
于 2012-05-04T16:32:40.747 に答える