33

面白いものを見つけました。エラーメッセージはそれをすべて言います。非静的メンバー関数のアドレスを取得するときに括弧を許可しない理由は何ですか? gcc 4.3.4でコンパイルしました。

#include <iostream>

class myfoo{
    public:
     int foo(int number){
         return (number*10);
     }
};

int main (int argc, char * const argv[]) {

    int (myfoo::*fPtr)(int) = NULL;

    fPtr = &(myfoo::foo);  // main.cpp:14

    return 0;

}

エラー: main.cpp:14: エラー: ISO C++ では、メンバー関数へのポインターを形成するために、非修飾または括弧付きの非静的メンバー関数のアドレスを取得することを禁止しています。「&myfoo::foo」と言います

4

2 に答える 2

29

エラー メッセージから、括弧で囲まれた式のアドレスを取得することは許可されていないようです。それはあなたが書き直すことを示唆しています

fPtr = &(myfoo::foo);  // main.cpp:14

fPtr = &myfoo::foo;

これは、次の仕様 (§5.3.1/3) の一部によるものです。

メンバーへのポインターは、明示的な & が使用され、そのオペランドが括弧で囲まれていない修飾 ID である場合にのみ形成されます[...]

(私の強調)。なぜこれが規則なのかはわかりません (そして、今まで実際にこれを知りませんでした) が、コンパイラが不満を言っているようです。

お役に立てれば!

于 2011-08-20T19:44:04.567 に答える
16

次のコードを想像してください。

struct B { int data; };
struct C { int data; };

struct A : B, C {
  void f() {
    // error: converting "int B::*" to "int*" ?
    int *bData = &B::data;

    // OK: a normal pointer
    int *bData = &(B::data);
  }
};

括弧を使用したトリックがなければ、B のデータ メンバーへのポインターを直接取得することはできません (基本クラスのキャストとゲームが必要になりますが、thisこれは良くありません)。


ARM から:

メンバーへのポインターを取得するには、アドレス取得演算子を明示的に使用する必要があることに注意してください。暗黙的な変換はありません...もしあったとしたら、メンバー関数のコンテキストにあいまいさが生じます...たとえば、

void B::f() {
    int B::* p = &B::i; // OK
    p = B::i; // error: B::i is an int
    p = &i; // error: '&i'means '&this->i' which is an 'int*'

    int *q = &i; // OK
    q = B::i; // error: 'B::i is an int
    q = &B::i; // error: '&B::i' is an 'int B::*'
}

IS はこの標準化前の概念を維持し、メンバーへのポインターを取得しないように括弧を使用することを明示的に述べました。

于 2011-08-21T13:42:30.673 に答える