6

我正在使用karma来生成大型结构的表示,但是在生成过程中会复制这些结构。我认为他们不需要,所以想知道如何避免它。

下面的快速示例打印“复制!”,因为目标结构被复制到rule::generate

namespace karma = spirit::karma;
namespace phoenix = boost::phoenix;

struct foo
{
    foo() { }
    foo( foo const &other ) { std::cout << "Copy!"; }
    int f() const { return 42; }
};

std::string output;
typedef std::back_insert_iterator< std::string > iterator;
karma::rule< iterator, foo() > foo_rule = 
    karma::int_[ karma::_1 = phoenix::bind( &foo::f, karma::_val ) ];
foo my_foo;
iterator it( output );
karma::generate( it, foo_rule, my_foo );

foo_rule我可以通过引用声明的属性来停止复制:

karma::rule< iterator, foo &() > foo_rule

但这不适用于向量 [显然foos 因此是可复制的,但在向量构造时复制可能很便宜,但在生成时复制成本高:-)]

下面的示例打印“复制!” 生成期间五次(即在向量 ctor 期间忽略副本);10 次 iffoo_rule的属性不是参考:

std::vector<foo> my_vec_foo(5);
karma::rule< iterator, std::vector<foo>() > vec_foo_rule = *foo_rule;
karma::generate(it, vec_foo_rule, my_vec_foo);

在 VC 2008 上使用 Boost 1.47 无法编译这两个规则的引用。也就是说,使用:

karma::rule< iterator, foo &() > foo_rule /* = ... */;
karma::rule< iterator, std::vector<foo> &() > vec_foo_rule /* = ... */;

我用andextract_from_container实例化。在 extract_from.hpp 的第 131 行,它尝试形成并且编译器在创建 refrence-to-reference 时失败。Attribute = std::vector<foo>Exposed=std::vector<foo> &Exposed const &

我觉得我错过了一些东西,所以任何指针将不胜感激!

4

1 回答 1

3

我相信你已经尝试过了,但我还是会说出来。您是否尝试过如下操作:

std::vector<foo> my_vec_foo(5);
karma::rule< iterator, std::vector<foo>&() > vec_foo_rule = *foo_rule;
karma::generate(it, vec_foo_rule, my_vec_foo);

更新我刚刚用下面的代码片段测试了它(g++ 4.6 with Boost 1.47.0)。它证实了上述工作。但是,存在混淆的空间,因为该std::vector<foo> my_vec_foo(5)文件还将显示正在制作 5 个副本。请参阅代码和输出中的 BIG LETTER 警告:

#include <boost/spirit/include/karma.hpp>
#include <boost/spirit/include/phoenix.hpp>

namespace karma = boost::spirit::karma;
namespace phoenix = boost::phoenix;

struct foo
{
    foo() { }
    foo( foo const &other ) { std::cerr << "Copy!\n"; }
    int f() const { return 42; }
};

int main()
{
    std::string output;
    typedef std::back_insert_iterator< std::string > iterator;
    iterator it( output );
    karma::rule< iterator, foo&() > foo_rule = 
        karma::int_[ karma::_1 = phoenix::bind( &foo::f, karma::_val ) ];

    foo my_foo;
    karma::generate( it, foo_rule, my_foo );

    std::vector<foo> my_vec_foo(5);

    std::cerr << "\nSTART WATCHING NOW" << std::endl;

    karma::rule< iterator, std::vector<foo>&() > vec_foo_rule = *foo_rule;
    karma::generate(it, vec_foo_rule, my_vec_foo);
}

输出:

Copy!
Copy!
Copy!
Copy!
Copy!

START WATCHING NOW
于 2011-11-09T22:39:00.367 回答