如何从变量数据创建std :: initializer_list

时间:2013-05-02 22:49:03

标签: c++ c++11 initializer-list

我正在尝试使用传递给类构造函数的数据创建一个std::discrete_distribution对象。我知道如何使用静态数据创建它,但无法弄清楚如何使用可变数据(干净利落)。我现在所拥有的“有效”,但却很痛苦。有没有更合适的方法呢?

distInit = { distArray[0], ... };行是问题所在。

#include <iostream>
#include <iomanip>
#include <initializer_list>
#include <map>
#include <random>

class Die {
    private:
        int loadSide;
        double loadAmount;
        std::mt19937 generator;
        std::discrete_distribution<> distribution;
        std::initializer_list<double> distInit;
        std::array<double, 7> distArray;
    public:
        Die( int loadSide, double loadAmount ) : loadSide(loadSide), loadAmount(loadAmount) {
            distArray.fill( 1 );
            distArray[0] = 0;
            distArray[this->loadSide] = this->loadAmount;

            distInit = { distArray[0], distArray[1], distArray[2], distArray[3], distArray[4], distArray[5], distArray[6] };
            distribution.param( distInit );
        };
        int roll( ) {
                return distribution( generator );
        };
};

const int ROUNDS = 10000;

int main() {
    Die* die = new Die( 5, 20 );

    std::map<int, int> m;
    for(int n=0; n < ROUNDS; n++) {
        m[die->roll()]++;
    }
    for(auto p : m) {
        std::cout << p.first << " generated " << std::setiosflags(std::ios::fixed) << std::setprecision(2) << (float) p.second / ROUNDS << " times\n";
    }
}

我可能不会问正确的问题,如果是的话,我会提前道歉。这是一个很大的可能性,因为我很惊讶我无法在这个主题上找到任何(显然)相关的点击。

我的编译器是g++-mp-4.8 (MacPorts gcc48 4.8-20130411_0) 4.8.1 20130411 (prerelease)

命令行/opt/local/bin/g++-mp-4.8 -std=c++11 test.cpp -o test

3 个答案:

答案 0 :(得分:2)

如果您有可变数据,则应使用discrete_distribution constructor taking a pair of iterators

template< class InputIt >
discrete_distribution( InputIt first, InputIt last );

你不应该试图直接构建param_type;而是使用辅助函数来构建您的分布:

class Die {
    private:
        std::mt19937 generator;
        std::discrete_distribution<> distribution;
        static std::discrete_distribution<> makeDistribution(
            int loadSide, double loadAmount )
        {
            std::array<double, 7> distArray;
            distArray.fill( 1 );
            distArray[0] = 0;
            distArray[loadSide] = loadAmount;
            return {std::begin(distArray), std::end(distArray)};
        }
    public:
        Die( int loadSide, double loadAmount ) :
            generator{ },
            distribution{ makeDistribution( loadSide, loadAmount ) }
        {}
        int roll( ) {
                return distribution( generator );
        }
};

答案 1 :(得分:1)

std::initializer_list仅用作临时对象(函数参数)或局部变量。它不是一个容器,它没有任何东西;它是匿名临时数组的访问者。

标准包含一个类似于您的代码的示例,§8.5.4/ 6,其中提及

  

initializer_list对象在构造函数的ctor-initializer中初始化,因此数组只会持续存在,直到构造函数退出,因此在构造函数退出后使用i4元素会产生未定义的行为。

在你的情况下,它是构造函数的主体,而不是正文之前的ctor-initializer,但故事是相同的。你的程序现在正在运作,这真是太笨了。

要将分布存储在对象中,请使用std::arraystd::vectorarray效率更高,但不支持arr = { … }语法。 (有一些简单的替代方案。)vector确实支持使用大括号和=运算符的语法;此支持使用隐式std::initializer_list

答案 2 :(得分:0)

除了OP中显示的容器之外,我不知道从std::initializer_list等容器创建std::array的更好方法。

然而,对于原始问题,即将参数传递给分布,我可以提出更简单的建议。

    typedef std::discrete_distribution<>::param_type param_type;
    distribution.param(param_type(distArray.begin(), distArray.end()));

标准规定分发必须提供类型成员param_typeparam()采用的参数类型),但不指定它。但是,[rand.req.dist]说明了

  

对于D [分布类型]的每个构造函数,取参数对应于分布的参数,P [param_type]应具有相同的构造函数,这些构造函数具有相同的要求并且参数相同   数量,类型和默认值。

嗯,事实证明std::discrete_distribution<>有一个构造函数,它将迭代器指向参数范围。因此,无论std::discrete_distribution<>::param_type是什么,它都必须具有类似的构造函数。因此,我建议从param_typedistArray.begin()创建distArray.end()并将其传递给distribution.param()

附注:您的班级不再需要std::initializer_list<double> distInit;。在我看来,你不需要std::array<double, 7> distArray作为类成员(它可能是Die的构造函数中的局部变量)。