2017-06-13 112 views
2

我一直在试图编写一段非常简单的代码,但它似乎不工作,我无法从编译器中做出任何意义错误。无法将std :: bind返回值分配给std :: function

代码:

#include <iostream> 
#include <sstream> 
#include <functional> 

class c 
{ 
public: 
    void test(std::stringstream ss){ 
     std::cout<<ss.str()<<std::endl; 
    } 

    void test2(std::stringstream ss){ 
     const auto bound=std::bind(&c::test,this, ss); 
     std::function<void()> f(bound); 
     f(); 
    } 

}; 

void test1(const std::stringstream ss){ 
    std::cout<<ss.str()<<std::endl; 
} 

int main(){   
    std::stringstream ss; 
    ss<<"hello world"; 

    //first 
    const auto bound=std::bind(&test1,ss); 
    std::function<void()> f(bound); 
    f(); 
    //second 
    C obj; 
    obj.test2(ss); 
    return 0;    
}     

错误:g++ -std=c++14 bind.cpp

bind.cpp:14:32: error: no matching function for call to ‘std::function<void()>::function(const std::_Bind<std::_Mem_fn<void (c::*)(std::__cxx11::basic_stringstream<char>)>(c*, std::__cxx11::basic_stringstream<char>)>&)’ 
    std::function<void()> f(bound); 
          ^
bind.cpp:30:31: error: no matching function for call to ‘std::function<void()>::function(const std::_Bind<void (*(std::__cxx11::basic_stringstream<char>))(std::__cxx11::basic_stringstream<char>)>&)’ 
    std::function<void()> f(bound); 
          ^

我与编译。 我看到here其中接受的答案建议使用lambdas而不是std :: bind,但任何人都可以告诉为什么上述代码中的第一次和第二次使用都不起作用?

回答

4

您面临的问题与std::stringstream没有复制构造函数有关。

这些问题可以通过在几个函数中使用const std::stringstream&作为参数类型而不是std::stringstream来解决。

#include <iostream> 
#include <sstream> 
#include <functional> 

class c 
{ 
    public: 

    // Change the argument to a const& 
    void test(std::stringstream const& ss){ 
     std::cout<<ss.str()<<std::endl; 
    } 

    // Use std::cref to use a const& in the call to std::bind 
    void test2(std::stringstream ss){ 
     const auto bound=std::bind(&c::test,this, std::cref(ss)); 
     std::function<void()> f(bound); 
     f(); 
    } 

}; 

// Change the argument to a const& 
void test1(const std::stringstream & ss){ 
    std::cout<<ss.str()<<std::endl; 
} 

int main(){   
    std::stringstream ss; 
    ss<<"hello world"; 

    //first 
    // Use std::cref to use a const& in the call to std::bind 
    const auto bound = std::bind(&test1, std::cref(ss)); 
    std::function<void()> f = bound; 
    f(); 
    return 0;    
}