随机相关的函数

  头文件 stdlib.h

  相关函数 :rand 、srand

  rand( rand C++ Reference )

  函数声明:int rand( void );

  rand函数返回一个位于 0 - RAND_MAX之间的伪随机整数。其中RAND_MAX在头文件 stdlib.h 中定义( 一般为int类型可表示的最大正整数 )。

  rand函数通过一个特定的随机数生成算法生成伪随机数序列,该算法依据一个初始的种子值进行伪随机数生成。伪随机数生成可以看做一个特殊的处理过程,对于同一个输入(种子),输出的伪随机值序列总是相同的( 相对于一次程序运行 )。可使用函数 srand 设置伪随机数算法的种子,rand函数默认的种子值为1。

  srand( srand C++ Reference )

  函数声明: void srand( unsigned int seed);

  初始化伪随机序列生成器的种子值。srand函数将伪随机数生成算法的种子使用参数 seed 初始化,使得伪随机数生成依赖于不同的种子,避免重复。通过将种子值 seed 设置为不同的可区分的值(如程序运行时的时间),可以保证生成的伪随机序列较为随机。

  seed = 1时,即为伪随机序列生成算法的默认初始初始值。

  示例:

#include<stdio.h>
#include<stdlib.h> #define num_of_loop 8 /*循环次数*/
void generate_random_num(void); /*产生随机序列并输出*/ int main(void)
{
generate_random_num(); /*使用默认的随机种子,即 1*/ srand( ); /*使用 5 初始化随机种子*/
generate_random_num(); srand( ); /*使用 1 初始化随机种子*/
generate_random_num(); return ;
} void generate_random_num(void)
{
int i;
for( i = ; i < num_of_loop ; i++)
printf("%d\t",rand()); printf("\n");
}

rand示例

  结果如下图:

  

  可以看到默认种子的随机序列和种子值为1的随机序列是一样的,而种子值为5的随机序列则不相同。事实上,对于同一个种子值,每次程序运行产生的随机序列都是一样的( 注意不是每次 rand 函数调用,而是每次程序运行 )。即下一次运行上述示例时,仍产生一样的伪随机序列。

  为了使得由固定种子值产生的固定伪随机序列是较为随机的,可以在每次运行程序或调用 rand 函数之前使用不同的种子值来进行初始化。常用的方法是使用在函数调用之时的时间来作为种子值。

时间相关的函数

  头文件:time.h

  相关函数:time、clock

  time( time C++ Reference )

  函数声明:time_t time( time_t *timer );

  time函数的参数可以为空,即 0 / NULL ,此时 time 函数返回一个 time_t 类型的日历时间;

  time函数接受一个指向 time_t 类型变量的指针作为参数,函数返回一个 time_t 类型的日历时间,并将参数指针指向的变量值设置为该日历时间。

  time函数返回的日历时间精确到秒级。

  根据上文对随机函数的描述,我们可以使用下面的语句来对随机数序列的种子初始化,这样每次进行初始化时的种子在秒级别的精度上是不一样,故而可以做到较好的随机。

 srand( (int) time(NULL) )     //使用 time 函数返回的日历时间初始化函数,在秒级进度上保证每次种子值一致

  clock( clock C++ Reference )

  函数声明:clock_t clock( void );

  clock函数返回程序运行至clock函数处所消耗的处理器时间( 时间片数 ),若调用失败则返回 -1。

  clock函数返回值为运行消耗的时间片数,一个时间片为系统设置的一个常量时间单元,不同的系统时间片长度可能不同。可以通过在程序运行的不同位置设置两个clock函数,并根据其差值获得程序运行所消耗的时间片数。在头文件 time.h 中,存在宏定义 CLOCKS_PER_SEC,定义每秒所包含的时间片数。使用时可以通过 clock 函数得到消耗的时间片数,再通过除法获得具体消耗的秒数。

  

  其他函数:

  gmtime

  函数声明:struct tm* gmtime( const time_t  timer );

  将time_t 类型的值转换为UTC时间,并存放在一个 tm 类型的结构体中。

  

  localtime

  函数声明:struct tm* localtime( const time_t timer );

  将time_t类型的值转换为本地时间,并存放在一个 tm 类型的结构体中。

  

  actime

  函数声明:char * actime( const tm *timer);

  将 tm 结构体的数据转换为一个格式固定的字符串。

  

  结构体 tm ( 摘自头文件time.h )

