0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

PLC中随机数产生的几种方式

CHANBAEK 来源:头条号朝拾忆 作者:头条号朝拾忆 2023-03-23 13:50 次阅读

有时为了某些测试需求,需要仿真产生一些数据。 这时,我们可以通过调取指令或自行编写程序来生成这些随机数据。

以下以博途为例,简要说明了随机数产生的几种方式:

一、读取系统时间的纳秒作为随机数

以固定周期直接将系统时间中的纳秒输出到对应变量。

系统时间中的纳秒为UDINT类型,转为INT后,丢弃了高字部分

监视实时的系统时间,取其中的纳秒(NANOSECOND,1秒=10^9纳秒)

监视一下生成的随机数的范围:-32258—32751(5分钟)

后续可继续对此数据处理,缩放到需要的区间。

二、由LGF库(官方提供的通用函数库)内的随机数程序生成

该指令原理也是采用纳秒,不过处理过程更加细化、完善。

LGF库

随机数生成程序如下(只贴了其中关键的计算过程):

REGION Calculating random number
    // 将纳秒转换为双字以便寻址单个字节
    #tempNanoSecondInDWord := UDINT_TO_DWORD(#tempTime.NANOSECOND);
    // 以片段访问方式将纳秒进行字节交换
    #tempRandomValue.%B3 := #tempNanoSecondInDWord.%B0;
    #tempRandomValue.%B2 := #tempNanoSecondInDWord.%B1;
    #tempRandomValue.%B1 := #tempNanoSecondInDWord.%B2;
    #tempRandomValue.%B0 := #tempNanoSecondInDWord.%B3;
    // 随机数标准化
    #tempNormReal := UDINT_TO_REAL(DWORD_TO_UDINT(#tempRandomValue)) / UDINT_TO_REAL(#MAX_UDINT);
    // 随机数缩放
    #LGF_RandomRange_Real := ((#tempNormReal * (#maxValue - #minValue) + (#minValue)));
    #error := false;
    #status := #STATUS_FINISHED_NO_ERROR;
    #subfunctionStatus := #SUB_STATUS_NO_ERROR;
    // ENO mechanism is not used
    ENO := TRUE;
END_REGION

在主程序中调用,可设置上下限

三、线性同余法(LCG,Linear Congruential Method)

该方法的核心是以下递归公式:

RandNum =(A * RandNum + B)% M

A、B、M均为常数,其中A是 乘数 ,B是 增量 ,M是 模数 ,RandNum是 初始值 ,A、C、M的取值是保证产生高质量随机数的关键。

可以看出,每次新产生的随机数都跟上一次的数有关系。 随机数序列中的初始值,我们通常叫做种子。 随机数的产生需要设置种子,否则随机数的结果每次运行都将一样。 通常,我们使用系统时间的纳秒作为种子(某些将此作为缺省设置),这在一定程度上保证了种子的唯一性。

由于计算过程最后是对M取余数,余数的范围就是0—(M-1),这决定了产生的随机数是有周期性的。 M的大小决定了最大周期的长短,一般取值域的最大值,而A和B也会影响周期。 A、B、M的选取多种多样,只要保证产生的随机数有较好的均匀性和随机性即可。

FC块,变量定义为双整型。 模数M可以取值域最大值2^32

种子seed可以采用系统时间或自行设置

随机数曲线

线性同余法的初始值一旦确定,输出的序列将固定。 而当获取某些随机数序列后,其初始值以及A、B、M也会被反向计算出来。

对于其缺点,可以考虑以下改进方式,每产生n个数,将当前时钟值MOD M得到的余数作为新的种子。

四、平方取中法

平方取中法由冯·诺依曼提出,它的原理是:首先取一个2s位的整数(种子),平方,得4s位整数,然后取此4s位中间的2s位作为下次运算的种子。 重复该过程,即可得到一个随机数序列。 (序列中每个数缩放至0.0—1.0范围内)

例如:取种子365,平方得133225,高位补0,取中间1332,平方得1774224,高位补0,取7742,以此类推.........

#RandInt := SQR(#Seed);
#Seed := (#RandInt MOD 1000000 - #RandInt MOD 100) / 100;
#RandReal := DINT_TO_REAL(#Seed) / 9999.0;

随机数测试结果

在实践中,这种方法其实并不好用。 很难说明取什么种子才能保证足够长的周期。 以种子123为例,在40多个周期后,种子末位便退化产生了00,之后的随机数成了固定的几个数值,周期极短。 该算法也有改进空间。

梅森旋转算法_Mersenne Twister

梅森旋转算法可以产生高质量的伪随机数,且效率高效,弥补了以上伪随机数生成器的不足和缺陷。 它在C++Python编程语言中均有应用。

