<random>
Define instalações para a geração aleatória de números, permitindo a criação de números aleatórios distribuídos uniformemente.
#include <random>
Resumo
Um gerador de número aleatório é um objeto que produz uma sequência de valor pseudoaleatórios. Um gerador que produz valores que são distribuídos uniformemente em um intervalo especificado é um URNG (Gerador de Número Aleatório Uniforme). Uma classe de modelo designada para funcionar como um URNG é referida como um mecanismo se essa classe possuir determinados traços em comum, discutidos posteriormente neste artigo. Um URNG pode ser, e geralmente é, combinado a uma distribuição transmitindo o URNG como um argumento para que o operator() da distribuição produza valores que são distribuídos de maneira definida pela distribuição.
Esses links levam para as seções principais deste artigo:
Exemplos de código
Lista categorizada
Mecanismos e distribuições
Comentários
Dicas rápidas
Aqui estão algumas dicas para manter em mente ao usar o <random>:
Para a maioria dos propósitos, os URNGs produzem bits brutos que podem ser moldados pelas distribuições. (Uma exceção notável para isso é std::shuffle(), uma vez que ela utiliza um URNG diretamente.)
Uma única instanciação de um URNG ou distribuição não pode ser chamada com segurança simultaneamente, pois executar um URNG ou uma distribuição é uma operação de modificação. Para obter mais informações, consulte Segurança de threads na Biblioteca Padrão C++.
São fornecidos typedefs predefinidos de diversos mecanismos; esse é o modo preferido de criar um URNG se um mecanismo estiver sendo usado.
O emparelhamento mais útil para a maioria dos aplicativos é o mecanismo mt19937 com uniform_int_distribution, como mostrado no exemplo de código posteriormente neste artigo.
Há muitas opções dentre as quais escolher no cabeçalho <random>, e deve-se dar preferência a qualquer uma delas em relação à antiga função de Tempo de Execução Crand(). Para obter informações sobre o que há de errado com o rand() e como o <random> lida com esses imprevistos, veja este vídeo.
Exemplos
O exemplo de código a seguir mostra como gerar alguns números aleatórios;nesse caso cinco deles usando um gerador criado com semente não determinista.
#include <random>
#include <iostream>
using namespace std;
int main()
{
random_device rd; // non-deterministic generator
mt19937 gen(rd()); // to seed mersenne twister.
// replace the call to rd() with a
// constant value to get repeatable
// results.
for (int i = 0; i < 5; ++i) {
cout << gen() << " "; // print the raw output of the generator.
}
cout << endl;
}
Saída:
Ao passo que esses são números aleatórios de alta qualidade e diferentes sempre que este programa seja executado, eles não são necessariamente em um intervalo útil. Para controlar o intervalo, use uma distribuição uniforme, conforme mostrado no código a seguir:
#include <random>
#include <iostream>
using namespace std;
int main()
{
random_device rd; // non-deterministic generator
mt19937 gen(rd()); // to seed mersenne twister.
uniform_int_distribution<> dist(1,6); // distribute results between 1 and 6 inclusive.
for (int i = 0; i < 5; ++i) {
cout << dist(gen) << " "; // pass the generator to the distribution.
}
cout << endl;
}
Saída:
O seguinte exemplo de código mostra um conjunto mais realista de casos de uso com geradores de números aleatórios uniformemente distribuídos misturando o conteúdo de um vetor e uma matriz.
// cl.exe /EHsc /nologo /W4 /MTd
#include <algorithm>
#include <array>
#include <iostream>
#include <random>
#include <string>
#include <vector>
#include <functional> // ref()
using namespace std;
template <typename C> void print(const C& c) {
for (const auto& e : c) {
cout << e << " ";
}
cout << endl;
}
template <class URNG>
void test(URNG& urng) {
// Uniform distribution used with a vector
// Distribution is [-5, 5] inclusive
uniform_int_distribution<int> dist(-5, 5);
vector<int> v;
for (int i = 0; i < 20; ++i) {
v.push_back(dist(urng));
}
cout << "Randomized vector: ";
print(v);
// Shuffle an array
// (Notice that shuffle() takes a URNG, not a distribution)
array<string, 26> arr = { { "H", "He", "Li", "Be", "B", "C", "N", "O", "F",
"Ne", "Na", "Mg", "Al", "Si", "P", "S", "Cl", "Ar", "K", "Ca", "Sc",
"Ti", "V", "Cr", "Mn", "Fe" } };
shuffle(arr.begin(), arr.end(), urng);
cout << "Randomized array: ";
print(arr);
cout << "--" << endl;
}
int main()
{
// First run: non-seedable, non-deterministic URNG random_device
// Slower but crypto-secure and non-repeatable.
random_device rd;
cout << "Using random_device URNG:" << endl;
test(rd);
// Second run: simple integer seed, repeatable results
cout << "Using constant-seed mersenne twister URNG:" << endl;
mt19937 engine1(12345);
test(engine1);
// Third run: random_device as a seed, different each run
// (Desirable for most purposes)
cout << "Using non-deterministic-seed mersenne twister URNG:" << endl;
mt19937 engine2(rd());
test(engine2);
// Fourth run: "warm-up" sequence as a seed, different each run
// (Advanced uses, allows more than 32 bits of randomness)
cout << "Using non-deterministic-seed \"warm-up\" sequence mersenne twister URNG:" << endl;
array<unsigned int, mt19937::state_size> seed_data;
generate_n(seed_data.begin(), seed_data.size(), ref(rd));
seed_seq seq(begin(seed_data), end(seed_data));
mt19937 engine3(seq);
test(engine3);
}
Saída de exemplo e comentários de códigos
Esse código demonstra duas aleatorizações diferentes, aleatorizar um vetor de inteiros e embaralhar uma matriz de dados indexados, com uma função de modelo de teste. A primeira chamada para a função de teste usa URNG protegido por criptografia, não determinista, não passível de propagação e não repetível random_device. A segunda execução do teste usa mersenne_twister_engine como URNG, com semeação constante determinista de 32 bits, o que significa que os resultados podem ser repetidos. A terceira execução do teste propaga mersenne_twister_engine com um resultado não determinista de 32 bits de random_device. A quarta execução do teste é expandida usando uma sequência de semeação preenchida com resultados de random_device, que oferece efetivamente mais do que a aleatorização não determinista de 32 bits (mas ainda sem proteção por criptografia). Para obter mais informações, continue lendo.
[Vá para a parte superior da página]
Lista categorizada
Geradores de Números Aleatórios Uniformes
URNGs geralmente são descritos de acordo com essas propriedades:
Período: Quantas iterações são necessárias para repetir a sequência de números gerada. Quanto maior, melhor.
Desempenho: A velocidade com que os números podem ser gerados e quanta memória isso ocupa. Quanto menor, melhor.
Qualidade: A proximidade de uma sequência gerada de números verdadeiramente aleatórios. Isso geralmente é chamado de "aleatoriedade".
As seções a seguir listam os URNGs (geradores de números aleatórios uniformes) fornecidos no cabeçalho <random>.
[Vá para a parte superior da página]
Gerador Não Determinista
Gera uma sequência aleatória não determinista protegida por criptografia usando um dispositivo externo. Geralmente usado para propagar um mecanismo. Baixo desempenho, qualidade bastante alta. Para obter mais informações, consulte comentários. |
[Vá para a parte superior da página]
Typedefs de Mecanismo com Parâmetros Predefinidos
Para mecanismos de instanciação e adaptadores de mecanismos. Para obter mais informações, consulte mecanismos e distribuições.
Nome |
Descrição |
---|---|
default_random_engine |
Definição do tipo para o mecanismo padrão.
|
knuth_b |
Mecanismo de Knuth.
|
minstd_rand0 |
Mecanismo padrão mínimo de 1988 (Lewis, Goodman e Miller, 1969).
|
minstd_rand |
Mecanismo padrão mínimo atualizado minstd_rand0 (Park, Miller e Stockmeyer, 1993).
|
mt19937 |
Mecanismo Mersenne Twister de 32 bits (Matsumoto e Nishimura, 1998).
|
mt19937_64 |
Mecanismo Mersenne Twister de 64 bits (Matsumoto e Nishimura, 2000).
|
ranlux24 |
Mecanismo RANLUX de 24 bits (Martin Lüscher e Fred James, 1994).
|
ranlux24_base |
Usado como base para ranlux24.
|
ranlux48 |
Mecanismo RANLUX de 48 bits (Martin Lüscher e Fred James, 1994).
|
ranlux48_base |
Usado como base para ranlux48.
|
[Vá para a parte superior da página]
Modelos de Mecanismos
Modelos de mecanismos são usados como URNGs independentes ou como mecanismos de base transmitidos a adaptadores de mecanismo. Geralmente, eles são instanciados com um typedef de mecanismo predefinido e transmitidos a uma distribuição. Para obter mais informações, consulte a seção Mecanismos e Distribuições.
Gera uma sequência aleatória usando o algoritmo congruente linear. Mais simplista e com qualidade mais baixa. |
|
Gera uma sequência aleatória usando o algoritmo mersenne twister. Mais complexo e de qualidade mais alta, exceto pela classe random_device. Desempenho muito rápido. |
|
Gera uma sequência aleatória usando o algoritmo de subtração com transferência. Um aprimoramento em relação ao linear_congruential_engine, mas com qualidade e desempenho muito menores que mersenne_twister_engine. |
[Vá para a parte superior da página]
Modelos de Adaptador de Mecanismo
Adaptadores de mecanismo são modelos que adaptam outros mecanismos (básicos). Geralmente, eles são instanciados com um typedef de mecanismo predefinido e transmitidos a uma distribuição. Para obter mais informações, consulte a seção Mecanismos e Distribuições.
Gera uma sequência aleatória descartando valores retornados pelo mecanismo de base. |
|
Gera uma sequência aleatória com um número especificado de bits recompactando bits dos valores retornados pelo mecanismo básico. |
|
Gera uma sequência aleatória reordenando os valores retornados pelo mecanismo de base. |
[Vá para a parte superior da seção]
[Vá para a parte superior da página]
Distribuições de Número Aleatório
As seções a seguir listam as distribuições fornecidas no cabeçalho <random>. As distribuições são um mecanismo de pós-processamento, que geralmente usam uma saída de URNG como entrada e distribuem a saída de acordo com uma função de densidade probabilística estatística definida. Para obter mais informações, consulte a seção Mecanismos e Distribuições.
[Vá para a parte superior da página]
Distribuições Uniformes
Produz uma distribuição de valor inteiro uniforme em um intervalo no intervalo fechado [a, b] (inclusivo-inclusivo). |
|
Produz uma distribuição de valor real uniforme (ponto flutuante) em um intervalo no intervalo [a, b) (inclusivo-exclusivo). |
|
Produz uma distribuição igual de valores reais (ponto flutuante) de uma determinada precisão em [0, 1) (inclusivo-exclusivo). |
[Vá para a parte superior da seção]
Distribuições Bernoulli
Produz uma distribuição Bernoulli de valores bool. |
|
Produz uma distribuição binomial de valores inteiros. |
|
Produz uma distribuição geométrica de valores inteiros. |
|
Produz uma distribuição binomial negativa de valores inteiros. |
[Vá para a parte superior da seção]
Distribuições normais
Produz uma distribuição de Cauchy de valores reais (ponto flutuante). |
|
Produz uma distribuição qui-quadrada de valores reais (ponto flutuante). |
|
Produz uma distribuição F (também conhecida como distribuição F de Snedecor ou distribuição de Fisher–Snedecor) de valores reais (ponto flutuante). |
|
Produz uma distribuição normal de log de valores reais (ponto flutuante). |
|
Produz uma distribuição normal (Gaussian) de valores reais (ponto flutuante). |
|
Produz uma distribuição t Student de valores reais (ponto flutuante). |
[Vá para a parte superior da seção]
Distribuições Poisson
Produz uma distribuição exponencial de valores reais (ponto flutuante). |
|
Produz uma distribuição de valor extremo de valores reais (ponto flutuante). |
|
Produz uma distribuição gama de valores reais (ponto flutuante). |
|
Produz uma distribuição Poisson de valores inteiros. |
|
Produz uma distribuição Weibull de valores reais (ponto flutuante). |
[Vá para a parte superior da seção]
Distribuições de Amostragem
Produz uma distribuição inteira discreta. |
|
Produz uma distribuição constante em partes de valores reais (ponto flutuante). |
|
Produz uma linear constante em partes de valores reais (ponto flutuante). |
[Vá para a parte superior da seção]
[Vá para a parte superior da página]
Funções do Utilitário
Esta seção lista as funções de utilitário gerais fornecidas no cabeçalho <random>.
Gera uma sequência de transmissão embaralhada e não polarizada. Usadas para evitar a replicação de transmissões variadas aleatórias. Útil quando vários URNGs são instanciados dos mecanismos. |
Operadores
Esta seção lista os operadores fornecidos no cabeçalho <random>.
operator== |
Testa se o URNG no lado esquerdo do operador é igual ao mecanismo do lado direito. |
operator!= |
Testa se o URNG no lado esquerdo do operador não é igual ao mecanismo do lado direito. |
operator<< |
Grava informações de estado em uma transmissão. |
operator>> |
Extrai informações de estado de uma transmissão. |
[Vá para a parte superior da página]
Mecanismos e distribuições
Consulte as seções a seguir para obter informações sobre cada um desses categorias de classe de modelo definidas em <random>. Ambas essas categorias de classe de modelo assumem um tipo como um argumento e usam nomes de parâmetro de modelo compartilhados para descrever as propriedades de tipo que são permitidas como um tipo de argumento real, como a seguir:
IntType indica um short, int, long, long long, unsigned short, unsigned int, unsigned long, ou unsigned long long.
UIntType indica unsigned short, unsigned int, unsigned long, ou unsigned long long.
RealType indica um float, double, ou long double.
Mecanismos
Mecanismos de e adaptadores do mecanismo são modelos cujos parâmetros personalizam o gerador criado.
Um mecanismo é uma classe ou classe de modelos cujas instâncias (geradores) agem como uma fonte de números aleatórios distribuídos uniformemente entre um valor mínimo e um máximo. Um adaptador do mecanismo oferece uma sequência de valores que possuem propriedades de aleatoriedade diferentes assumindo valores produzidos por outro mecanismo de número aleatório e aplicando um algoritmo de algum tipo a esses valores.
Cada mecanismo e adaptador do mecanismo possuem os seguintes membros:
typedef numeric-type result_type é o tipo retornado pelo operator()do gerador. O numeric-type é transferido como um parâmetro de modelo na instanciação.
result_type operator() retorna valores que são distribuídos uniformemente entre min() e max().
result_type min() retorna o valor mínimo retornado pelo operator()do gerador. Adaptadores do mecanismo usam o resultado min() do mecanismo de base.
result_type max() retorna o valor máximo retornado pelo operator()do gerador. Quando result_type é um tipo integral (com valor inteiro), max() é o valor máximo que pode ser retornado de fato (inclusivo); quando result_type é um tipo de ponto flutuante (com valor real), max() é o menor valor superior a todos os valores que podem ser retornados (não inclusivo). Adaptadores do mecanismo usam o resultado max() do mecanismo de base.
void seed(result_type s) propaga o gerador com o valor de semente s. Para mecanismos, a assinatura é void seed(result_type s = default_seed) para suporte ao parâmetro padrão (adaptadores do mecanismo definem um void seed() separado, consulte a próxima subseção).
template <class Seq> void seed(Seq& q) propaga o gerador usando um seed_seq Seq.
Um construtor explícito com o argumento result_type x que cria uma gerador propagado como se fosse chamando seed(x).
Um construtor explícito com o argumento seed_seq& seq que cria uma gerador propagado como se fosse chamando seed(seq).
void discard(unsigned long long count) chama de maneira eficaz operator() count vezes e descarta cada valor.
Adaptadores do mecanismo também oferecem suporte a esses membros (Engine é o primeiro parâmetro de modelo de um adaptador do mecanismo, designando o tipo do mecanismo básico):
Um construtor padrão para inicializar o gerado como se fosse a partir do construtor padrão do mecanismo básico.
Um construtor explícito com o argumento const Engine& eng. Isso serve para oferecer suporte à construção de cópia usando o mecanismo básico.
Um construtor explícito com o argumento Engine&& eng. Isso serve para oferecer suporte à construção de movimentação usando o mecanismo básico.
void seed() que inicializa o gerador com o valor de semente padrão do mecanismo básico.
const Engine& base() função da propriedade que retorna o mecanismo básico usado para construir o gerador.
Cada mecanismo mantém um estado que determina a sequência de valores que será gerada pelas chamadas subsequentes a operator(). Os estados de dois gerados instanciados a partir de mecanismos do mesmo tipo podem ser comparados usando operator== e operator!=. Se os dois estados forem comparados e forem iguais, eles gerarão a mesma sequência de valores. O estado de um objeto pode ser salvo em uma transmissão como uma sequência de valores sem sinal de 32 bits usando o operator<< do gerador. O estado não é alterado ao salvá-lo. Um estado salvo pode ser lido no gerador instanciado de um mecanismo do mesmo tipo usando operator>>.
[Vá para a parte superior da página]
Distribuições
Um distribuição é uma classe ou classe de modelo cujas instâncias transformam uma transmissão de números aleatórios distribuídos uniformemente obtidos de um mecanismo em um fluxo de números aleatórios que possuem uma distribuição específica. Cada distribuição possui os seguintes membros:
typedef numeric-type result_type é o tipo retornado pelo operator() de distribuição. O numeric-type é transferido como um parâmetro de modelo na instanciação.
template <class URNG> result_type operator()(URNG& gen) retorna valores que são distribuídos de acordo com a definição da distribuição, usando gen como uma fonte de valores aleatórios distribuídos uniformemente e os parâmetros armazenados da distribuição.
template <class URNG> result_type operator()(URNG& gen, param_type p) retorna valores distribuídos de acordo com a definição da distribuição, usando gen como uma fonte de valores aleatórios distribuídos uniformemente e a estrutura de parâmetros p.
typedef unspecified-type param_type é o pacote de parâmetros transmitidos opcionalmente para operator() e é usado no lugar dos parâmetros armazenados para gerar o valor retornado.
Um construtor const param& inicializa os parâmetros armazenados do seu argumento.
param_type param() const obtém os parâmetros armazenados.
void param(const param_type&) define os parâmetros armazenados do seu argumento.
result_type min() retorna o valor mínimo retornado pelo operator() da distribuição.
result_type max() retorna o valor máximo retornado pelo operator() da distribuição. Quando result_type é um tipo integral (com valor inteiro), max() é o valor máximo que pode ser retornado de fato (inclusivo); quando result_type é um tipo de ponto flutuante (com valor real), max() é o menor valor superior a todos os valores que podem ser retornados (não inclusivo).
void reset() descarta qualquer valor armazenado em cache, de forma que o resultado da próxima chamada para operator() não dependa dos valores obtidos do mecanismo antes da chamada.
Uma estrutura de parâmetro é um objeto que armazena todos os parâmetros necessários para uma distribuição. Ela contém:
typedef distribution-type distribution_type, que é o tipo da distribuição.
Um ou mais construtores que assumem as mesmas listas de parâmetros assumidos pelos construtores da distribuição.
As mesmas funções de acesso do parâmetro que a distribuição.
Operadores de comparação de equalidade e inequalidade.
Para obter mais informações, consulte os subtópicos de referência abaixo deste, vinculados anteriormente neste artigo.
[Vá para a parte superior da página]
Comentários
Há dois URNGs bastante úteis no Visual Studio — mt19937 e random_device — como mostrado na tabela de comparação:
URNG |
Rápido? |
Protegido por criptografia? |
Propagável? |
Determinista? |
---|---|---|---|---|
mt19937 |
Sim |
Não |
Sim |
Sim* |
random_device |
Não |
Sim |
Não |
Não |
* Quando fornecido com uma semente conhecida.
Embora o Padrão ISO C++ não exija que o random_device seja protegido por criptografia, no Visual Studio, ele é implementado com proteção criptográfica. (O termo “proteção criptográfica” não implica em garantias, mas se refere ao nível mínimo de entropia e, portanto, ao nível de previsibilidade, que um determinado algoritmo aleatório oferece. Para obter mais informações, consulte o artigo da Wikipedia Cryptographically secure pseudorandom number generator (em inglês).) Como o Padrão ISO C++ não requer isso, outras plataformas podem implementar random_device como um gerador de número pseudoaleatório simples (sem proteção criptográfica) e pode ser adequado somente como uma fonte de semeação para outro gerador. Verifique a documentação dessas plataformas ao usar random_device no código entre plataformas.
Por definição, resultados do random_device não são reproduzíveis e um efeito colateral é que ele pode ser executado de maneira significativamente mais lenta do que outros URNGs. A maioria dos aplicativos que não exigem proteção criptográfica usam o mt19937 ou um mecanismo semelhante, embora você possa querer propagá-lo com uma chamada para random_device, como mostrado no exemplo de código.
[Vá para a parte superior da página]