2017-04-16 96 views
3

我想在我的類中包含指向成員函數的表。 MS C++(Visual Studio 2015)編譯器抱怨類中_requests表的初始值設定項以及std :: bind的用法。編譯器的錯誤消息不是非常有幫助的:如何聲明指向類初始化程序中成員函數的指針並調用這些指針?

1> d:\ TEMP \ win32project1 \ win32project1 \ source1.cpp(59):錯誤C3867: 'TLV :: parse_read':非標準語法;使用 '&' 創建一個指針 構件

1> d:\ TEMP \ win32project1 \ win32project1 \ source1.cpp(62):錯誤C3867: 'TLV :: parse_write':非標準語法;使用 '&' 創建一個指針 到構件

1> d:\ TEMP \ win32project1 \ win32project1 \ source1.cpp(40):錯誤C2440: '初始化':無法從「STD轉換:: _粘合劑& >」到 '的std ::函數'

1> d:\ TEMP \ win32project1 \ win32project1 \ source1.cpp(40):注意:沒有 構造可以採取源類型或構造函數重載 分辨率爲曖昧

放置一個安培沙在表中的日常名稱前面只是改變了錯誤的非法操作上綁定的成員函數

#include <cstdint> 
#include <functional> 
#include <vector> 
#include <iostream> 

using namespace std; 

enum TLV_TYPE : uint32_t 
    { 
    tlv_type_read, 
    tlv_type_write 
    }; 

typedef struct 
    { 
    TLV_TYPE type; 
    uint32_t  length; 
    } TLV_RECORD, *pTLV_RECORD; 

typedef function <uint32_t (const TLV_RECORD& Record)> pTLV_PARSER; 

typedef struct 
    { 
    TLV_TYPE   type; 
    pTLV_PARSER   parse_routine; 
    } TABLE, *pTABLE; 

class tlv 
    { 
    public: 
     tlv() 
      { 
      }; 
     ~tlv() 
      { 
      }; 

     uint32_t start_parse (const TLV_RECORD& Record) 
      { 
      pTLV_PARSER parser = std::bind (_requests [0].parse_routine, this, placeholders::_1); 
      parser (Record); 
      return 0; 
      }; 

    protected: 

     uint32_t parse_read (const TLV_RECORD& Record) 
      { 
      cout << "parse read: type " << Record.type << endl; 
      return 0; 
      } 

     uint32_t parse_write (const TLV_RECORD& Record) 
      { 
      cout << "parse write: type " << Record.type << endl; 
      return 0; 
      } 

     const TABLE _requests [2] = 
      { 
       {tlv_type_read, parse_read}, 
       {tlv_type_write, parse_write}, 
      }; 

    }; // End of class tlv 

int main() 
    { 
    tlv   foo; 
    TLV_RECORD rec = {tlv_type_read, 4}; 

    foo.start_parse (rec); 
    return 0; 
    } 
+0

'{tlv_type_read,parse_read}'應'{tlv_type_read,&parse_read}'和同上,用於下面的行,閉嘴前兩個編譯器警告。 – cdhowie

+0

*「編譯器的錯誤消息不是非常有用:」* - Visual Studio 2015? ...與十年前C++編譯器產生的錯誤消息相比,這是一個非常有用的錯誤消息。 – WhiZTiM

回答

2

一種方法是在tlv的構造函數初始值設定項列表中初始化_requests,以便可以綁定一個this指針。我還建議使用lambda函數而不是std :: bind,因爲它們通常更快。

tlv() : 
    _requests{ 
     {tlv_type_read, [this](const TLV_RECORD& Record){ return parse_read(Record); }}, 
     {tlv_type_write, [this](const TLV_RECORD& Record){ return parse_write(Record); }} 
    } 
{ 
} 

uint32_t start_parse (const TLV_RECORD& Record) 
{ 
    pTLV_PARSER parser = _requests[0].parse_routine; 
    parser (Record); 
    return 0; 
}; 

const TABLE _requests [2]; 
+0

感謝您的幫助,它的工作 – Brian

2

什麼你有效地試圖在你的tlv::_requests初始化做的是轉換指針到成員到std::function但沒有隱式轉換。

看起來正確的解決辦法是改變你的TABLE.parse_routine成員是一個指針到成員函數而不是std::function(你不需要爲std::function你在做什麼):

uint32_t (tlv::*parse_routine)(const TLV_RECORD &); 

(編譯器要接受這一點,你需要轉發由TABLE結構的定義之前加入class tlv;聲明tlv類)。

您還需要調整你的語法服用成員指針:

  {tlv_type_read, parse_read}, 
      {tlv_type_write, parse_write}, 

應該是:

  {tlv_type_read, &tlv::parse_read}, 
      {tlv_type_write, &tlv::parse_write}, 

然後,調用這個函數的時候,你並不需要使用std::bind可言,只是做:

(this->*(_requests[0].parse_routine))(Record); 
+0

感謝您的幫助。編譯器被parse_routine指針的定義弄糊塗了;它不編譯,因爲它不喜歡類標籤,它聲稱它是一個函數返回函數,並且'uint32'是一個缺少的返回類型。我添加了一個類tlv的聲明;就在結構的定義之前。 – Brian

+0

我的不好,既然你在類之前聲明瞭這個結構,你需要在定義TABLE結構之前添加一個類的前向聲明('class tlv;'),以便編譯器知道'tlv'存在。而'uint32'應該是'uint32_t',只是我的一個錯字。我會將這些信息添加到我的答案中。 – cdhowie

+0

我得到就行聲明parse_routine以下錯誤: 1> d:\ TEMP \ win32project1 \ win32project1 \ source1.cpp(28):警告C4121: '':一個構件的對準是包裝 然後敏感,在聲明表初始值設定項的行上,我得到: 1> d:\ temp \ win32project1 \ win32project1 \ source1.cpp(80):錯誤C2276:'&':綁定成員函數表達式上的非法操作 最後,調用parse_routine的行,我得到: 1> d:\ temp \ win32project1 \ win32project1 \ source1.cpp(62):錯誤C2296:'。*':非法,左操作數的類型爲'tlv * const' – Brian