struct tm
{
int tm_sec; /* Seconds: 0-59 (K&R says 0-61?) */
int tm_min; /* Minutes: 0-59 */
int tm_hour; /* Hours since midnight: 0-23 */
int tm_mday; /* Day of the month: 1-31 */
int tm_mon; /* Months *since* january: 0-11 */
int tm_year; /* Years since 1900 */
int tm_wday; /* Days since Sunday (0-6) */
int tm_yday; /* Days since Jan. 1: 0-365 */
int tm_isdst; /* +1 Daylight Savings Time, 0 No DST,
    /* -1 don't know */
};

  可以看到,tm结构体记录了年 月 日 时 分 秒 等信息,可通过 asctime 函数转换为方便显示的字符串形式。

  示例: 

display time

#include<time.h>
#include<stdio.h> int main(void)
{
char *local_time = NULL ;
char *UTC_time = NULL ;
struct tm *t_in_tm = NULL ; time_t t = time( NULL );
t_in_tm = localtime( &t ); //将time_t类型的数值参数转换为tm结构体中保存(本地时间)
local_time = asctime( t_in_tm ); //将tm类型结构体中的数值转换为字符串形式
printf("local time is %s\n",local_time); t_in_tm = gmtime( &t ); //将time_t类型的数值参数转换为tm结构体中保存(UTC时间)
UTC_time = asctime( t_in_tm ); //将tm类型结构体中的数值转换为字符串形式 printf("UTC time is %s\n",UTC_time); return ;
}

  可以看到本地时间与UTC时间相差八个小时

  

  注意:可以从示例代码中看到,在程序运行过程中,并没有声明一个具体的 tm 类型的结构体和存放字符串的数组用于存放数据,而只是声明了用于指向这些结构的指针用于存放返回值。这是由于上述函数的操作是针对一个公用的静态缓冲区来进行的,返回的参数为指向该缓冲区内具体结构的指针,故而不需要使用者另行声明。

  在头文件time.h中有如下警告:

  

  也就是说上述静态缓冲区是由函数共享的,可能被其他函数覆盖数据,使用需要小心。

  实际上,笔者在书写示例代码时,首先书写的是如下程序(只保留核心部分,读者可以比较一下与上面示例的微小不同),结果输出的两个时间总是相同的,而不是相隔8个小时。

     time_t t = time( NULL );
t_in_tm = localtime( &t ); //将time_t类型的数值参数转换为tm结构体中保存(本地时间)
local_time = asctime( t_in_tm ); //将tm类型结构体中的数值转换为字符串形式 t_in_tm = gmtime( &t ); //将time_t类型的数值参数转换为tm结构体中保存(UTC时间)
UTC_time = asctime( t_in_tm ); //将tm类型结构体中的数值转换为字符串形式 printf("local time is %s\nUTC time is %s\n",local_time,UTC_time);

  原因解释:在第一次函数调用返回( 2,3行),函数返回的是指向静态缓冲区中 tm 结构和字符串所在内存的地址。在第二次返回对应结构的内存地址指针时( 5,6行),由于公用静态缓冲区,后续 gmtime 和 asctime 函数使用的内存空间与 localtime 和 asctime使用的内存空间相同。即第二次操作是在同样的缓冲区中操作。结果就是,local_time和UTC_time指向的是同一片内存区域,且该区域最后一次是由第6行操作改变的,从而使的两个字符串输出的都是UTC时间形式。这也从侧面反应出共用静态缓冲区可能带来的弊端。

  更高精度的时间获取,可以参考博客 Windows获取当前系统时间

随机以及时间相关函数——C语言描述的更多相关文章

  1. 数据结构(java语言描述)

    概念性描述与<数据结构实例教程>大同小异,具体参考:http://www.cnblogs.com/bookwed/p/6763300.html. 概述 基本概念及术语 数据 信息的载体,是 ...

  2. Jquery如何序列化form表单数据为JSON对象 C# ADO.NET中设置Like模糊查询的参数 从客户端出现小于等于公式符号引发检测到有潜在危险的Request.Form 值 jquery调用iframe里面的方法 Js根据Ip地址自动判断是哪个城市 【我们一起写框架】MVVM的WPF框架(三)—数据控件 设计模式之简单工厂模式(C#语言描述)

    jquery提供的serialize方法能够实现. $("#searchForm").serialize();但是,观察输出的信息,发现serialize()方法做的是将表单中的数 ...

  3. 最小正子序列(序列之和最小,同时满足和值要最小)(数据结构与算法分析——C语言描述第二章习题2.12第二问)

    #include "stdio.h" #include "stdlib.h" #define random(x) (rand()%x) void creat_a ...

  4. c++学习书籍推荐《面向对象程序设计:C++语言描述(原书第2版)》下载

    百度云及其他网盘下载地址:点我 <面向对象程序设计:C++语言描述(原书第2版)>内容丰富,结构合理,写作风格严谨,深刻地论述了c++语言的面向对象编程的各种技术,主要内容包括:面向对象编 ...

  5. C语言学习书籍推荐《数据结构与算法分析:C语言描述(原书第2版)》下载

    维斯 (作者), 冯舜玺 (译者) <数据结构与算法分析:C语言描述(原书第2版)>内容简介:书中详细介绍了当前流行的论题和新的变化,讨论了算法设计技巧,并在研究算法的性能.效率以及对运行 ...

  6. 数据结构与抽象 Java语言描述 第4版 pdf (内含标签)

    数据结构与抽象 Java语言描述 第4版 目录 前言引言组织数据序言设计类P.1封装P.2说明方法P.2.1注释P.2.2前置条件和后置条件P.2.3断言P.3Java接口P.3.1写一个接口P.3. ...

  7. 数据结构(Java语言描述)-第一章:概述

    第一章 概述 1.0 序言 自己为啥要学数据结构嘞,我觉得主要有以下三个原因: 前段时间在看并发编程时,发现aqs,corrunthashmap等底层都用到了数据结构,主要的有队列,还有链表,学习数据 ...

  8. 《算法详解:C++11语言描述》已出版

    经过漫长的编写.修订和印刷过程,书籍<算法详解:C++11语言描述>终于出版了!目前本书已在各大电商平台上架,搜索书名即可找到对应商品.本书的特色在于: 介绍最新的C++11.C++14和 ...

  9. 我想立刻辞职,然后闭关学习编程语言,我给自己3个月时间学习C语言!这样行的通吗

    文章背景,回答提问:我想立刻辞职,然后闭关学习编程语言,我给自己3个月时间学习C语言!这样行的通吗? 我的建议是这样:1. 不要辞职.首先说,你对整个开发没有一个简单的了解,或一个系统的入门学习.换句 ...

随机推荐

  1. <数据挖掘导论>读书笔记6关联分析的高级概念

    处理联系属性: 基于离散化的方法 基于统计学的方法 非离散化方法 处理概念分层 定义在一个特定领域的各种实体或者概念的多层组织.概念分层可以用有向无环图DAG来标示. 序列模式 可选计数方案 COBJ ...

  2. org.springframework.web.util.NestedServletException: Request processing failed; nested exception is org.springframework.beans.BeanInstantiationException: Failed to instantiate [org.jboss.resteasy.plug

    之前做的项目是resteasy的上传,代码没有问题,断点都不进来呢. 我以为可以直接移植到SpringMVC,但是SpringMVC不支持MultipartFormDataInput , 用Multi ...

  3. 使用 Qt 获取 UDP 数据并显示成图片

    一个项目,要接收 UDP 数据包,解析并获取其中的数据,主要根据解析出来的行号和序号将数据拼接起来,然后将拼接起来的数据(最重要的数据是 R.G.B 三个通道的像素值)显示在窗口中.考虑到每秒钟要接收 ...

  4. [PY3]——内置数据结构(4)——字符串格式化(format)

    字符串格式化是拼接字符串的一种手段 join和+拼接字符串的方法,难以控制格式 printf style 字符串格式化 这种方法是从c语言继承过来的 # 待格式化的字符串:一个字符串存在占位符 In ...

  5. css兼容小问题

    1.RGBA在CSS3.0体现,不向下兼容: 2.非float元素和float元素在一起版本时,非float元素会排斥float元素,为避免换行,float元素应优先显示(放非float元素之前)

  6. 实现把dgv里的数据完整的复制到一张内存表

    /// <summary> /// 方法实现把dgv里的数据完整的复制到一张内存表 /// </summary> /// <param name="dgv&qu ...

  7. BI简介

    一.BI简介 BI全称是business intelligence,直译过来就是商业智能.BI表示的是一个体系,一套完整的解决方案.主要用于数据的整合.分析.挖掘等,为帮助企业决策而提供如报表.预测分 ...

  8. Spring学习笔记:jdbcTemplate和数据源配置

    一.使用Spring框架jdbcTemplate实现数据库的增删改查 1.数据库 /* SQLyog Ultimate v8.32 MySQL - 5.7.19-log : Database - in ...

  9. Idea 配置启动JDK___在windows中使用Intellij Idea时选择自定义的64位JVM

    略过下面的叙述快速配置 配置你的Idea启动jdk  (Idea15之后的版本启动不要JDK1.8) 添加环境变量IDEA_JDK_64,配置和JAVA_HOME环境变量相同,路径写到1.8的jdk就 ...

  10. unity3d之切换场景不销毁物体

    using System.Collections; using System.Collections.Generic; using UnityEngine; /// <summary> / ...