Heim Backend-Entwicklung PHP-Tutorial PHP 随机数 C扩展随机数_PHP

PHP 随机数 C扩展随机数_PHP

May 27, 2016 am 10:18 AM
php 随机数

由于要用到固定长度的随机字符串。

首先是一段PHP代码

1

2

3

4

5

6

7

8

$str_md5=md5(uniqid());

  $rand = mt_rand(1, 28);

  $str1=substr($str_md5,$rand,6);

  $rand = mt_rand(1, 28);

  $str2=substr($str_md5,$rand,6);

  $rand = mt_rand(1, 28);

  $str3=substr($str_md5,$rand,6);

  $code=substr($str1.$str2.$str3,0,8);

Nach dem Login kopieren


生成180000个随机字符串,图中是按照重复数量倒序排列,可以看到基本都有重复的。不过也是比较理想的。

由于想提升一下自己的C语言能力,所以用C重新写了一下随机生成字符串。

其中用到了随机数函数srand(),rand();

不过折腾一两个小时,随机数还是有问题。并发访问时时间可能几乎为同时,那么srand给的种子时间可以视为相同的。这样就导致了,产生的随机数也是一样的。从而产生的随机字符串也是一样的。循环输出随机字符串,几乎都是一模一样的。

后来想到了ukey,这个扩展可以实现唯一的ID,那么访问都产生唯一的ID,是不是可以将这个ID作为种子时间。答案是肯定的。

上图是产生的随机字符串,可以自定义长度。也同样可以输出只有数字的字符串。相较PHP所产生的随机字符串重复率更低且速度更快。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

95

96

97

98

99

100

101

102

103

104

105

106

107

108

109

110

111

112

113

114

115

116

117

118

119

120

121

122

123

124

125

126

127

128

129

130

131

132

133

134

135

136

137

138

139

140

141

142

143

144

145

146

147

148

149

150

151

152

PHP_FUNCTION(get_random__num_str)

{

   int length=8;

    

   if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &length) == FAILURE)

   {

    length=8;

      

   }

   length++;

  int flag, i;

  char* string;

  __uint64_t timestamp = realtime();

  __uint64_t retval;

  int len;

  char buf[128];

  

  if (timestamp == 0ULL) {

    RETURN_FALSE;

  }

  

  spin_lock(lock, pid);

  

  if (context->last_timestamp == timestamp) {

    context->sequence = (context->sequence + 1) & context->sequence_mask;

    if (context->sequence == 0) {

      timestamp = skip_next_millis();

    }

  

  } else {

    context->sequence = 0; /* Back to zero */

  }

  

  context->last_timestamp = timestamp;

  

  retval = ((timestamp - context->twepoch) << context->timestamp_left_shift)

      | (context->datacenter_id << context->datacenter_id_shift)

      | (worker_id << context->worker_id_shift)

      | context->sequence;

  

  spin_unlock(lock, pid);

  //printf('%ld',retval);

  srand((unsigned)retval);

  //srand((unsigned) time(NULL ));

  if ((string = (char*) emalloc(length)) == NULL )

  {

    //myLog("Malloc failed!flag:14\n");

    RETURN_NULL() ;

  }

   

  for (i = 0; i < length - 1; i++)

  {

    flag = rand() % 3;

      

    switch (flag)

    {

      case 0:

        string[i] = '1' + rand() % 5;

        break;

      case 1:

        string[i] = '2' + rand() % 7;

        break;

      case 2:

        string[i] = '0' + rand() % 10;

        break;

      default:

        string[i] = '9';

        break;

    }

      

      

      

  }

  string[length - 1] = '\0';

  RETURN_STRINGL(string,length,0);

}

 PHP_FUNCTION(get_random_str)

{

   int length=8;

    

   if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &length) == FAILURE)

   {

    length=8;

      

   }

   length++;

  int flag, i;

  char* string;

  __uint64_t timestamp = realtime();

  __uint64_t retval;

  int len;

  char buf[128];

  

  if (timestamp == 0ULL) {

    RETURN_FALSE;

  }

  

  spin_lock(lock, pid);

  

  if (context->last_timestamp == timestamp) {

    context->sequence = (context->sequence + 1) & context->sequence_mask;

    if (context->sequence == 0) {

      timestamp = skip_next_millis();

    }

  

  } else {

    context->sequence = 0; /* Back to zero */

  }

  

  context->last_timestamp = timestamp;

  

  retval = ((timestamp - context->twepoch) << context->timestamp_left_shift)

      | (context->datacenter_id << context->datacenter_id_shift)

      | (worker_id << context->worker_id_shift)

      | context->sequence;

  

  spin_unlock(lock, pid);

  //printf('%ld',retval);

  srand((unsigned)retval);

  //srand((unsigned) time(NULL ));

  if ((string = (char*) emalloc(length)) == NULL )

  {

    //myLog("Malloc failed!flag:14\n");

    RETURN_NULL() ;

  }

   

  for (i = 0; i < length - 1; i++)

  {

    flag = rand() % 3;

      

    switch (flag)

    {

      case 0:

        string[i] = 'A' + rand() % 26;

        break;

      case 1:

        string[i] = 'a' + rand() % 26;

        break;

      case 2:

        string[i] = '0' + rand() % 10;

        break;

      default:

        string[i] = 'x';

        break;

    }

      

      

      

  }

  string[length - 1] = '\0';

  RETURN_STRINGL(string,length,0);

}

