• JS 中的正则


    一般在哪里用得到正则?

    1. RegExp.prototype.test()

      test() 方法执行一个检索,用来查看正则表达式与指定的字符串是否匹配。返回 true 或 false。

      function test(str: string): boolean;

      若正则对象带了全局标志符号时,test() 的执行会改变正则表达式的 lastIndex 属性。连续执行 test() 方法,后续的执行将会从 lastIndex 处开始匹配字符串。

      1. var regex = /foo/g;
      2. var str = 'foo bar foo bar';
      3. console.log(regex.lastIndex); // 0  初始值为 0
      4. regex.test(str); // true
      5. console.log(regex.lastIndex); // 3
      6. regex.test(str); // true
      7. console.log(regex.lastIndex); // 11
      8. regex.test(str); // false
      9. console.log(regex.lastIndex); // 0  匹配为 false 后将 lastIndex 重置为 0
    2. RegExp.prototype.exec()

      在一个指定字符串中执行一个搜索匹配。返回一个结果数组或 null。

      正常情况下,如果匹配成功,则返回一个数组,数组的第 0 项是匹配的所有字符串,第 1 项及以后的表示括号中的分组捕获,index 表示匹配到的索引

      1. var regex = /b(a)r/;
      2. var str = 'foo bar foo bar';
      3. regex.exec(str); // [ 'bar', 'a', index: 4, input: 'foo bar foo bar', groups: undefined ]

      如果正则对象带了全局标志符号时,exec 的执行也可以被多次执行,只不过该正则对象的 lastIndex 会随着改变。当匹配失败后,exec() 方法返回 null,并将 lastIndex 重置为 0 。

      1. var regex = /bar/g;
      2. var str = 'foo bar foo bar';
      3. console.log(regex.lastIndex); // 0
      4. regex.exec(str); // [ 'bar', index: 4, input: 'foo bar foo bar', groups: undefined ]
      5. console.log(regex.lastIndex); // 7
      6. regex.exec(str); // [ 'bar', index: 12, input: 'foo bar foo bar', groups: undefined ]
      7. console.log(regex.lastIndex); // 15
      8. regex.exec(str); // null
      9. console.log(regex.lastIndex); // 0
    3. String.prototype.search()

      search() 方法执行正则表达式和 String 对象之间的一个搜索匹配。

      该方法传入一个正则表达式对象(若为非正则表达式会隐式地转换成正则表达式对象new RegExp(regexp)),返回正则表达式在字符串中首次匹配项的索引;否则,返回 -1。

      function search(reg: RegExp): number;
    4. String.prototype.match()

      检索返回一个字符串匹配正则表达式的结果。

      如果未使用 g 标志,则仅返回第一个完整匹配及其相关的捕获组(Array)。在这种情况下,返回的项目将具有如下所述的其他属性。

      1. var regex = /bar/;
      2. var str = 'foo bar foo bar';
      3. regex.exec(str); // [ 'bar', 'a', index: 4, input: 'foo bar foo bar', groups: undefined ]

      如果使用 g ,则将返回与完整正则表达式匹配的所有结果,但不会返回捕获组。

      1. var regex = /bar/g;
      2. var str = 'foo bar foo bar';
      3. regex.exec(str); // [ 'bar', 'bar' ]
    5. String.prototype.replace

      返回一个由替换值(replacement)替换部分或所有的模式(pattern)匹配项后的新字符串。模式可以是一个字符串或者一个正则表达式,替换值可以是一个字符串或者一个每次匹配都要调用的回调函数。如果 pattern 是字符串,则仅替换第一个匹配项。

    实际场景

    命名方式的转换

    假如存在这样一个对象 origin, 需要实现一个函数 format,将 origin 的下划线键名转换为 target 的驼峰式键名:

    1. // 待转换的对象
    2. var origin = {
    3.   first_name: 'a',
    4.   last_name: 'b',
    5.   say_hi: function () {
    6.     console.log('hi');
    7.   },
    8.   dream_to_be: 'teacher',
    9.   best_friends: [
    10.     {
    11.       first_name: 'c',
    12.       last_name: 'd',
    13.       favorite_sport: 'basketball',
    14.     },
    15.     {
    16.       first_name: 'e',
    17.       last_name: 'f',
    18.       say_hi: function () {
    19.         console.log('hello');
    20.       },
    21.     },
    22.   ],
    23. };
    24. // 期待转换后的对象
    25. var target = {
    26.   firstName: 'a',
    27.   lastName: 'b',
    28.   sayHi: function () {
    29.     console.log('hi');
    30.   },
    31.   dreamToBe: 'teacher',
    32.   bestFriends: [
    33.     {
    34.       firstName: 'c',
    35.       lastName: 'd',
    36.       favoriteSport: 'basketball',
    37.     },
    38.     {
    39.       firstName: 'e',
    40.       lastName: 'f',
    41.       sayHi: function () {
    42.         console.log('hello');
    43.       },
    44.     },
    45.   ],
    46. };
    47. // 需要实现的转化函数
    48. function format(origin) {}

    实现思路:遍历源对象的 key,用正则将其命名风格转换过来,如果该 key 对应的 value 是个对象或数组则递归地遍历它。

    1. const formatKey = (key) => {
    2.   return key.replace(/_(\w)/g, ($0, $1) => {
    3.     return $1.toUpperCase();
    4.   });
    5. };
    6. const isObject = (obj) => typeof obj === 'object' && obj !== null;
    7. const format = (origin) => {
    8.   if (isObject(origin)) {
    9.     return Object.keys(origin).reduce((target, key) => {
    10.       target[formatKey(key)] = format(origin[key]);
    11.       return target;
    12.     }, {});
    13.   } else if (Array.isArray(origin)) {
    14.     return origin.map(format);
    15.   } else {
    16.     return origin;
    17.   }
    18. };

    金额千分位分割

    这也是一个很常见的场景,给定一个较长字符串表示的金额,返回一个千分位分割的字符串。举例:

    1. //  待转化的字符串
    2. var originStr = '123456789.00';
    3. // 期待转换后的字符串
    4. var targetStr = '123,456,789.00';
    5. // 需要实现的转化函数
    6. function format(originStr) {}

    写这个正则需要了解位置匹配(?=p) 和 (?!p)。(?=p),其中 p 是一个子模式,即 p 前面的位置,或者说,该位置后面的字符要匹配 p。

    比如 (?=l),表示 "l" 字符前面的位置,例如:

    1. var result = 'hello'.replace(/(?=l)/g, '#');
    2. console.log(result);
    3. // => "he#l#lo"

    而 (?!p) 就是 (?=p) 的反面意思,比如:

    1. var result = 'hello'.replace(/(?!l)/g, '#');
    2. console.log(result);
    3. // => "#h#ell#o#"

    在本题中的具体实现:

    1. function format(originStr) {
    2.   var regex = /(?!^)(?=(\d{3})+\.)/g;
    3.   return originStr.replace(regex, ',');
    4. }

    或者更为直观的:

    1. function format(originStr) {
    2.   var regex = /(\d)(?=(\d{3})+\.)/g;
    3.   return originStr.replace(regex, '$1,');
    4. }

    模板字符串函数

    实现类似 ES6 中模板字符串功能的函数,将字符串中特定分割的字符替换为对象中对应的值。

    1. //  输入的字符串和对象
    2. var originStr = 'Hello, ${name}, your grade is ${grade}.';
    3. var obj = {
    4.   name: 'Micah',
    5.   grade: 90,
    6. };
    7. // 期待输出的字符串
    8. var targetStr = 'Hello, Micah, your grade is 90.';
    9. // 需要实现的转化函数
    10. function format(originStr, obj) {}

    用正则中捕获组的技巧提取位于${}的变量名,再将其与对象中的值替换:

    1. function format(originStr, obj) {
    2.   return originStr.replace(/\$\{(\w*)\}/g, (_, key) => {
    3.     return obj[key];
    4.   });
    5. }

    参考资料

    1. MDN RegExp

    2. MDN String

    3. 老姚 正则表达式

    - END -

    关于奇舞团

    奇舞团是 360 集团最大的大前端团队,代表集团参与 W3C 和 ECMA 会员(TC39)工作。奇舞团非常重视人才培养,有工程师、讲师、翻译官、业务接口人、团队 Leader 等多种发展方向供员工选择,并辅以提供相应的技术力、专业力、通用力、领导力等培训课程。奇舞团以开放和求贤的心态欢迎各种优秀人才关注和加入奇舞团。

    bdd3ac33ed43a3152c7c1f1eacb1ea64.png

  • 相关阅读:
    设计模式——责任链
    修改node_modules中安装的依赖(如第三方ui组件样式)并在下次安装时保留
    Python大数据之Python进阶(七)线程的注意点
    Redis(八) - Redis企业实战之优惠券秒杀
    【源码+课程】Java精选课程_Java基础课程_名师讲解,从入门到精通,只需这一套课程_Java300集_附Java最全学习路线图和就业分析_持续更新中
    YOLOV5、V7 训练格式转换-训练自己的数据集-猛男技术控
    代码随想录动态规划——背包问题总结篇
    NAT 技术概览(二)
    Servlet全生命周期
    matplotlib入门之抛砖引玉
  • 原文地址:https://blog.csdn.net/qiwoo_weekly/article/details/127437816