请教有关随机数产生 ( 积分: 100 )

  • 主题发起人 主题发起人 uiit
  • 开始时间 开始时间
U

uiit

Unregistered / Unconfirmed
GUEST, unregistred user!
产生一个随机数
有上限下限
而且又有个中心值
随机数产生最好能围绕这个中心值近似值服从正态分布
 
产生一个随机数
有上限下限
而且又有个中心值
随机数产生最好能围绕这个中心值近似值服从正态分布
 
Randomize;
随即数:=下限+Random(上限-下限);
 
关键是近似服从正态分布
 
只有自己写了,delphi 的random好象没有这个功能。
 
Randomize是中就用了RandSeed,只不过种子是有系统时间生成的。
Random是产生平均分布的随机数。
RandG是产生高斯分布(即正态分布)的随机数。
 
>Avalon
请给各简单例子好不?
在help里我看不是很明白
3ku
 
RandG:产生一个随机数,随机数符合正态分布于Mean数附近,
参数StdDev可是标准差的意思(方差开平方)
用于模拟数据抽取样品误差
呵呵 只能帮助你这些了 说实话 这个我也没怎么用过
 
RandG function
Generates random numbers with Gaussian distribution.
Unit
Math
Category
random number routines
Delphi syntax:
function RandG(Mean, StdDev: Extended): Extended;
C++ syntax:
extern PACKAGE Extended __fastcall RandG(Extended Mean, Extended StdDev);
Description
RandG produces random numbers with Gaussian distribution about the Mean. This is useful for simulating data with sampling errors and expected deviations from the Mean.
又学了一招
 
3ku!
楼上把help给我贴出来
晕~
我想大家有兴趣的人继续讨论一下
怎么用
暂时不给大家加分哈
 
to:iamyournee
自己写
你又是如何想的呢
 
关注,拟实现一个正态分布的概率图,试试
 
var
rv,rv1,rv2,rv3,rv4,i:integer;
begin
rv1:=0;
rv2:=0;
rv3:=0;
rv4:=0;
i:=0;
repeat
rv:=random(100);
rv:=random(rv);
rv:=random(rv);
if (rv>=0) and (rv<25) then
rv1:=rv1+1;
if (rv>=25) and (rv<50) then
rv2:=rv2+1;
if (rv>=50) and (rv<75) then
rv3:=rv3+1;
if (rv>=75) and (rv<=100) then
rv4:=rv4+1;
i:=i+1;
until i=strtoint(edit5.Text);
edit1.Text:=inttostr(rv1);
edit2.Text:=inttostr(rv2);
edit3.Text:=inttostr(rv3);
edit4.Text:=inttostr(rv4);
end;
我写了一个函数,但只统计了30000个记录
 
我发现越random次数越多曲线越陡峭,不知道你是要什么样的正态曲线
 
{*
标准正态分布测试程序(RandG)
注意:
映射与转换时不可避免的有误差,分辨率未精确计算,仅仅在往实际高度转换时扩大了100倍,
更改:我把这个数试着改为200倍,发现曲线更像那么一回事了:)

2005-4-26
}
unit Unit1;
interface
uses
Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
Dialogs, ExtCtrls, StdCtrls, Math;
type
TfrmMain = class(TForm)
imgAxis: TImage;
btnRandG: TButton;
GroupBox1: TGroupBox;
Splitter1: TSplitter;
grpControl: TGroupBox;
procedure FormShow(Sender: TObject);
procedure btnRandGClick(Sender: TObject);
procedure FormCreate(Sender: TObject);
private
{ Private declarations }
totalCount :Integer;
sampleDatas :array[1..1000] of Integer;
procedure DrawAxis;
procedure GaussIt(const ASampleData:Extended);
procedure ReDraw;
procedure DrawEdge;//画边
procedure Log(const logName:string);
public
{ Public declarations }
end;