Nach dem Login kopieren


上图是PHP生成18W随机字符串所用的时间

上图是C扩展生成18W随机字符串所用的时间

所用的服务器都是1G内存 双核的阿里云服务器。

只要在ukey中加入上如代码就可以生产随机字符串和随机长度数字字符串,PHP唯一ID生成扩展ukey。

php.ini的配置项:

1

2

3

4

[ukey]

ukey.datacenter = integer

ukey.worker = integer

ukey.twepoch = uint64

Nach dem Login kopieren

datacenter配置项是一个整数, 用于设置数据中心;
worker配置项是一个整数, 用于设置数据中心的机器序号;
twepoch配置项是一个64位的整数, 用于设置时间戳基数, 此值越大, 生成的ID越小;

安装:

1

2

3

4

5

$ cd ./ukey

$ phpize

$ ./configure

$ make

$ sudo make install

Nach dem Login kopieren

Ukey提供3个有用的函数:

ukey_next_id() -- 用于生成唯一ID
ukey_to_timestamp(ID) -- 用于将ID转换成时间戳
ukey_to_machine(ID) -- 用于将ID转换成机器信息

使用实例:

1

2

3

4

5

6

7

8

9

10

<&#63;php

$id = ukey_next_id();

echo $id;

  

$timestamp = ukey_to_timestamp($id);

echo date('Y-m-d H:i:s', $timestamp);

  

$info = ukey_to_machine($id)

var_dump($info);

&#63;>

Nach dem Login kopieren

以上就是本文的全部内容,希望对大家的学习有所帮助。

Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn

Heißer Artikel

R.E.P.O. Energiekristalle erklärten und was sie tun (gelber Kristall)
2 Wochen vor By 尊渡假赌尊渡假赌尊渡假赌
Repo: Wie man Teamkollegen wiederbelebt
4 Wochen vor By 尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island Abenteuer: Wie man riesige Samen bekommt
3 Wochen vor By 尊渡假赌尊渡假赌尊渡假赌

Heißer Artikel

R.E.P.O. Energiekristalle erklärten und was sie tun (gelber Kristall)
2 Wochen vor By 尊渡假赌尊渡假赌尊渡假赌
Repo: Wie man Teamkollegen wiederbelebt
4 Wochen vor By 尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island Abenteuer: Wie man riesige Samen bekommt
3 Wochen vor By 尊渡假赌尊渡假赌尊渡假赌

Heiße Artikel -Tags

Notepad++7.3.1

Notepad++7.3.1

Einfach zu bedienender und kostenloser Code-Editor

SublimeText3 chinesische Version

SublimeText3 chinesische Version

Chinesische Version, sehr einfach zu bedienen

Senden Sie Studio 13.0.1

Senden Sie Studio 13.0.1

Leistungsstarke integrierte PHP-Entwicklungsumgebung

Dreamweaver CS6

Dreamweaver CS6

Visuelle Webentwicklungstools

SublimeText3 Mac-Version

SublimeText3 Mac-Version

Codebearbeitungssoftware auf Gottesniveau (SublimeText3)

PHP 8.4 Installations- und Upgrade-Anleitung für Ubuntu und Debian PHP 8.4 Installations- und Upgrade-Anleitung für Ubuntu und Debian Dec 24, 2024 pm 04:42 PM

PHP 8.4 Installations- und Upgrade-Anleitung für Ubuntu und Debian

CakePHP-Projektkonfiguration CakePHP-Projektkonfiguration Sep 10, 2024 pm 05:25 PM

CakePHP-Projektkonfiguration

CakePHP Datum und Uhrzeit CakePHP Datum und Uhrzeit Sep 10, 2024 pm 05:27 PM

CakePHP Datum und Uhrzeit

CakePHP-Datei hochladen CakePHP-Datei hochladen Sep 10, 2024 pm 05:27 PM

CakePHP-Datei hochladen

CakePHP-Routing CakePHP-Routing Sep 10, 2024 pm 05:25 PM

CakePHP-Routing

Besprechen Sie CakePHP Besprechen Sie CakePHP Sep 10, 2024 pm 05:28 PM

Besprechen Sie CakePHP

So richten Sie Visual Studio-Code (VS-Code) für die PHP-Entwicklung ein So richten Sie Visual Studio-Code (VS-Code) für die PHP-Entwicklung ein Dec 20, 2024 am 11:31 AM

So richten Sie Visual Studio-Code (VS-Code) für die PHP-Entwicklung ein

CakePHP-Kurzanleitung CakePHP-Kurzanleitung Sep 10, 2024 pm 05:27 PM

CakePHP-Kurzanleitung

See all articles