【问题标题】:Discrete Event simulation generates different results in windows and linux离散事件模拟在windows和linux中产生不同的结果
【发布时间】:2014-05-27 04:45:34
【问题描述】:

我正在进行 m/m/2/2 离散事件模拟。结果与理论值相同,说明程序是正确的(在Ubuntu,g++中)。但是,当我在windows(visual studio)中再次测试代码时,结果只有在试验次数小于10000时才正确。我调试了很长时间,终于发现我必须使用(double) rand() / ( RAND_MAX + 1 )而不仅仅是@ 987654322@。

这是为什么呢?这两个编译器生成随机的方式不同吗?

#include <iostream>
#include <stdlib.h>
#include <fstream>
#include <cmath>
#include "iomanip"
#include <queue>
using namespace std;

#define TRIAL_NUM 10000000

//structure that represent a customer
struct cust
{
int cust_id;
double arrival_time;
double depart_time;
};

//uniform random generator
double uniRand()
{
    return (double) rand() / RAND_MAX;
}

//exponential random generator
double expRand(double lam)
{
double u,x;
u=uniRand();
x=(-1/lam)*log(1-u);
return(x);
}

int main()
{
    //seed the random generator
srand (time(NULL));                   
//queue that hold all customers
std::queue<cust*> customers;
double lam = 0, mu = 0;
bool showStatus = false;
cout<<"M/M/1/1 Discrete Event Simulation with "<<TRIAL_NUM<<" customers\n"<<" Please spacify the value of lambda:";
cin>>lam;
while(lam<= 0)
{
    cout<<"lambda value must be a positive number, try again:";
    cin>>lam;
}
cout<<"Please specify the value of mu:";
cin>>mu;
while(mu<= 0)
{
    cout<<"lambda value must be a positive number, try again:";
    cin>>mu;
}
cout<<"Show status for each customer? (y/n)";
char show;
cin>>show;
while( tolower(show) != 'y' && tolower(show) != 'n')
{
    cout<<"Invalid input, try again:";
    cin>>show;
}
if ( tolower(show) =='y' )
{
    showStatus = true;
}

//Generating all arrival time, service time for all customers
double temp_time = 1;
cust* temp;
for (int i = 1; i <= TRIAL_NUM; ++i)
{
    temp= new cust;
    temp -> arrival_time = temp_time + expRand(lam);
    temp_time = temp->arrival_time;
    temp->depart_time = temp->arrival_time + expRand(mu);
    temp->cust_id = i;
    customers.push(temp);
}

//Blocking customer count
double block = 0;
bool blockFlag = false;
temp = customers.front();
double server1 = 0, server2 = 0; 

//perform simulation
while( !customers.empty())
{
    if ( server1 < temp->arrival_time)
    {
        server1 = temp->depart_time;
    }
    else if ( server2 < temp->arrival_time)
    {
        server2 = temp->depart_time;
    }
    else
    {
        block++;
        blockFlag = true;
    }
    if (showStatus)
    {
        cout<<"Customer "<<temp->cust_id<<"\tarrived at:"<<temp->arrival_time<<"\tservice time:"<<temp->depart_time<<"\tstatus:";
        if (blockFlag)
        {   
            cout<<"Blocked"<<endl;
        }
        else{
            cout<<"Served"<<endl;
        }
    }
        customers.pop();
        if (!customers.empty())
        {
            temp = customers.front();
        }
        blockFlag = false;
}
cout<<"Blocking probability is: "<<block/TRIAL_NUM<<endl;
return 0;
}

【问题讨论】:

  • 不要使用 rand()。它会在不同的编译器上给出不同的结果,结果会很糟糕。例如,对于它在 windows 上生成的前 20 个值打印 rand() % 2 并在输出时感到困惑……C++ 有许多选择,std::mt19937 可能适合你。

标签: c++ linux random simulation


【解决方案1】:

(double) rand() / (RAND_MAX + 1) 是一个危险的表达方式。

如果RAND_MAX 等于INT_MAX,就像经常发生的那样,那么RAND_MAX + 1 是一种溢出和未定义的行为(它通常会导致INT_MIN,但任何规范都不能保证这一点)。

在 Visual C++ 中,RAND_MAX 是 32767,你没有这个问题(但 32767 是一个很小的值,它“开始循环”很快)。

无论如何我认为真正的问题在这里:

u = uniRand();
x = (-1 / lam) * log(1 - u);

uniRand() 定义为(double) rand() / RAND_MAX 时,它会产生范围内的值:

0.0 <= uniRand() <= 1.0

当值等于 1.0 时,您的 log(1-u) 中有一个 pole error。由于 RAND_MAX 的值较小,因此在 Visual C++ 中会更频繁地发生这种情况。

【讨论】:

  • 谢谢,这个解释很有帮助!那么我应该怎么做才能使代码在两个平台上都可以工作并避免“危险的表达”呢?
  • 我会使用(double) rand() / RAND_MAX 表达式,检查x = (-1 / lam) * log(1 - u); 以避免域错误(lam1-u 接近0.0)。还要检查内存分配(您在temp = new cust; 中分配内存但没有释放它)。
  • 另一种可能的解决方案是获取Mersenne Twister 的副本并使用生成器的 [0,1) 变体。
猜你喜欢
  • 2021-07-16
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2012-10-22
  • 2011-04-29
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多