mirror of
https://github.com/vlang/v.git
synced 2025-09-14 15:02:33 +03:00
examples: add a template example, update the regex examples (#8829)
This commit is contained in:
parent
6e262b5d84
commit
3f3bec45fa
12 changed files with 654 additions and 52 deletions
127
examples/regex/regex_with_memoization.v
Normal file
127
examples/regex/regex_with_memoization.v
Normal file
|
@ -0,0 +1,127 @@
|
|||
import os
|
||||
|
||||
fn regex_match(src string, pat string) bool {
|
||||
src_size := src.len + 1
|
||||
pat_size := pat.len + 1
|
||||
mut memo := [][]int{len: src_size, init: []int{len: pat_size, init: -1}}
|
||||
return regex_match_core(src, pat, 0, 0, mut memo)
|
||||
}
|
||||
|
||||
fn regex_match_core(src string, pat string, src_pos int, pat_pos int, mut memo [][]int) bool {
|
||||
if memo[src_pos][pat_pos] != -1 {
|
||||
return memo[src_pos][pat_pos] == 1
|
||||
}
|
||||
mut spos := src_pos
|
||||
mut ppos := pat_pos
|
||||
if spos >= src.len && ppos >= pat.len {
|
||||
memo[src_pos][pat_pos] = 1
|
||||
return true
|
||||
} else if spos < src.len && ppos >= pat.len {
|
||||
memo[src_pos][pat_pos] = 0
|
||||
return false
|
||||
} else if spos >= src.len && ppos < pat.len {
|
||||
if pat[ppos] == `\\` {
|
||||
ppos++
|
||||
}
|
||||
res := ppos + 1 < pat.len && pat[ppos + 1] in [`*`, `?`]
|
||||
&& regex_match_core(src, pat, spos, ppos + 2, mut memo)
|
||||
memo[src_pos][pat_pos] = if res { 1 } else { 0 }
|
||||
return res
|
||||
} else {
|
||||
first_is_bslash := pat[ppos] == `\\`
|
||||
if first_is_bslash {
|
||||
ppos++
|
||||
}
|
||||
first_bslash_and_match := first_is_bslash && ppos < pat.len
|
||||
&& (((pat[ppos] == `d` && src[spos].is_digit())
|
||||
|| (pat[ppos] == `D` && !src[spos].is_digit())
|
||||
|| (pat[ppos] == `s` && src[spos].is_space())
|
||||
|| (pat[ppos] == `S` && !src[spos].is_space())
|
||||
|| (pat[ppos] == `w` && (src[spos].is_digit() || src[spos].is_letter()
|
||||
|| src[spos] == `_`)) || (pat[ppos] == `W` && !(src[spos].is_digit()
|
||||
|| src[spos].is_letter() || src[spos] == `_`)))
|
||||
|| (pat[ppos] in [`d`, `D`, `s`, `S`, `w`, `W`] && ppos + 1 < pat.len
|
||||
&& pat[ppos + 1] in [`*`, `?`, `+`])
|
||||
|| (pat[ppos] !in [`d`, `D`, `s`, `S`, `w`, `W`] && src[spos] == pat[ppos]))
|
||||
if ppos + 1 < pat.len {
|
||||
match pat[ppos + 1] {
|
||||
`*` {
|
||||
if first_bslash_and_match {
|
||||
res := regex_match_core(src, pat, spos + 1, ppos - 1, mut memo)
|
||||
|| regex_match_core(src, pat, spos, ppos + 2, mut memo)
|
||||
memo[src_pos][pat_pos] = if res { 1 } else { 0 }
|
||||
return res
|
||||
} else if src[spos] == pat[ppos] || pat[ppos] == `.` {
|
||||
res := regex_match_core(src, pat, spos + 1, ppos, mut memo)
|
||||
|| regex_match_core(src, pat, spos, ppos + 2, mut memo)
|
||||
memo[src_pos][pat_pos] = if res { 1 } else { 0 }
|
||||
return res
|
||||
} else {
|
||||
res := regex_match_core(src, pat, spos, ppos + 2, mut memo)
|
||||
memo[src_pos][pat_pos] = if res { 1 } else { 0 }
|
||||
return res
|
||||
}
|
||||
}
|
||||
`+` {
|
||||
if first_bslash_and_match {
|
||||
res := regex_match_core(src, pat, spos + 1, ppos - 1, mut memo)
|
||||
|| regex_match_core(src, pat, spos + 1, ppos + 2, mut memo)
|
||||
memo[src_pos][pat_pos] = if res { 1 } else { 0 }
|
||||
return res
|
||||
} else if src[spos] == pat[ppos] || pat[ppos] == `.` {
|
||||
res := regex_match_core(src, pat, spos + 1, ppos, mut memo)
|
||||
|| regex_match_core(src, pat, spos + 1, ppos + 2, mut memo)
|
||||
memo[src_pos][pat_pos] = if res { 1 } else { 0 }
|
||||
return res
|
||||
} else {
|
||||
memo[src_pos][pat_pos] = 0
|
||||
return false
|
||||
}
|
||||
}
|
||||
`?` {
|
||||
if first_bslash_and_match || src[spos] == pat[ppos] || pat[ppos] == `.` {
|
||||
res := regex_match_core(src, pat, spos + 1, ppos + 2, mut memo)
|
||||
|| regex_match_core(src, pat, spos, ppos + 2, mut memo)
|
||||
memo[src_pos][pat_pos] = if res { 1 } else { 0 }
|
||||
return res
|
||||
} else {
|
||||
res := regex_match_core(src, pat, spos, ppos + 2, mut memo)
|
||||
memo[src_pos][pat_pos] = if res { 1 } else { 0 }
|
||||
return res
|
||||
}
|
||||
}
|
||||
else {}
|
||||
}
|
||||
}
|
||||
if first_is_bslash {
|
||||
res := first_bslash_and_match
|
||||
&& regex_match_core(src, pat, spos + 1, ppos + 1, mut memo)
|
||||
memo[src_pos][pat_pos] = if res { 1 } else { 0 }
|
||||
return res
|
||||
} else {
|
||||
res := (src[spos] == pat[ppos] || pat[ppos] == `.`) && pat[ppos] != `\\`
|
||||
&& regex_match_core(src, pat, spos + 1, ppos + 1, mut memo)
|
||||
memo[src_pos][pat_pos] = if res { 1 } else { 0 }
|
||||
return res
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn main() {
|
||||
mut cnt := 0
|
||||
println('currently supported patterns: . ? + * \\ \\d \\D \\s \\S \\w \\W')
|
||||
println('example: source `address@domain.net` matches pattern `\\w+@domain\\.net`')
|
||||
println('enter `exit` to quit\n')
|
||||
for {
|
||||
cnt++
|
||||
src := os.input('[$cnt] enter source string: ')
|
||||
if src == 'exit' {
|
||||
break
|
||||
}
|
||||
pat := os.input('[$cnt] enter pattern string: ')
|
||||
if pat == 'exit' {
|
||||
break
|
||||
}
|
||||
println('[$cnt] whether `$src` matches `$pat`: ${regex_match(src, pat)}')
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue