Source file
src/strings/strings.go
1
2
3
4
5
6
7
8 package strings
9
10 import (
11 "internal/bytealg"
12 "internal/stringslite"
13 "math/bits"
14 "unicode"
15 "unicode/utf8"
16 )
17
18 const maxInt = int(^uint(0) >> 1)
19
20
21
22
23 func explode(s string, n int) []string {
24 l := utf8.RuneCountInString(s)
25 if n < 0 || n > l {
26 n = l
27 }
28 a := make([]string, n)
29 for i := 0; i < n-1; i++ {
30 _, size := utf8.DecodeRuneInString(s)
31 a[i] = s[:size]
32 s = s[size:]
33 }
34 if n > 0 {
35 a[n-1] = s
36 }
37 return a
38 }
39
40
41
42 func Count(s, substr string) int {
43
44 if len(substr) == 0 {
45 return utf8.RuneCountInString(s) + 1
46 }
47 if len(substr) == 1 {
48 return bytealg.CountString(s, substr[0])
49 }
50 n := 0
51 for {
52 i := Index(s, substr)
53 if i == -1 {
54 return n
55 }
56 n++
57 s = s[i+len(substr):]
58 }
59 }
60
61
62 func Contains(s, substr string) bool {
63 return Index(s, substr) >= 0
64 }
65
66
67 func ContainsAny(s, chars string) bool {
68 return IndexAny(s, chars) >= 0
69 }
70
71
72 func ContainsRune(s string, r rune) bool {
73 return IndexRune(s, r) >= 0
74 }
75
76
77 func ContainsFunc(s string, f func(rune) bool) bool {
78 return IndexFunc(s, f) >= 0
79 }
80
81
82 func LastIndex(s, substr string) int {
83 n := len(substr)
84 switch {
85 case n == 0:
86 return len(s)
87 case n == 1:
88 return bytealg.LastIndexByteString(s, substr[0])
89 case n == len(s):
90 if substr == s {
91 return 0
92 }
93 return -1
94 case n > len(s):
95 return -1
96 }
97
98 hashss, pow := bytealg.HashStrRev(substr)
99 last := len(s) - n
100 var h uint32
101 for i := len(s) - 1; i >= last; i-- {
102 h = h*bytealg.PrimeRK + uint32(s[i])
103 }
104 if h == hashss && s[last:] == substr {
105 return last
106 }
107 for i := last - 1; i >= 0; i-- {
108 h *= bytealg.PrimeRK
109 h += uint32(s[i])
110 h -= pow * uint32(s[i+n])
111 if h == hashss && s[i:i+n] == substr {
112 return i
113 }
114 }
115 return -1
116 }
117
118
119 func IndexByte(s string, c byte) int {
120 return stringslite.IndexByte(s, c)
121 }
122
123
124
125
126
127 func IndexRune(s string, r rune) int {
128 const haveFastIndex = bytealg.MaxBruteForce > 0
129 switch {
130 case 0 <= r && r < utf8.RuneSelf:
131 return IndexByte(s, byte(r))
132 case r == utf8.RuneError:
133 for i, r := range s {
134 if r == utf8.RuneError {
135 return i
136 }
137 }
138 return -1
139 case !utf8.ValidRune(r):
140 return -1
141 default:
142
143
144
145 rs := string(r)
146 last := len(rs) - 1
147 i := last
148 fails := 0
149 for i < len(s) {
150 if s[i] != rs[last] {
151 o := IndexByte(s[i+1:], rs[last])
152 if o < 0 {
153 return -1
154 }
155 i += o + 1
156 }
157
158 for j := 1; j < len(rs); j++ {
159 if s[i-j] != rs[last-j] {
160 goto next
161 }
162 }
163 return i - last
164 next:
165 fails++
166 i++
167 if (haveFastIndex && fails > bytealg.Cutover(i)) && i < len(s) ||
168 (!haveFastIndex && fails >= 4+i>>4 && i < len(s)) {
169 goto fallback
170 }
171 }
172 return -1
173
174 fallback:
175
176 if haveFastIndex {
177 if j := bytealg.IndexString(s[i-last:], string(r)); j >= 0 {
178 return i + j - last
179 }
180 } else {
181 c0 := rs[last]
182 c1 := rs[last-1]
183 loop:
184 for ; i < len(s); i++ {
185 if s[i] == c0 && s[i-1] == c1 {
186 for k := 2; k < len(rs); k++ {
187 if s[i-k] != rs[last-k] {
188 continue loop
189 }
190 }
191 return i - last
192 }
193 }
194 }
195 return -1
196 }
197 }
198
199
200
201 func IndexAny(s, chars string) int {
202 if chars == "" {
203
204 return -1
205 }
206 if len(chars) == 1 {
207
208 r := rune(chars[0])
209 if r >= utf8.RuneSelf {
210 r = utf8.RuneError
211 }
212 return IndexRune(s, r)
213 }
214 if len(s) > 8 {
215 if as, isASCII := makeASCIISet(chars); isASCII {
216 for i := 0; i < len(s); i++ {
217 if as.contains(s[i]) {
218 return i
219 }
220 }
221 return -1
222 }
223 }
224 for i, c := range s {
225 if IndexRune(chars, c) >= 0 {
226 return i
227 }
228 }
229 return -1
230 }
231
232
233
234
235 func LastIndexAny(s, chars string) int {
236 if chars == "" {
237
238 return -1
239 }
240 if len(s) == 1 {
241 rc := rune(s[0])
242 if rc >= utf8.RuneSelf {
243 rc = utf8.RuneError
244 }
245 if IndexRune(chars, rc) >= 0 {
246 return 0
247 }
248 return -1
249 }
250 if len(s) > 8 {
251 if as, isASCII := makeASCIISet(chars); isASCII {
252 for i := len(s) - 1; i >= 0; i-- {
253 if as.contains(s[i]) {
254 return i
255 }
256 }
257 return -1
258 }
259 }
260 if len(chars) == 1 {
261 rc := rune(chars[0])
262 if rc >= utf8.RuneSelf {
263 rc = utf8.RuneError
264 }
265 for i := len(s); i > 0; {
266 r, size := utf8.DecodeLastRuneInString(s[:i])
267 i -= size
268 if rc == r {
269 return i
270 }
271 }
272 return -1
273 }
274 for i := len(s); i > 0; {
275 r, size := utf8.DecodeLastRuneInString(s[:i])
276 i -= size
277 if IndexRune(chars, r) >= 0 {
278 return i
279 }
280 }
281 return -1
282 }
283
284
285 func LastIndexByte(s string, c byte) int {
286 return bytealg.LastIndexByteString(s, c)
287 }
288
289
290
291 func genSplit(s, sep string, sepSave, n int) []string {
292 if n == 0 {
293 return nil
294 }
295 if sep == "" {
296 return explode(s, n)
297 }
298 if n < 0 {
299 n = Count(s, sep) + 1
300 }
301
302 if n > len(s)+1 {
303 n = len(s) + 1
304 }
305 a := make([]string, n)
306 n--
307 i := 0
308 for i < n {
309 m := Index(s, sep)
310 if m < 0 {
311 break
312 }
313 a[i] = s[:m+sepSave]
314 s = s[m+len(sep):]
315 i++
316 }
317 a[i] = s
318 return a[:i+1]
319 }
320
321
322
323
324
325
326
327
328
329
330
331
332
333 func SplitN(s, sep string, n int) []string { return genSplit(s, sep, 0, n) }
334
335
336
337
338
339
340
341
342
343
344
345 func SplitAfterN(s, sep string, n int) []string {
346 return genSplit(s, sep, len(sep), n)
347 }
348
349
350
351
352
353
354
355
356
357
358
359
360
361 func Split(s, sep string) []string { return genSplit(s, sep, 0, -1) }
362
363
364
365
366
367
368
369
370
371
372
373 func SplitAfter(s, sep string) []string {
374 return genSplit(s, sep, len(sep), -1)
375 }
376
377 var asciiSpace = [256]uint8{'\t': 1, '\n': 1, '\v': 1, '\f': 1, '\r': 1, ' ': 1}
378
379
380
381
382
383
384 func Fields(s string) []string {
385
386
387 n := 0
388 wasSpace := 1
389
390 setBits := uint8(0)
391 for i := 0; i < len(s); i++ {
392 r := s[i]
393 setBits |= r
394 isSpace := int(asciiSpace[r])
395 n += wasSpace & ^isSpace
396 wasSpace = isSpace
397 }
398
399 if setBits >= utf8.RuneSelf {
400
401 return FieldsFunc(s, unicode.IsSpace)
402 }
403
404 a := make([]string, n)
405 na := 0
406 fieldStart := 0
407 i := 0
408
409 for i < len(s) && asciiSpace[s[i]] != 0 {
410 i++
411 }
412 fieldStart = i
413 for i < len(s) {
414 if asciiSpace[s[i]] == 0 {
415 i++
416 continue
417 }
418 a[na] = s[fieldStart:i]
419 na++
420 i++
421
422 for i < len(s) && asciiSpace[s[i]] != 0 {
423 i++
424 }
425 fieldStart = i
426 }
427 if fieldStart < len(s) {
428 a[na] = s[fieldStart:]
429 }
430 return a
431 }
432
433
434
435
436
437
438
439
440
441 func FieldsFunc(s string, f func(rune) bool) []string {
442
443
444 type span struct {
445 start int
446 end int
447 }
448 spans := make([]span, 0, 32)
449
450
451
452
453
454 start := -1
455 for end, rune := range s {
456 if f(rune) {
457 if start >= 0 {
458 spans = append(spans, span{start, end})
459
460
461
462 start = ^start
463 }
464 } else {
465 if start < 0 {
466 start = end
467 }
468 }
469 }
470
471
472 if start >= 0 {
473 spans = append(spans, span{start, len(s)})
474 }
475
476
477 a := make([]string, len(spans))
478 for i, span := range spans {
479 a[i] = s[span.start:span.end]
480 }
481
482 return a
483 }
484
485
486
487 func Join(elems []string, sep string) string {
488 switch len(elems) {
489 case 0:
490 return ""
491 case 1:
492 return elems[0]
493 }
494
495 var n int
496 if len(sep) > 0 {
497 if len(sep) >= maxInt/(len(elems)-1) {
498 panic("strings: Join output length overflow")
499 }
500 n += len(sep) * (len(elems) - 1)
501 }
502 for _, elem := range elems {
503 if len(elem) > maxInt-n {
504 panic("strings: Join output length overflow")
505 }
506 n += len(elem)
507 }
508
509 var b Builder
510 b.Grow(n)
511 b.WriteString(elems[0])
512 for _, s := range elems[1:] {
513 b.WriteString(sep)
514 b.WriteString(s)
515 }
516 return b.String()
517 }
518
519
520 func HasPrefix(s, prefix string) bool {
521 return stringslite.HasPrefix(s, prefix)
522 }
523
524
525 func HasSuffix(s, suffix string) bool {
526 return stringslite.HasSuffix(s, suffix)
527 }
528
529
530
531
532 func Map(mapping func(rune) rune, s string) string {
533
534
535
536
537
538
539 var b Builder
540
541 for i, c := range s {
542 r := mapping(c)
543 if r == c && c != utf8.RuneError {
544 continue
545 }
546
547 var width int
548 if c == utf8.RuneError {
549 c, width = utf8.DecodeRuneInString(s[i:])
550 if width != 1 && r == c {
551 continue
552 }
553 } else {
554 width = utf8.RuneLen(c)
555 }
556
557 b.Grow(len(s) + utf8.UTFMax)
558 b.WriteString(s[:i])
559 if r >= 0 {
560 b.WriteRune(r)
561 }
562
563 s = s[i+width:]
564 break
565 }
566
567
568 if b.Cap() == 0 {
569 return s
570 }
571
572 for _, c := range s {
573 r := mapping(c)
574
575 if r >= 0 {
576
577
578
579 if r < utf8.RuneSelf {
580 b.WriteByte(byte(r))
581 } else {
582
583 b.WriteRune(r)
584 }
585 }
586 }
587
588 return b.String()
589 }
590
591
592
593
594
595 const (
596 repeatedSpaces = "" +
597 " " +
598 " "
599 repeatedDashes = "" +
600 "----------------------------------------------------------------" +
601 "----------------------------------------------------------------"
602 repeatedZeroes = "" +
603 "0000000000000000000000000000000000000000000000000000000000000000"
604 repeatedEquals = "" +
605 "================================================================" +
606 "================================================================"
607 repeatedTabs = "" +
608 "\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t" +
609 "\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t"
610 )
611
612
613
614
615
616 func Repeat(s string, count int) string {
617 switch count {
618 case 0:
619 return ""
620 case 1:
621 return s
622 }
623
624
625
626
627 if count < 0 {
628 panic("strings: negative Repeat count")
629 }
630 hi, lo := bits.Mul(uint(len(s)), uint(count))
631 if hi > 0 || lo > uint(maxInt) {
632 panic("strings: Repeat output length overflow")
633 }
634 n := int(lo)
635
636 if len(s) == 0 {
637 return ""
638 }
639
640
641 switch s[0] {
642 case ' ', '-', '0', '=', '\t':
643 switch {
644 case n <= len(repeatedSpaces) && HasPrefix(repeatedSpaces, s):
645 return repeatedSpaces[:n]
646 case n <= len(repeatedDashes) && HasPrefix(repeatedDashes, s):
647 return repeatedDashes[:n]
648 case n <= len(repeatedZeroes) && HasPrefix(repeatedZeroes, s):
649 return repeatedZeroes[:n]
650 case n <= len(repeatedEquals) && HasPrefix(repeatedEquals, s):
651 return repeatedEquals[:n]
652 case n <= len(repeatedTabs) && HasPrefix(repeatedTabs, s):
653 return repeatedTabs[:n]
654 }
655 }
656
657
658
659
660
661
662
663
664
665
666
667 const chunkLimit = 8 * 1024
668 chunkMax := n
669 if n > chunkLimit {
670 chunkMax = chunkLimit / len(s) * len(s)
671 if chunkMax == 0 {
672 chunkMax = len(s)
673 }
674 }
675
676 var b Builder
677 b.Grow(n)
678 b.WriteString(s)
679 for b.Len() < n {
680 chunk := min(n-b.Len(), b.Len(), chunkMax)
681 b.WriteString(b.String()[:chunk])
682 }
683 return b.String()
684 }
685
686
687 func ToUpper(s string) string {
688 isASCII, hasLower := true, false
689 for i := 0; i < len(s); i++ {
690 c := s[i]
691 if c >= utf8.RuneSelf {
692 isASCII = false
693 break
694 }
695 hasLower = hasLower || ('a' <= c && c <= 'z')
696 }
697
698 if isASCII {
699 if !hasLower {
700 return s
701 }
702 var (
703 b Builder
704 pos int
705 )
706 b.Grow(len(s))
707 for i := 0; i < len(s); i++ {
708 c := s[i]
709 if 'a' <= c && c <= 'z' {
710 c -= 'a' - 'A'
711 if pos < i {
712 b.WriteString(s[pos:i])
713 }
714 b.WriteByte(c)
715 pos = i + 1
716 }
717 }
718 if pos < len(s) {
719 b.WriteString(s[pos:])
720 }
721 return b.String()
722 }
723 return Map(unicode.ToUpper, s)
724 }
725
726
727 func ToLower(s string) string {
728 isASCII, hasUpper := true, false
729 for i := 0; i < len(s); i++ {
730 c := s[i]
731 if c >= utf8.RuneSelf {
732 isASCII = false
733 break
734 }
735 hasUpper = hasUpper || ('A' <= c && c <= 'Z')
736 }
737
738 if isASCII {
739 if !hasUpper {
740 return s
741 }
742 var (
743 b Builder
744 pos int
745 )
746 b.Grow(len(s))
747 for i := 0; i < len(s); i++ {
748 c := s[i]
749 if 'A' <= c && c <= 'Z' {
750 c += 'a' - 'A'
751 if pos < i {
752 b.WriteString(s[pos:i])
753 }
754 b.WriteByte(c)
755 pos = i + 1
756 }
757 }
758 if pos < len(s) {
759 b.WriteString(s[pos:])
760 }
761 return b.String()
762 }
763 return Map(unicode.ToLower, s)
764 }
765
766
767
768 func ToTitle(s string) string { return Map(unicode.ToTitle, s) }
769
770
771
772 func ToUpperSpecial(c unicode.SpecialCase, s string) string {
773 return Map(c.ToUpper, s)
774 }
775
776
777
778 func ToLowerSpecial(c unicode.SpecialCase, s string) string {
779 return Map(c.ToLower, s)
780 }
781
782
783
784 func ToTitleSpecial(c unicode.SpecialCase, s string) string {
785 return Map(c.ToTitle, s)
786 }
787
788
789
790 func ToValidUTF8(s, replacement string) string {
791 var b Builder
792
793 for i, c := range s {
794 if c != utf8.RuneError {
795 continue
796 }
797
798 _, wid := utf8.DecodeRuneInString(s[i:])
799 if wid == 1 {
800 b.Grow(len(s) + len(replacement))
801 b.WriteString(s[:i])
802 s = s[i:]
803 break
804 }
805 }
806
807
808 if b.Cap() == 0 {
809 return s
810 }
811
812 invalid := false
813 for i := 0; i < len(s); {
814 c := s[i]
815 if c < utf8.RuneSelf {
816 i++
817 invalid = false
818 b.WriteByte(c)
819 continue
820 }
821 _, wid := utf8.DecodeRuneInString(s[i:])
822 if wid == 1 {
823 i++
824 if !invalid {
825 invalid = true
826 b.WriteString(replacement)
827 }
828 continue
829 }
830 invalid = false
831 b.WriteString(s[i : i+wid])
832 i += wid
833 }
834
835 return b.String()
836 }
837
838
839
840 func isSeparator(r rune) bool {
841
842 if r <= 0x7F {
843 switch {
844 case '0' <= r && r <= '9':
845 return false
846 case 'a' <= r && r <= 'z':
847 return false
848 case 'A' <= r && r <= 'Z':
849 return false
850 case r == '_':
851 return false
852 }
853 return true
854 }
855
856 if unicode.IsLetter(r) || unicode.IsDigit(r) {
857 return false
858 }
859
860 return unicode.IsSpace(r)
861 }
862
863
864
865
866
867
868 func Title(s string) string {
869
870
871
872 prev := ' '
873 return Map(
874 func(r rune) rune {
875 if isSeparator(prev) {
876 prev = r
877 return unicode.ToTitle(r)
878 }
879 prev = r
880 return r
881 },
882 s)
883 }
884
885
886
887 func TrimLeftFunc(s string, f func(rune) bool) string {
888 i := indexFunc(s, f, false)
889 if i == -1 {
890 return ""
891 }
892 return s[i:]
893 }
894
895
896
897 func TrimRightFunc(s string, f func(rune) bool) string {
898 i := lastIndexFunc(s, f, false)
899 if i >= 0 && s[i] >= utf8.RuneSelf {
900 _, wid := utf8.DecodeRuneInString(s[i:])
901 i += wid
902 } else {
903 i++
904 }
905 return s[0:i]
906 }
907
908
909
910 func TrimFunc(s string, f func(rune) bool) string {
911 return TrimRightFunc(TrimLeftFunc(s, f), f)
912 }
913
914
915
916 func IndexFunc(s string, f func(rune) bool) int {
917 return indexFunc(s, f, true)
918 }
919
920
921
922 func LastIndexFunc(s string, f func(rune) bool) int {
923 return lastIndexFunc(s, f, true)
924 }
925
926
927
928
929 func indexFunc(s string, f func(rune) bool, truth bool) int {
930 for i, r := range s {
931 if f(r) == truth {
932 return i
933 }
934 }
935 return -1
936 }
937
938
939
940
941 func lastIndexFunc(s string, f func(rune) bool, truth bool) int {
942 for i := len(s); i > 0; {
943 r, size := utf8.DecodeLastRuneInString(s[0:i])
944 i -= size
945 if f(r) == truth {
946 return i
947 }
948 }
949 return -1
950 }
951
952
953
954
955
956
957
958
959
960 type asciiSet [8]uint32
961
962
963
964 func makeASCIISet(chars string) (as asciiSet, ok bool) {
965 for i := 0; i < len(chars); i++ {
966 c := chars[i]
967 if c >= utf8.RuneSelf {
968 return as, false
969 }
970 as[c/32] |= 1 << (c % 32)
971 }
972 return as, true
973 }
974
975
976 func (as *asciiSet) contains(c byte) bool {
977 return (as[c/32] & (1 << (c % 32))) != 0
978 }
979
980
981
982 func Trim(s, cutset string) string {
983 if s == "" || cutset == "" {
984 return s
985 }
986 if len(cutset) == 1 && cutset[0] < utf8.RuneSelf {
987 return trimLeftByte(trimRightByte(s, cutset[0]), cutset[0])
988 }
989 if as, ok := makeASCIISet(cutset); ok {
990 return trimLeftASCII(trimRightASCII(s, &as), &as)
991 }
992 return trimLeftUnicode(trimRightUnicode(s, cutset), cutset)
993 }
994
995
996
997
998
999 func TrimLeft(s, cutset string) string {
1000 if s == "" || cutset == "" {
1001 return s
1002 }
1003 if len(cutset) == 1 && cutset[0] < utf8.RuneSelf {
1004 return trimLeftByte(s, cutset[0])
1005 }
1006 if as, ok := makeASCIISet(cutset); ok {
1007 return trimLeftASCII(s, &as)
1008 }
1009 return trimLeftUnicode(s, cutset)
1010 }
1011
1012 func trimLeftByte(s string, c byte) string {
1013 for len(s) > 0 && s[0] == c {
1014 s = s[1:]
1015 }
1016 return s
1017 }
1018
1019 func trimLeftASCII(s string, as *asciiSet) string {
1020 for len(s) > 0 {
1021 if !as.contains(s[0]) {
1022 break
1023 }
1024 s = s[1:]
1025 }
1026 return s
1027 }
1028
1029 func trimLeftUnicode(s, cutset string) string {
1030 for len(s) > 0 {
1031 r, n := rune(s[0]), 1
1032 if r >= utf8.RuneSelf {
1033 r, n = utf8.DecodeRuneInString(s)
1034 }
1035 if !ContainsRune(cutset, r) {
1036 break
1037 }
1038 s = s[n:]
1039 }
1040 return s
1041 }
1042
1043
1044
1045
1046
1047 func TrimRight(s, cutset string) string {
1048 if s == "" || cutset == "" {
1049 return s
1050 }
1051 if len(cutset) == 1 && cutset[0] < utf8.RuneSelf {
1052 return trimRightByte(s, cutset[0])
1053 }
1054 if as, ok := makeASCIISet(cutset); ok {
1055 return trimRightASCII(s, &as)
1056 }
1057 return trimRightUnicode(s, cutset)
1058 }
1059
1060 func trimRightByte(s string, c byte) string {
1061 for len(s) > 0 && s[len(s)-1] == c {
1062 s = s[:len(s)-1]
1063 }
1064 return s
1065 }
1066
1067 func trimRightASCII(s string, as *asciiSet) string {
1068 for len(s) > 0 {
1069 if !as.contains(s[len(s)-1]) {
1070 break
1071 }
1072 s = s[:len(s)-1]
1073 }
1074 return s
1075 }
1076
1077 func trimRightUnicode(s, cutset string) string {
1078 for len(s) > 0 {
1079 r, n := rune(s[len(s)-1]), 1
1080 if r >= utf8.RuneSelf {
1081 r, n = utf8.DecodeLastRuneInString(s)
1082 }
1083 if !ContainsRune(cutset, r) {
1084 break
1085 }
1086 s = s[:len(s)-n]
1087 }
1088 return s
1089 }
1090
1091
1092
1093 func TrimSpace(s string) string {
1094
1095 start := 0
1096 for ; start < len(s); start++ {
1097 c := s[start]
1098 if c >= utf8.RuneSelf {
1099
1100
1101 return TrimFunc(s[start:], unicode.IsSpace)
1102 }
1103 if asciiSpace[c] == 0 {
1104 break
1105 }
1106 }
1107
1108
1109 stop := len(s)
1110 for ; stop > start; stop-- {
1111 c := s[stop-1]
1112 if c >= utf8.RuneSelf {
1113
1114 return TrimRightFunc(s[start:stop], unicode.IsSpace)
1115 }
1116 if asciiSpace[c] == 0 {
1117 break
1118 }
1119 }
1120
1121
1122
1123
1124 return s[start:stop]
1125 }
1126
1127
1128
1129 func TrimPrefix(s, prefix string) string {
1130 return stringslite.TrimPrefix(s, prefix)
1131 }
1132
1133
1134
1135 func TrimSuffix(s, suffix string) string {
1136 return stringslite.TrimSuffix(s, suffix)
1137 }
1138
1139
1140
1141
1142
1143
1144
1145 func Replace(s, old, new string, n int) string {
1146 if old == new || n == 0 {
1147 return s
1148 }
1149
1150
1151 if m := Count(s, old); m == 0 {
1152 return s
1153 } else if n < 0 || m < n {
1154 n = m
1155 }
1156
1157
1158 var b Builder
1159 b.Grow(len(s) + n*(len(new)-len(old)))
1160 start := 0
1161 if len(old) > 0 {
1162 for range n {
1163 j := start + Index(s[start:], old)
1164 b.WriteString(s[start:j])
1165 b.WriteString(new)
1166 start = j + len(old)
1167 }
1168 } else {
1169 b.WriteString(new)
1170 for range n - 1 {
1171 _, wid := utf8.DecodeRuneInString(s[start:])
1172 j := start + wid
1173 b.WriteString(s[start:j])
1174 b.WriteString(new)
1175 start = j
1176 }
1177 }
1178 b.WriteString(s[start:])
1179 return b.String()
1180 }
1181
1182
1183
1184
1185
1186
1187 func ReplaceAll(s, old, new string) string {
1188 return Replace(s, old, new, -1)
1189 }
1190
1191
1192
1193
1194 func EqualFold(s, t string) bool {
1195
1196 i := 0
1197 for ; i < len(s) && i < len(t); i++ {
1198 sr := s[i]
1199 tr := t[i]
1200 if sr|tr >= utf8.RuneSelf {
1201 goto hasUnicode
1202 }
1203
1204
1205 if tr == sr {
1206 continue
1207 }
1208
1209
1210 if tr < sr {
1211 tr, sr = sr, tr
1212 }
1213
1214 if 'A' <= sr && sr <= 'Z' && tr == sr+'a'-'A' {
1215 continue
1216 }
1217 return false
1218 }
1219
1220 return len(s) == len(t)
1221
1222 hasUnicode:
1223 s = s[i:]
1224 t = t[i:]
1225 for _, sr := range s {
1226
1227 if len(t) == 0 {
1228 return false
1229 }
1230
1231
1232 var tr rune
1233 if t[0] < utf8.RuneSelf {
1234 tr, t = rune(t[0]), t[1:]
1235 } else {
1236 r, size := utf8.DecodeRuneInString(t)
1237 tr, t = r, t[size:]
1238 }
1239
1240
1241
1242
1243 if tr == sr {
1244 continue
1245 }
1246
1247
1248 if tr < sr {
1249 tr, sr = sr, tr
1250 }
1251
1252 if tr < utf8.RuneSelf {
1253
1254 if 'A' <= sr && sr <= 'Z' && tr == sr+'a'-'A' {
1255 continue
1256 }
1257 return false
1258 }
1259
1260
1261
1262 r := unicode.SimpleFold(sr)
1263 for r != sr && r < tr {
1264 r = unicode.SimpleFold(r)
1265 }
1266 if r == tr {
1267 continue
1268 }
1269 return false
1270 }
1271
1272
1273 return len(t) == 0
1274 }
1275
1276
1277 func Index(s, substr string) int {
1278 return stringslite.Index(s, substr)
1279 }
1280
1281
1282
1283
1284
1285 func Cut(s, sep string) (before, after string, found bool) {
1286 return stringslite.Cut(s, sep)
1287 }
1288
1289
1290
1291
1292
1293 func CutPrefix(s, prefix string) (after string, found bool) {
1294 return stringslite.CutPrefix(s, prefix)
1295 }
1296
1297
1298
1299
1300
1301 func CutSuffix(s, suffix string) (before string, found bool) {
1302 return stringslite.CutSuffix(s, suffix)
1303 }
1304
View as plain text