/bstorm/reg.go

http://github.com/moovweb/rubex · Go · 135 lines · 114 code · 15 blank · 6 comment · 23 complexity · 652a25568bc02d6ace838acd3feab921 MD5 · raw file

  1. // Comparing the speeds of the golang native regex library and rubex.
  2. // The numbers show a dramatic difference, with rubex being nearly 400
  3. // times slower than the native go libraries. Unfortunately for us,
  4. // the native go libraries have a different regex behavior than rubex,
  5. // so we'll have to hack at it a bit to fit our needs if we decide to use it.
  6. // (which we should, I mean, come on, 400 times faster? That's mad wins.)
  7. package main
  8. import "fmt"
  9. import re "github.com/moovweb/rubex"
  10. import "time"
  11. import "regexp"
  12. import "runtime"
  13. import "os"
  14. import "strconv"
  15. import "sync"
  16. var mu sync.Mutex
  17. var count = 0
  18. var re1 []Matcher
  19. var re2 []Matcher
  20. const NUM = 100
  21. const NNN = 1000
  22. const CCC = 100000
  23. var STR = "abcdabc"
  24. type Matcher interface {
  25. MatchString(string) bool
  26. }
  27. type Task struct {
  28. str string
  29. m Matcher
  30. t time.Time
  31. }
  32. var TaskChann chan *Task
  33. func init() {
  34. re1 = make([]Matcher, NUM)
  35. re2 = make([]Matcher, NUM)
  36. for i := 0; i < NUM; i++ {
  37. re1[i] = regexp.MustCompile("[a-c]*$")
  38. re2[i] = re.MustCompile("[a-c]*$")
  39. }
  40. TaskChann = make(chan *Task, 100)
  41. for i := 0; i < 10; i++ {
  42. STR += STR
  43. }
  44. fmt.Println("len:", len(STR))
  45. }
  46. func render_pages(name string, marray []Matcher, num_routines, num_renders int) {
  47. for i := 0; i < num_routines; i++ {
  48. m := marray[i]
  49. go func() {
  50. runtime.LockOSThread()
  51. for j := 0; j < num_renders; j++ {
  52. var totalDuration int64 = 0
  53. for i := 0; i < NNN; i++ {
  54. t := time.Now()
  55. mu.Lock()
  56. if count > CCC {
  57. mu.Unlock()
  58. return
  59. }
  60. count += 1
  61. m.MatchString(STR)
  62. mu.Unlock()
  63. totalDuration += time.Since(t).Nanoseconds()
  64. }
  65. fmt.Println(name+"-average: ", totalDuration/int64(1000*NNN), "us")
  66. }
  67. }()
  68. }
  69. }
  70. func render_pages2(name string, marray []Matcher, num_routines, num_renders int) {
  71. go func() {
  72. for i := 0; i < CCC; i++ {
  73. t := &Task{str: STR, m: marray[0], t: time.Now()}
  74. TaskChann <- t
  75. }
  76. }()
  77. for i := 0; i < num_routines; i++ {
  78. m := marray[i]
  79. go func() {
  80. runtime.LockOSThread()
  81. for j := 0; j < num_renders; j++ {
  82. var totalDuration int64 = 0
  83. for i := 0; i < NNN; i++ {
  84. task := <-TaskChann
  85. m.MatchString(task.str)
  86. totalDuration += time.Since(task.t).Nanoseconds()
  87. }
  88. fmt.Println(name+"-average: ", totalDuration/int64(1000*NNN), "us")
  89. }
  90. }()
  91. }
  92. }
  93. func main() {
  94. cpu, _ := strconv.Atoi(os.Args[1])
  95. lib := os.Args[2]
  96. method := os.Args[3]
  97. fmt.Println("using CPUs:", cpu)
  98. runtime.GOMAXPROCS(cpu)
  99. num_routines := 6
  100. num_renders := 20
  101. if method == "chan" {
  102. if lib == "rubex" {
  103. render_pages2("rubex", re2, num_routines, num_renders)
  104. } else {
  105. render_pages2("regexp", re1, num_routines, num_renders)
  106. }
  107. } else {
  108. if lib == "rubex" {
  109. render_pages("rubex", re2, num_routines, num_renders)
  110. } else {
  111. render_pages("regexp", re1, num_routines, num_renders)
  112. }
  113. }
  114. d, _ := time.ParseDuration("5s")
  115. for i := 0; i < 100; i++ {
  116. fmt.Println("goroutine:", runtime.NumGoroutine())
  117. time.Sleep(d)
  118. }
  119. fmt.Println("Done")
  120. }