【javascript杂谈】你所不知道的replace函数

原文:【javascript杂谈】你所不知道的replace函数

前言

最近在做面试题的时候总会用到这个函数,这个函数总是和正则表达式联系到一起,并且效果很是不错,总能很简单出色的完成字符串的实际问题,大家肯定都会使用这个函数,像我一样的初学者可能对这个函数的了解还是不够深的,今天就总结一下,了解一下,再做几道网上的题目练练手,给将要面试的同学打打气。

介绍

使用一个替换值替换掉一个替换模式在原字符串中一个或所有的匹配项,并返回替换后的字符串,这个替换模式可以是字符串或者正则表达式,替换值可以是一个字符串或者是一个函数。

后面的大的黑体就是我们开始没有注意到的。

语法

newstring = str.replace(regexp|substr, newSubStr|function[,  flags]);这是官方的解释不是很清楚,一会好好解释下。

参数

regexp
一个 REGEXP(正则)对象,该正则所匹配的内容会被第二个参数的返回值替换掉。
substr
替换掉的一个字符串
newSubStr
替换掉第一个参数在原字符串中的匹配部分,该字符串中可以内插一些特殊的变量名。对于正则replace约定了一个特殊标记符$:
  1. $n (n:1-99) : 表示从左到右正则子表达式所匹配的文本。各个分组匹配的字符串
  2. $&:表示与正则表达式匹配的全文本。
  3. $`(`:切换技能键):表示匹配字符串的左边文本。
  4. $’(‘:单引号):表示匹配字符串的右边文本。
  5. $$:表示$转移。
function
是一个函数,可以有返回值,也可以无返回值。参数的个数不固定,参数挺麻烦的,咱主要就是说这个函数,
  1. 第一个参数为每次匹配的全文本($&),就是整个匹配的字符串
  2. 中间参数为子表达式匹配字符串,个数不限.( $n (n:1-99)),有分组的时候
  3. 倒数第二个参数为匹配文本字符串的匹配下标位置。
  4. 最后一个参数表示字符串本身。

这个函数在执行replace时执行,如果有返回值,返回值用来替换replace的第一个参数。

flags
就是正则表达式那些特殊的标志,可有可无的,当第一个参数是字符串的时候使用。
g   全局替换
i   忽略大小写
m   多行模式
y      sticky

注意:replace函数并不改变原来的字符串,而是返回来一个新的字符串。

好多例子

说估计大家也是不明白,现在我把所有的情况弄个例子,帮助大家理解:

例子1:

//replace(str,str);var str = "I am hainan";
var newStr = str.replace("hainan","Allenxing");
console.log(newStr);//I am Allenxing
console.log(str);//I am hainan

例子2:

// replace(str,str,flag)
var str = "I am hainan hainan";
var newStr = str.replace("hainan","Allenxing","gi");
console.log(newStr);//I am Allenxing Allenxing(FireFox下)  I am Allenxing Allenxing(chrome IE下)
console.log(str);//I am hainan hainan

这种各情况个浏览器支持的不好,建议使用带标记的正则表达式,而不是使用字符串的这种标记,官方这样说

A string specifying a combination of regular expression flags. The use of the flags parameter in the String.replace method is non-standard, use a RegExp object with the corresponding flags.

例子3:

// replace(RegExp,str)
var str = "I am hainan hainan";
var newStr1 = str.replace(/hainan/,"Allenxing");
console.log(newStr1);//I am Allenxing hainan 

var newStr2 = str.replace(/hainan/g,"Allenxing");
console.log(newStr2);//I am Allenxing Allenxing

var str1 = "I am hainan Hainan";
console.log(str1.replace(/hainan/ig,"Allenxing"));//I am Allenxing Allenxing
console.log(str1.replace(/hainan/g,"Allenxing"));//I am Allenxing Hainan

例子4:$1是匹配第一个分组的内容,$2是匹配第一个分组的内容......

//replace(RegExp,'特殊标记')
var str = "I am hainan hainan";
console.log( str.replace(/(hainan)/,"*$1*") );//I am *hainan* hainan 

解释一下:/(hainan)/ 匹配的是第一个hainan,我们使用"*$1*"的意思就是"*hainan*",用它替换hainan,是第二个参数替换前面的整体。

例子5:

//replace(RegExp,'特殊标记')
var str = "I am hainan Allenxing";
console.log( str.replace(/(hainan)\s*(Allenxing)/g,"$2 $1") );//I am Allenxing hainan 

例子6:

//replace(RegExp,'特殊标记')
var str = "I am hainan Allenxing";
console.log( str.replace(/(hainan)\s*(?:Allenxing)/g,"$2 $1") );//I am $2 hainan 

解释:(?:)是非捕获分组,匹配内容但是不捕获,也就是$2中现在没有内容,那么就按照$2是个字符串输出。

例子7:$&代表正则匹配的整体

//replace(RegExp,'特殊标记')
var str = "hainan";
console.log( str.replace(/(h)ainan/g,"$& starts with $1") );//hainan starts with h 

例子8:$`表示匹配字符串的左边文本,$’(‘:单引号)表示匹配字符串的右边文本。

//replace(RegExp,"特殊标记")
var str = "javascript";
console.log(str.replace(/java/,"$&$' is "));//javascript is script 

console.log(str.replace(/script/,"$& is not $`"));//avascript is not java 

参数是function 

我们前面简单介绍了参数function,我们写个例子看看这个函数的参数都有什么东西,

//replace(RegExp,function)
var str = "I am hainan Allenxing";
console.log( str.replace(/(hainan)\s*(Allenxing)/g,function(){
  console.(arguments);//看看参数的样子
}) );

看截图吧,清楚些

这样可以清楚的看出来,函数的第一个参数是正则匹配的全文本(hainan Allenxing),第二个是第一个分组匹配的文本(hainan),第三个是第二个分组匹配的文本(Allenxing),第四个参数是匹配文本字符串的匹配下标位置,第五个参数是整个输入的字符串(I am hainan Allenxing)。后面的几个参数的位置肯能会改变,因为再有分组的话,中间分组的参数个数会增加。

再看下结果,I am undefined  ,由于我们function没有明确返回值,所以就是undefined,用undefined替换了正则匹配的全文本。

还有一个问题就是,匹配的时候这个function要执行几次呢?每一次里面的参数的值会发生什么变化呢?看例子

//replace(RegExp,function)
var str = "I am hainan hainan";
console.log( str.replace(/(hainan)/g,function(){
  console.log(arguments);
}) );

结果就是这样

这个可以看出,function执行了两遍,就是因为正则(/(hainan)/g)匹配了两遍,我们可以再验证一下,

//replace(RegExp,function)
var str = "I am hainan hainan";
console.log( str.replace(/(hainan)/,function(){
  console.log(arguments);
}) );

确实是这样吧,里面的参数时每一次匹配的结果。而我们每次替换的值就是第一个参数的值,因为第一的参数就是正则匹配的全文本嘛。

用function解决几个前面的例子吧

先看例子5那个

//replace(RegExp,'特殊标记')
var str = "I am hainan Allenxing";
console.log(str.replace(/(hainan)\s*(Allenxing)/g,function(arg0,arg1,arg2){
     return arg2 + " " + arg1;
}));
//I am Allenxing hainan

再看例子8

var str = "javascript";
console.log(str.replace(/java(script)/g,function(arg0,arg1){
   return arg0 +" is " + arg1;
}))

都是差不多的,只是用function做参数时,我们不能使用原来的正则表达式来得到匹配字符的左右的字符,需要修改正则表达式。

弄一个实际的例子,查找字符串中出现最多的字符和个数?   例如:sdjksfssscfssdd  -> 字符最多的是s,出现了7次。用replace函数解决。

var str = "sdjksfssscfssdd";
str = str.split("").sort().join("");
var count = 0;
var val;
str.replace(/(\w)\1*/g,function(arg0,arg1){
    if(arg0.length > count){
      count = arg0.length;
      val = arg1;
    }
});
console.log(val+" 出现了"+count+" 次");

小结

replace函数就总结到这,其实它能做好多事情,其实主要是正则表达式的功劳,正则表达式这里就不详细说了,园子里好多牛人都总结过,大家可以找找看看,想要面试的同学好好准备吧。

 

时间: 2024-11-30 10:19:16

【javascript杂谈】你所不知道的replace函数的相关文章

关于JavaScript数组你所不知道的3件事_javascript技巧

在编程语言当中,数组(Array)是一个非常常用的功能:它是一种特殊的变量,可以用来同时储存多个数值.然而,在JavaScript方面,数组的功能还有很多其他值得探索的地方. 在这篇文章中,我们将会讨论JavaScript数组的三个并不那么常见的功能. 1. 给数组添加自定义属性 在网上搜寻有关JavaScript数组的定义时,你会发现几乎所有人对于数组的定义都一样:一种对象. 事实上,我们用JavaScript处理的所有东西,都可以视为是一种对象.JavaScript中有两种数据类型,基本类型

你所不知道的SQL Server数据库启动过程,以及启动不起来的各种问题的分析及解决技巧

