c - 为什么 srand(time(NULL)) 在 main 上出现段错误?

标签 c random time srand ucontext

这里需要一些帮助。

我想了解这段代码中发生了什么。

我正在尝试生成随机数,作为在 ccreate 函数内创建的 TCP_t 结构的门票

问题是,每次我在没有 srand(time(NULL)) 的情况下执行此代码时,它都会一遍又一遍地返回相同的“随机”数字序列,对于示例:

TID: 0 | TICKET : 103
TID: 1 | TICKET : 198

所以我用时间播种它,以生成真正的随机数。

当我将种子放入 newTicket 函数中时,它在每次执行中都会带来不同的数字,但每个线程都会带来相同的数字。以下是输出示例:

执行1:

TID: 0 | TICKET : 148
TID: 1 | TICKET : 148

执行2:

TID: 0 | TICKET : 96
TID: 1 | TICKET : 96

所以,经过一些研究,我发现我不应该每次调用 rand 时都播种它,而只能在程序开始时播种一次。现在,将种子放入 main 函数后,它给了我段错误,我不知道为什么。

这可能是一个愚蠢的问题,但我真的很想了解发生了什么。

种子是否以某种方式搞砸了什么? 我错过了什么吗? 我应该以其他方式生成随机数吗?

#include <ucontext.h>
#include <stdlib.h>
#include <stdio.h>
#include <time.h>

#define MAX_TICKET 255
#define STACK_SIZE 32000

typedef struct s_TCB {
  int threadId;
  int ticket;
  ucontext_t context;
} TCB_t;

void test();
int newTicket();
int newThreadId();
int ccreate (void* (*start)(void*), void *arg);
int threadId = 0;

int main(){
  srand(time(NULL)); //<<<============== HERE = SEGMENTATION FAULT
  ccreate((void*)&test, 0);
  ccreate((void*)&test, 0);
}

int ccreate (void* (*start)(void*), void *arg){
    if(start == NULL) return -1;

    ucontext_t threadContext;
    getcontext(&threadContext);
    makecontext(&threadContext, (void*)start, 0);
    threadContext.uc_stack.ss_sp = malloc(STACK_SIZE);
    threadContext.uc_stack.ss_size = STACK_SIZE;

    TCB_t * newThread = malloc(sizeof(TCB_t));
    if (newThread == NULL) return -1;

    int threadThreadId = newThreadId();
    newThread->threadId = threadThreadId;
    newThread->ticket = newTicket();

    printf("TID: %d | TICKET : %d\n", newThread->threadId, newThread->ticket);

    return threadThreadId;
}

int newThreadId(){
  int newThreadId = threadId;
  threadId++;
  return newThreadId;
}

int newTicket(){
  //srand(time(NULL)); //<<<============== HERE = IT PARTIALLY WORKS
  return (rand() % (MAX_TICKET+1));
}

void test(){
  printf("this is a test function");
}

感谢所有向我伸出援手的人。

如果代码太难看而无法阅读,我们深表歉意。尽我所能地简化它。

最佳答案

问题不在于 srand(time(NULL)),而在于 makecontext
您可以通过 Sanatizer 运行代码来确认:

gcc-6 -fsanitize=undefined -fsanitize=address -fsanitize=leak -fsanitize-recover=all -fuse-ld=gold -o main main.c
./main 
ASAN:DEADLYSIGNAL
=================================================================
==8841==ERROR: AddressSanitizer: SEGV on unknown address 0x7fc342ade618 (pc 0x7fc340aad235 bp 0x7ffd1b945950 sp 0x7ffd1b9454f8 T0)
    #0 0x7fc340aad234 in makecontext (/lib/x86_64-linux-gnu/libc.so.6+0x47234)
    #1 0x400d2f in ccreate (/home/malko/Desktop/main+0x400d2f)
    #2 0x400c19 in main (/home/malko/Desktop/main+0x400c19)
    #3 0x7fc340a87f44 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21f44)
    #4 0x400b28  (/home/malko/Desktop/main+0x400b28)

AddressSanitizer can not provide additional info.
SUMMARY: AddressSanitizer: SEGV (/lib/x86_64-linux-gnu/libc.so.6+0x47234) in makecontext
==8841==ABORTING

您可以通过创建上下文之前设置堆栈大小来解决问题:

char stack[20000];
threadContext.uc_stack.ss_sp = stack;
threadContext.uc_stack.ss_size = sizeof(stack);
makecontext(&threadContext, (void*)start, 0);

不相关,但请确保您还在示例代码中释放了 malloc 的内存。

关于c - 为什么 srand(time(NULL)) 在 main 上出现段错误?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39379824/

相关文章:

algorithm - 编写一个 36 位随机数生成器

java - 选择一个随机项目但投票分数存在偏差?

c - 指向指针的指针有什么作用?字符**

c - C 中数组可以保存超出范围的值

C语言 : left shift value with negative sign

javascript - 什么是好的 Javascript 时间选择器?

Python timeit 和程序输出

c - 一次又一次初始化时,指针在循环内做了什么?

C# Random() 不是很随机地工作

java - Scala - 组合 LocalDate 和 LocalTime 时如何正确添加 ZoneOffset?