百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分类 > 正文

C语言rand函数详解:随机数的「魔法骰子」

ztj100 2025-03-04 16:00 9 浏览 0 评论

核心定位

rand 是C语言中生成伪随机数的「魔法骰子」,它能返回一个范围在 0 到 RAND_MAX 之间的整数。就像掷骰子一样,每次调用都会得到一个看似随机的数字,但背后的规律由「随机种子」决定!


函数原型与参数

int rand(void);
  • 入口参数:无(void 表示无需任何参数)
  • 返回参数
    • 返回一个伪随机整数,范围在 0 到 RAND_MAX 之间
    • RAND_MAX 是标准库定义的常量,通常为 32767(2^15 - 1)

实战代码演示

场景1 生成随机数

#include 
#include 
#include 

int main() {
    //  初始化随机种子(基于当前时间)
    srand(time(NULL));

    // 生成随机数
    int random_num = rand();
    printf("随机数:%d\n", random_num);  // 输出:0 到 RAND_MAX 之间的随机数
    return 0;
}

场景2 生成指定范围的随机数

// 生成 [min, max] 范围内的随机数
int rand_range(int min, int max) {
    return min + rand() % (max - min + 1);
}

int main() {
    srand(time(NULL));

    // 生成 1 到 100 之间的随机数
    int dice = rand_range(1, 100);
    printf("掷骰子结果:%d\n", dice);  // 输出:1 到 100 之间的随机数
    return 0;
}

场景3 生成随机浮点数

// 生成 [0.0, 1.0) 范围内的随机浮点数
double rand_double() {
    return (double)rand() / (RAND_MAX + 1.0);
}

int main() {
    srand(time(NULL));

    double random_float = rand_double();
    printf("随机浮点数:%f\n", random_float);  // 输出:0.0 到 1.0 之间的随机浮点数
    return 0;
}

四大致命陷阱

陷阱

后果

防御方案

未初始化种子

每次运行结果相同

调用 srand(time(NULL)) 初始化

范围计算错误

随机数分布不均

使用 rand_range 函数封装

浮点数精度问题

结果不精确

使用 double 类型计算

RAND_MAX 过小

随机性受限

使用更高质量的随机数库(如C++11)


增强版随机数生成

生成高质量随机数(C++11风格)

#include 
#include 
#include 

// 使用更高质量的随机数生成器
int rand_high_quality(int min, int max) {
    static int initialized = 0;
    if (!initialized) {
        srand(time(NULL));
        initialized = 1;
    }
    return min + rand() / (RAND_MAX / (max - min + 1) + 1;
}

int main() {
    int random_num = rand_high_quality(1, 100);
    printf("高质量随机数:%d\n", random_num);  // 输出:1 到 100 之间的随机数
    return 0;
}

对比rand与C++11随机数库

特性

rand

C++11随机数库

随机性质量

较低

范围控制

需手动计算

内置支持

种子管理

需手动初始化

自动管理

性能

略低


黄金法则

  1. 初始化种子:每次程序启动时调用 srand(time(NULL))
  2. 封装范围函数:避免手动计算范围导致错误
  3. 浮点数技巧:用 (double)rand() / (RAND_MAX + 1.0) 生成 [0.0, 1.0) 的浮点数
  4. 高质量随机数:在需要高随机性时,使用C++11的

脑洞应用:随机密码生成器

#include 
#include 
#include 

// 生成随机密码
void generate_password(char *password, int length) {
    const char charset[] = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";
    srand(time(NULL));

    for (int i = 0; i < length; i++) {
        int index = rand() % (sizeof(charset) - 1);
        password[i] = charset[index];
    }
    password[length] = '\0';  // 添加字符串结束符
}

int main() {
    char password[9];  // 8位密码 + 1位结束符
    generate_password(password, 8);
    printf("随机密码:%s\n", password);  // 

相关推荐

告别手动操作:一键多工作表合并的实用方法

通常情况下,我们需要将同一工作簿内不同工作表中的数据进行合并处理。如何快速有效地完成这些数据的整合呢?这主要取决于需要合并的源数据的结构。...

【MySQL技术专题】「优化技术系列」常用SQL的优化方案和技术思路

概述前面我们介绍了MySQL中怎么样通过索引来优化查询。日常开发中,除了使用查询外,我们还会使用一些其他的常用SQL,比如INSERT、GROUPBY等。对于这些SQL语句,我们该怎么样进行优化呢...

9.7寸视网膜屏原道M9i双系统安装教程

泡泡网平板电脑频道4月17日原道M9i采用Win8安卓双系统,对于喜欢折腾的朋友来说,刷机成了一件难事,那么原道M9i如何刷机呢?下面通过详细地图文,介绍原道M9i的刷机操作过程,在刷机的过程中,要...

如何做好分布式任务调度——Scheduler 的一些探索

作者:张宇轩,章逸,曾丹初识Scheduler找准定位:分布式任务调度平台...

mysqldump备份操作大全及相关参数详解

mysqldump简介mysqldump是用于转储MySQL数据库的实用程序,通常我们用来迁移和备份数据库;它自带的功能参数非常多,文中列举出几乎所有常用的导出操作方法,在文章末尾将所有的参数详细说明...

大厂面试冲刺,Java“实战”问题三连,你碰到了哪个?

推荐学习...

亿级分库分表,如何丝滑扩容、如何双写灰度

以下是基于亿级分库分表丝滑扩容与双写灰度设计方案,结合架构图与核心流程说明:一、总体设计目标...

MYSQL表设计规范(mysql表设计原则)

日常工作总结,不是通用规范一、表设计库名、表名、字段名必须使用小写字母,“_”分割。...

怎么解决MySQL中的Duplicate entry错误?

在使用MySQL数据库时,我们经常会遇到Duplicateentry错误,这是由于插入或更新数据时出现了重复的唯一键值。这种错误可能会导致数据的不一致性和完整性问题。为了解决这个问题,我们可以采取以...

高并发下如何防重?(高并发如何防止重复)

前言最近测试给我提了一个bug,说我之前提供的一个批量复制商品的接口,产生了重复的商品数据。...

性能压测数据告诉你MySQL和MariaDB该怎么选

1.压测环境为了尽可能的客观公正,本次选择同一物理机上的两台虚拟机,一台用作数据库服务器,一台用作运行压测工具mysqlslap,操作系统均为UbuntuServer22.04LTS。...

屠龙之技 --sql注入 不值得浪费超过十天 实战中sqlmap--lv 3通杀全国

MySQL小结发表于2020-09-21分类于知识整理阅读次数:本文字数:67k阅读时长≈1:01...

破防了,谁懂啊家人们:记一次 mysql 问题排查

作者:温粥一、前言谁懂啊家人们,作为一名java开发,原来以为mysql这东西,写写CRUD,不是有手就行吗;你说DDL啊,不就是设计个表结构,搞几个索引吗。...

SpringBoot系列Mybatis之批量插入的几种姿势

...

MySQL 之 Performance Schema(mysql安装及配置超详细教程)

MySQL之PerformanceSchema介绍PerformanceSchema提供了在数据库运行时实时检查MySQL服务器的内部执行情况的方法,通过监视MySQL服务器的事件来实现监视内...

取消回复欢迎 发表评论: