[每日一个小算法]报童问题

某报童以每份0.03元的价格买进报纸,以0.05元的价格售出,根据长期统计,报纸每天的销售量及百分率为

销售量

200     210     220     230     240     250   

百分比

      0.10    0.20     0.40    0.15     0.10    0.05

已知当天销售不出去的报纸将以每份0.02元的价格退还给报社,试用模拟方法确定报童每天买进报纸数量时报童的平均总收入为最大?

一、系统的假设:

 1 模拟时间充分大; 

2 报童购买报纸量介于销售量最小值与最大值之间; 

3)不考虑有重大事件发生时卖报的高峰期,也不考虑风雨天气时卖报的低谷期。

二、问题分析 

报童购进数量应根据需求量确定,但需求量是随机的,所以报童每天如果购进的报纸太少,不够买的,会少赚钱;如果购进太多,卖不完就要赔钱,这样由于每天报纸的需求量是随机的,致使报童每天的收入也是随机的,因此衡量报童的收入,不能是报童每天的收入,而应该是他长期(几个月、一年)卖报的日平均收入。我们可以应用计算机模拟的方法 在模拟时间充分大的条件下(例如10000天),模拟每天的销售量,因而确定每天应买进多少报纸才能使平均总收入达到最大值。 

设该问题的销售量为离散型随机变量hh有分布列P(h =xi)=pi(i=1,2,…6)。所谓对离散型随机变量h的模拟, 其实质就是通过计算机上的模拟试验来取得在真实试验下的样本值 U,由分布列知U只取数值x1,x2…中的某一个.对离散型随机变量hh的一般模拟方法如下:  

(1) 在直线上将[0 ,1]区间划分成若干个子区间, 其分点记为A1,…A2,A6,分点A i的坐标

 i=1,2…6      

2)在计算机上取伪随机数rand

注释:计算机中的随机函数是按照一定算法模拟产生的,其结果是确定的,是可见的。我们可以这样认为这个可预见的结果其出现的概率是100%。所以用计算机随机函数所产生的“随机数”并不随机,是伪随机数。

3)判定randx轴上所落的位置,若rand∈,则可以认为h的一次真实试验样本值取xi。

三、符号假设

BUYMIN:每天的最小购买量              

BUYMAN:每天的最大购买量 

SIMUDAY:模拟时间 

sell_amount:报童销售量

buy_amount:报童购买量 

percentage:销售百分率

ave_profit:总平均利润 

loop_buy :当天购买量

loop_day :当天时间

计算机程序:在Matlab软件包中编程,共需两个M-文件:main.m, Getprofit.m, 主程序为main.m.

 %  主文件main.m 

BUYMIN=200;  每天的最小购买量 

BUYMAX=250;  每天的最大购买量  

SIMUDAY=1.0e+5;    模拟时间 %

sell_amount=200:10:250; 销售量 

percentage=[0.1 0.3 0.7 0.85 0.95 1 %] 百分率

buy_amount=0;

ave_profit=0;

for loop_buy=BUYMIN:BUYMAX

   sum_profit=0; 

   for loop_day=1:SIMUDAY 

index=find(percentage>=rand);  %产生随机数,用于决定当天的销售量

         sum_profit=sum_profit+GetProfit(loop_buy,sell_amount(index(1)));  

end 

buy_amount=[buy_amount,loop_buy];       %  循环嵌套 

ave_profit=[ave_profit,sum_profit/SIMUDAY]; %  循环嵌套

end

buy_amount(1)=[];              %  第一个元素置空

ave_profit(1)=[]; 

[val,id]=max(ave_profit)         % 显示最大平均收入val

buy=buy_amount(id)     % 显示在平均收入最大情况下的每天的购买量buy

xlabel='每天的购买量'; 

ylabel='平均利润';

plot(buy_amount,ave_profit,'*:');

%  函数GetProfit.m代码:

function re=GetProfit(a,b) 

if a<b                   %  供不应求:报童购买量小于销售量

        re=a*(0.05-0.03) ;                   

else                 %  供过于求:报童购买量大于销售量

          re=b*(0.05-0.03)+(a-b)*(0.02-0.03);

end 



所谓报童问题一个需求有关,而需求是随机的问题。一位报童刊发行处订后零售,每卖出一份可获利a元,若订后卖不出去,则退回发行处,每份将要赔钱b元。那么报童如何根据以往的卖情况(每报纸需求量为k份的概率为 )来推算出每收益达到最大的订量n? 算法解说分析: ① 我利用负指数分布公式“g(u)=-lg(u)”,其中“u=1.0*u/RAND_MAX(产生[0,1]均匀分布的随机数)”。函数中“g(u)=-lg(u)”的自变量“u”是均匀产生[0,1]之间的数,可知“g(u)”的函数值也是等可能的产生的,而且这些函数值是呈一种负指数分布趋向的。我可以通过一些巧妙地方法,就是让这些函数值乘上一个数值,让其不能超过1并且把这些值累加起来(今次加上上一次的),这是一个循环,结束的条件是累加的这些和的值≥1。到循环结束的时候,我可以算一下究竟循环了多少次,而这个循环了“多少次”就是我们所需要的需求量的模拟值。 ② 因为我们都知道“-lg(u)”的值是公平地呈负指数分布出现的,为何?很明显,“u=1.0*u/RAND_MAX(产生[0,1]均匀分布的随机数)”证明其“公平性”。“公平性”很重要,因为能出现通过“-lg(u)”计算得出的值必须要公平才有“可信性”。同理,“-lg(u)”乘上一个具体的数以后也是能“g(u)”的值是公平地呈负指数分布出现的“x=-1.0/t*log(u);/*产生负指数分布的随机数(t是确定的常数)*/ ” ③ 根据负指数函数的分布规律可知,每次让这些函数值缩小某个级别的时候在累加起来,直到其值“=1”才停止,其中循环的次
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

MC数据局

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值