理解该算法前需要先了解许多前置名词,线性反馈移位寄存器、级、反馈函数、抽头序列、本原多项式...... 实在有兴趣的可以搜索一下。 我,放弃了。

说到随机数,不禁想到了因果律:果由因生、有依空立 、事待理成。

所谓的“随机”,大概不过是事物发展中的个体因为信息偏差,产生的局限认知。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • plc
    plc
    +关注

    关注

    5016

    文章

    13385

    浏览量

    464965
  • 数据
    +关注

    关注

    8

    文章

    7134

    浏览量

    89456
  • 仿真
    +关注

    关注

    50

    文章

    4124

    浏览量

    133899
  • 程序
    +关注

    关注

    117

    文章

    3795

    浏览量

    81329
  • 随机数
    +关注

    关注

    0

    文章

    18

    浏览量

    12040
收藏 人收藏

    评论

    相关推荐

    matlab中产生随机数的十七种方式

    ) 的该分布的随机数。例如: (1) R = random('Normal',0,1,2,4): 生成期望为 0,标准差为 1 的(2 行 4 列)2× 4 个正态随机数 (2) R
    发表于 01-16 11:05

    keil 产生 随机数

    最近需要用 keil 软件产生一个0-10的随机数,请各位大神指点下,有原代码参考的话就更好了
    发表于 03-12 20:53

    怎么产生更大范围的随机数

    labview提供了产生0-1的随机数函数,怎么产生更大的范围的随机数
    发表于 10-26 21:17

    STM32的ADC产生随机数

    本帖最后由 sunhongdd26 于 2015-7-14 09:58 编辑 因为在做一样东西需要产生随机数,范围在[0,15],在网上找到一种方法,就是利用ADC悬空引脚产生随机数
    发表于 02-28 11:58

    随机数产生小程序求助

    刚开始学习Labview,想做一个小程序:打开布尔开关,自动产生0-10范围内的随机数,输出数值控件一直刷新随机数,开关关闭后,随机数停止刷新,输出控件显示最后一个
    发表于 04-18 23:50

    单片机C语言如何产生随机数

    单片机C语言如何产生随机数 随机数在单片机的应用也是很多的,当然产生随机数的方法有很多,当中有
    发表于 05-14 15:14

    产生随机数的方法有哪些

    随机数在单片机的应用也是很多的,当然产生随机数的方法有很多,当中有一个就是利用单片机定时器,取出未知的定时器THX和TLX的值,再加以运算得到一个规定范围内的
    发表于 07-15 09:08

    AT32的随机数产生

    AT32的随机数产生为设计者使用AT32芯片时,产生符合应用需求的随机数,提供设计建议。
    发表于 10-26 06:04

    产生随机数

    一个自己写的产生随机数的工程
    发表于 12-01 15:45 13次下载

    单片机产生随机数的方法

    随机数在单片机的应用也是很多的,当然产生随机数的方法有很多,当中有一个就是利用单片机定时器,取出未知的定时器THX和TLX的值,再加以运算得到一个规定范围内的
    发表于 02-23 10:37 2.2w次阅读

    单片机产生随机数的两种方法

    随机数在单片机的应用也是很多的,当然产生随机数的方法有很多,当中有一个就是利用单片机定时器,取出未知的定时器THX和TLX的值,再加以运算得到一个规定范围内的
    发表于 03-01 11:04 2330次阅读

    单片机C语言如何产生随机数

    C语言中有三个通用的随机数发生器,分别为 rand函数、random函数、randomize 函数,但是rand函数产生的并不是真意正义上的随机数,是一个伪随机数,是根据一个
    的头像 发表于 11-26 14:25 7598次阅读

    STM8单片机产生随机数

    最近需要用到单片机随机数,但是用rand()产生随机数,发现每次单片机上电时产生随机数都是一样的,没有实现真正的
    发表于 11-23 17:21 13次下载
    STM8单片机<b class='flag-5'>产生</b><b class='flag-5'>随机数</b>

    单片机C语言如何产生随机数

    随机数在单片机的应用也是很多的,当然产生随机数的方法有很多,当中有一个就是利用单片机定时器,取出未知的定时器THX和TLX的值,再加以运算得到一个规定范围内的
    发表于 02-08 17:12 11次下载
    单片机C语言如何<b class='flag-5'>产生</b><b class='flag-5'>随机数</b>

    PLC输出0~100之间的随机数编写

    由于西门子PLC不提供随机数相关函数,在需要用到随机数的情况下,只能自己手动去写,下面来教大家写一个简单的0~100之间的随机数
    发表于 10-11 12:22 4097次阅读
    <b class='flag-5'>PLC</b>输出0~100之间的<b class='flag-5'>随机数</b>编写