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
    +关注

    关注

    4975

    文章

    12265

    浏览量

    454996
  • 数据
    +关注

    关注

    8

    文章

    6515

    浏览量

    87615
  • 仿真
    +关注

    关注

    50

    文章

    3873

    浏览量

    132167
  • 程序
    +关注

    关注

    114

    文章

    3631

    浏览量

    79560
  • 随机数
    +关注

    关注

    0

    文章

    17

    浏览量

    11941
收藏 人收藏

    评论

    相关推荐

    【assingle分享】labview随机数发生器

    。下面创建一个正态分布随机数序列,并计算它的均值和标准方差。信号生成选板还提供了其它几种噪声函数,这里就不再详细讨论了,下一篇文章中介绍波形函数选板
    发表于 04-07 10:27

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

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

    随机数产生

    各位兄弟:小弟最近真在做一个项目,需要用ATmega329产生-12.5~12.5的随机数,有人做过吗?方法越简单越好。敬候指点
    发表于 10-24 08:49

    keil 产生 随机数

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

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

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

    【总结】LabVIEW随机数发生器

    还提供了其它几种噪声函数,这里就不再详细讨论了,下一篇文章中介绍波形函数选板几种随机数发生器,以及概率函数选板
    发表于 01-30 14:51

    STM32的ADC产生随机数

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

    【LinkIt 7687试用体验】8.加密算法随机数产生方法

    的安全性。在使用的过程,我们只需要初始化硬件随机数,然后使用hal_trng_get_generated_random_number函数进行获取值就可以了,最后再关闭随机数产生器就可
    发表于 12-17 11:11

    随机数产生小程序求助

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

    关于STM32真随机数算法的问题

    rand是伪随机数,真随机数配合定时器time()计算产生随机数,但是如果我想上电初始化的时候要产生一个
    发表于 11-13 10:12

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

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

    产生随机数的方法有哪些

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

    什么是随机数

    做开发的工程师们应该或多或少都接触过随机数,可能认为它就是一个随机生成的数字嘛,使用时也很简单,只要调用开发语言提供的函数即可。但实际上随机数后面还是有着比较复杂但也有趣的知识点的。根据一般定义
    发表于 07-22 09:42

    AT32的随机数产生

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

    单片机是如何产生随机数的?

    单片机如何产生随机数
    发表于 10-27 06:44