var
frmMain: TfrmMain;
implementation
{$R *.dfm}
procedure TfrmMain.DrawAxis;
begin
imgAxis.Canvas.Pen.Color :=clWhite;
imgAxis.Canvas.Pen.Mode :=pmXor;
imgAxis.Canvas.MoveTo(imgAxis.Width div 2,imgAxis.Height);
imgAxis.Canvas.LineTo(imgAxis.Width div 2,0);
imgAxis.Canvas.MoveTo(0,imgAxis.Height -10);
imgAxis.Canvas.LineTo(imgAxis.Width,imgAxis.Height -10);
end;
procedure TfrmMain.ReDraw;
var
bmp :TBitmap;
i :Integer;
begin
//根据数组里面的数和totalCount重新画图
bmp :=TBitmap.Create;
try
bmp.Width :=imgAxis.Width;
bmp.Height:=imgAxis.Height;
bmp.Canvas.Brush.Color :=clBlack;
bmp.Canvas.FillRect(bmp.Canvas.ClipRect);
bmp.Canvas.Pen.Color :=clRed;
bmp.Canvas.Pen.Width :=1;
//根据Image高度和1->1000发生的频数设置画线的高度
//应该计算出图像所能反映的最小分辨率,应能体现到频数有1的变化
for i:=1 to 1000do
begin
sampleDatas :=(sampleDatas*(bmp.Height-10)) *200 div totalCount;//频数->频率->实际图像高度值
sampleDatas :=bmp.Height-10 -sampleDatas;//高度转换为实际坐标
bmp.Canvas.MoveTo(i,bmp.Height-10);
bmp.Canvas.LineTo(i,sampleDatas);
end;
//Log('height.txt');
imgAxis.Canvas.Draw(0,0,bmp);
finally
bmp.Destroy;
end;
end;
procedure TfrmMain.GaussIt(const ASampleData:Extended);
var
x :Integer;
begin
Inc(totalCount);
//ShowMessage(FloatToStr(ASampleData));
//将这个随机数映射到1..1000,正数:500->1000,负数500->1
//ASampleData>=2 ---1000
// =0 ---500
// <=-2 ---1
x :=Round(ASampleData *1000);
//确定范围
if x>=4000 then
x:=4000
else
if x<=-4000 then
x:=-4000;

x := (x +4001) shr 3;
Inc(sampleDatas[x]);//记录发生的频数,除以总次数就是频率了
//ShowMessage(IntToStr(x));
//ShowMessage(IntToStr(sampleDatas[x]));
end;
procedure TfrmMain.FormCreate(Sender: TObject);
begin
do
ubleBuffered :=true;
DrawAxis;
end;

procedure TfrmMain.btnRandGClick(Sender: TObject);
var
i:Integer;
begin
totalCount :=0;
for i:=1 to 1000do
sampleDatas:=0;
for i:=1 to 100000do
//总计100000次
GaussIt(RandG(0,1));//标准正态分布
//Log('afterBtn.txt');
ReDraw;
DrawEdge;
DrawAxis;
end;

procedure TfrmMain.FormShow(Sender: TObject);
begin
Left :=0;
end;

procedure TfrmMain.Log(const logName:string);
var
i:Integer;
log:TStringList;
str:string;
begin
log :=TStringList.Create;
try
for i:=1 to 1000do
begin
str :=Format('%.4d : %d(频数)',[i,sampleDatas]);
log.Add(str);
end;
str :=Format('总数:%d',[totalCount]);
log.Add(str);
totalCount :=0;
for i:=1 to 1000do
inc(totalCount,sampleDatas);
str :=Format('实际总数:%d',[totalCount]);
log.Add(str);
log.SaveToFile(logName);
finally
log.Destroy;
end;
end;

procedure TfrmMain.DrawEdge;//画边
var
i:Integer;
points :array[1..1000] of TPoint;
begin
for i:=1 to 1000do
begin
points.x :=i;
points.y :=sampleDatas;
end;
imgAxis.Canvas.Polyline(points);
end;

end.
 
上面是第一次写的,测试出来的曲线效果不怎么理想
下面更改了一下数据,看起来像个标准正态分布了
将数据映射到1..1000
下面是关键部分:
//将这个随机数映射到1..1000,正数:500->1000,负数500->1
//ASampleData>=8 ---1000
// =0 ---500
// <=-8 ---1
//确定范围
if x>=4000 then
x:=4000
else
if x<=-4000 then
x:=-4000;

x := (x +4001) shr 3;
发生次数改为100000时,更加逼真
 
to iamyourneed:
我不需要曲线,我要的是这些数据
等于是一个模拟电压源的程序
电压范围200—250,但相对来说在220附近比较平滑
但在以220位中心这些数据服从正态分布
采样频率为250个点/15sec
如果正态分布曲线过抖,那不是正好吗
to dawnsong:
我试试你写的
谢谢大家
继续关注:
 
--uiit
留个信箱,我把代码程序给你发过去
 
好的,3ks
sxd269@163.com
 

Similar threads

S
回复
0
查看
1K
SUNSTONE的Delphi笔记
S
S
回复
0
查看
900
SUNSTONE的Delphi笔记
S
S
回复
0
查看
3K
SUNSTONE的Delphi笔记
S
后退
顶部