原文:你所不知道的SQL Server数据库启动过程,以及启动不起来的各种问题的分析及解决技巧 目前SQL Server数据库作为微软一款优秀的RDBMS,其本身启动的时候是很少出问题的,我们在平时用的时候,很少关注起启动过程,或者很少了解其底层运行过程,大部分的过程只关注其内部的表.存储过程.视图.函数等一系列应用方式,而当有一天它运行的正常的时候突然启动不起来了,这时候就束手无策了,能做的或许只能是重装.配置.还原等,但这一个过程其实是一个非常耗时的过程,尤其当我们面对是庞大的生产库的时候,

你真的会玩SQL吗?你所不知道的 数据聚合

  你真的会玩SQL吗?系列目录   你真的会玩SQL吗?之逻辑查询处理阶段   你真的会玩SQL吗?和平大使 内连接.外连接   你真的会玩SQL吗?三范式.数据完整性   你真的会玩SQL吗?查询指定节点及其所有父节点的方法   你真的会玩SQL吗?让人晕头转向的三值逻辑   你真的会玩SQL吗?EXISTS和IN之间的区别   你真的会玩SQL吗?无处不在的子查询   你真的会玩SQL吗?Case也疯狂   你真的会玩SQL吗?表表达式,排名函数   你真的会玩SQL吗?简单的 数据修改

你所不知道的关于网管的危险做法

  你所不知道的关于网管的危险做法 网络管理员是指向社会公众开放的营业性上网服务提供场所里的管理员.Jeff Dray 最近经过对IT行业的深入调查研究,通过总结和分析针对IT行业列出了一份类别名单.在这里,他定义了七类最不安全的网络管理员.如果你是一名网络管理员,并且已经意识到工作中还存在着不足,看看你属于名单中的哪一类? 大多数网络管理员对工作游刃有余,并且可以在一个具有高度挑战和技术难度的任务中,使工作顺利进行.然而,有时他们中的某些人会变得很难缠,并会阻碍事情的顺利进行.所以,我定义了一

【干货合集】你所不知道的蚂蚁技术系列之(一):系统设计、性能优化、运维

8月30-31日20:00-21:30,一场别开生面的技术大会-- "蚂蚁金服&阿里云在线金融技术峰会"将在线举办.本次将聚焦数据库.应用架构.移动开发.机器学习等热门领域,帮助金融业技术开发者深入解析互联网应用的前沿应用与技术实践. 蚂蚁金服&阿里云在线金融技术峰会专题:https://yq.aliyun.com/activity/109 峰会统一报名链接:http://yq.aliyun.com/webinar/join/38 2015双11,蚂蚁金服旗下支付宝共完

【干货合集】你所不知道的蚂蚁技术系列之(二):数据、Docker、测试与无线网络技术

8月30-31日20:00-21:30,一场别开生面的技术大会-- "蚂蚁金服&阿里云在线金融技术峰会"将在线举办.本次将聚焦数据库.应用架构.移动开发.机器学习等热门领域,帮助金融业技术开发者深入解析互联网应用的前沿应用与技术实践. 蚂蚁金服&阿里云在线金融技术峰会专题:https://yq.aliyun.com/activity/109 峰会统一报名链接:http://yq.aliyun.com/webinar/join/38 2015双11,蚂蚁金服旗下支付宝共完

【干货合集】你所不知道的蚂蚁技术系列之(三):咻红包、人脸识别、人工智能、金融技术

8月30-31日20:00-21:30,一场别开生面的技术大会-- "蚂蚁金服&阿里云在线金融技术峰会"将在线举办.本次将聚焦数据库.应用架构.移动开发.机器学习等热门领域,帮助金融业技术开发者深入解析互联网应用的前沿应用与技术实践. 蚂蚁金服&阿里云在线金融技术峰会专题:https://yq.aliyun.com/activity/109 峰会统一报名链接:http://yq.aliyun.com/webinar/join/38 2015双11,蚂蚁金服旗下支付宝共完

你所不知道的CSS滤镜技巧与细节

本文主要介绍 CSS 滤镜的不常用用法,希望能给读者带来一些干货! OK,下面直接进入正文.本文所描述的滤镜,指的是 CSS3 出来后的滤镜,不是 IE 系列时代的滤镜,语法如下,还未接触过这个属性的可以先简单到 MDN - filter 了解下: {      filter: blur(5px);      filter: brightness(0.4);      filter: contrast(200%);      filter: drop-shadow(16px 16px 20px 

QML中你所不知道的state

QML中你所不知道的state        最后一次写QML已经是2010年了,最近由于产品需要,重拾QML.之前nokia给我们培训QML的时候,对于state这个概念理解的不是很透彻.最近在做产品前期的QML热身,发现QML中的state有一种神奇的功能:历史记忆效应        state核心就是体现了一个状态机的原理,处在某一状态去改变某些属性以达到目的.关于state如何使用的我这里就不说了,看看nokia的QML文档就知道state如何使用.我这里主要讲讲state的历史记忆效应