当前位置:首页 > 科技  > 软件

面试官:你能实现一个 JavaScript 模板引擎吗?

来源: 责编: 时间:2024-05-29 08:57:07 282观看
导读Hello,大家好,我是 Sunday。这个问题具体是这样的:请为字符串增加一个 render 方法,可以实现如下最终的打印。const template = '我是 ${name}, 年龄 ${age} 岁'const employee = { name: 'Sunday', age: 18}const rend

Hello,大家好,我是 Sunday。6jL28资讯网——每日最新资讯28at.com

这个问题具体是这样的:请为字符串增加一个 render 方法,可以实现如下最终的打印。6jL28资讯网——每日最新资讯28at.com

const template = '我是 ${name}, 年龄 ${age} 岁'const employee = {  name: 'Sunday',  age: 18}const renderStr = template.render(employee)// 输出成字符串console.log(renderStr) // '我是 Sunday, 年龄 18 岁'

乍一看,这不就是 模板引擎 吗?合着这是让我手写一个 `` 的简易版出来啊。6jL28资讯网——每日最新资讯28at.com

不过还好,既然是简易版那就并不复杂。一共有三种方式,咱们来看看吧!6jL28资讯网——每日最新资讯28at.com

01:利用正则表达式

使用正则表达式应该是大多数的同学第一时间想到的方案了。6jL28资讯网——每日最新资讯28at.com

只需要通过正则替换 ${name} 和 ${age} 就可以直接实现对应的功能。6jL28资讯网——每日最新资讯28at.com

String.prototype.render = function (obj) {  const template = this  const variableRegex = //$/{([^${}]+)/}/g  template.replace(variableRegex, ($0, variable) => {     // 打印对应的属性    console.log(variable)  })}const template = '我是 ${name}, 年龄 ${age} 岁'template.render()

通过以上代码我们可以直接拿到 ${name} 之中的属性,所以接下来咱们就只需要完成替换即可。6jL28资讯网——每日最新资讯28at.com

// 为 String 对象的原型添加一个名为 render 的方法String.prototype.render = function (obj) {  // 保存调用该方法的字符串实例  const template = this;  // 定义一个正则表达式,用于匹配 ${variableName} 格式的变量  const variableRegex = //$/{([^${}]+)/}/g;  // 定义一个函数,用于根据传入的对象获取变量的值  const getVariableValue = (variable) => {    // 将变量名按照 '.' 分隔成数组,例如 'user.name' 会分隔成 ['user', 'name']    variable = variable.split('.');    // 初始化 variableValue,使其指向传入的对象 obj    let variableValue = obj;    // 遍历分隔后的变量名数组,逐层获取嵌套属性的值    while (variable.length) {      // 取出数组的第一个元素,并获取对应的属性值      variableValue = variableValue[variable.shift()];    }    // 返回最终获取到的变量值    return variableValue;  };  // 使用 replace 方法替换模板字符串中的变量  // $0 是匹配到的整个字符串,例如 ${name}  // variable 是捕获组中的变量名,例如 name  const renderStr = template.replace(variableRegex, ($0, variable) => {    // 获取变量值并替换模板中的变量    return getVariableValue(variable);  });  // 返回替换后的字符串  return renderStr;};

02:使用 eval

eval() 函数会将传入的字符串当做 JavaScript 代码进行执行。6jL28资讯网——每日最新资讯28at.com

比如:6jL28资讯网——每日最新资讯28at.com

const employee = {  name: 'Sunday',  age: 18}const { name } = employeeconsole.log(name) // Sunday

这样的代码使用 eval 方法可以这么写:6jL28资讯网——每日最新资讯28at.com

const employee = {  name: 'Sunday',  age: 18}// 注意:必须是 vareval('var { name } = employee')console.log(name) // Sunday

这样的好处在于 可以根据 obj 的 key 动态的生成新的变量。6jL28资讯网——每日最新资讯28at.com

因此,就可以得到如下代码:6jL28资讯网——每日最新资讯28at.com

// 为 String 对象的原型添加一个名为 render 的方法String.prototype.render = function (obj) {  // 保存调用该方法的字符串实例  const template = this;  // 使用 eval 动态解构 obj 对象,将其属性名作为变量名,并赋值给这些变量  // 例如,obj = { name: 'Sunday', age: 18}  // 生成的代码类似于:var { name, age, job } = obj;  eval(`var {${Object.keys(obj).join(',')}} = obj`);  // 使用模板字符串替换变量,并生成最终的字符串  // 这里的 eval 用于解析和执行模板字符串,其中包含 obj 对象的属性值  // 例如,template = '我是 ${name}, 年龄 ${age} 岁'  // 生成的代码类似于:`我是 ${name}, 年龄 ${age} 岁`  const renderStr = eval('`' + template + '`');  // 返回替换后的字符串  return renderStr;}

03:with 关键字

with 语句扩展一个语句的作用域链6jL28资讯网——每日最新资讯28at.com

with 关键字属于被弃用的语法(但是 Vue3 的源码中依然使用到了 with),但是在这里依然可以实现对应的功能。6jL28资讯网——每日最新资讯28at.com

图片图片6jL28资讯网——每日最新资讯28at.com

我们可以通过以下示例来演示 with 的作用:6jL28资讯网——每日最新资讯28at.com

const employee = {  name: 'Sunday',  age: 18}with (employee) {  console.log(name, age)  // Sunday 18}

基于这个特性,使用 with 实现这个功能就非常简单了。6jL28资讯网——每日最新资讯28at.com

String.prototype.render = function (obj) {  with(obj) {     // this 实例。即:我是 ${name}, 年龄 ${age} 岁     // 两边加上 ` ` 即可利用 ES6 的模板运算符实现此功能    return eval('`' + this + '`')  }}

本文链接:http://www.28at.com/showinfo-26-91363-0.html面试官:你能实现一个 JavaScript 模板引擎吗?

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 记一次 .NET某工控WPF程序被人恶搞的卡死分析

下一篇: Python 对象的行为是怎么区分的?

标签:
  • 热门焦点
  • 对标苹果的灵动岛 华为带来实况窗功能

    继苹果的灵动岛之后,华为也在今天正式推出了“实况窗”功能。据今天鸿蒙OS 4.0的现场演示显示,华为的实况窗可以更高效的展现出实时通知,比如锁屏上就能看到外卖、打车、银行
  • Redmi Pad评测:红米充满野心的一次尝试

    从Note系列到K系列,从蓝牙耳机到笔记本电脑,红米不知不觉之间也已经形成了自己颇有竞争力的产品体系,在中端和次旗舰市场上甚至要比小米新机的表现来得更好,正所谓“大丈夫生居
  • K8S | Service服务发现

    一、背景在微服务架构中,这里以开发环境「Dev」为基础来描述,在K8S集群中通常会开放:路由网关、注册中心、配置中心等相关服务,可以被集群外部访问;图片对于测试「Tes」环境或者
  • 一年经验在二线城市面试后端的经验分享

    忠告这篇文章只适合2年内工作经验、甚至没有工作经验的朋友阅读。如果你是2年以上工作经验,请果断划走,对你没啥帮助~主人公这篇文章内容来自 「升职加薪」星球星友 的投稿,坐
  • .NET 程序的 GDI 句柄泄露的再反思

    一、背景1. 讲故事上个月我写过一篇 如何洞察 C# 程序的 GDI 句柄泄露 文章,当时用的是 GDIView + WinDbg 把问题搞定,前者用来定位泄露资源,后者用来定位泄露代码,后面有朋友反
  • 腾讯VS网易,最卷游戏暑期档,谁能笑到最后?

    作者:无锈钵来源:财经无忌7月16日晚,上海1862时尚艺术中心。伴随着幻象的精准命中,硕大的荧幕之上,比分被定格在了14:12,被寄予厚望的EDG战队以绝对的优势战胜了BLG战队,拿下了总决
  • 消费结构调整丨巨头低价博弈,拼多多还卷得动吗?

    来源:征探财经作者:陈香羽随着流量红利的退潮,电商的存量博弈越来越明显。曾经主攻中高端与品质的淘宝天猫、京东重拾“低价”口号。而过去与他们错位竞争的拼多多,靠
  • 世界人工智能大会国际日开幕式活动在世博展览馆开启

    30日上午,世界人工智能大会国际日开幕式活动在世博展览馆开启,聚集国际城市代表、重量级院士专家、国际创新企业代表,共同打造人工智能交流平台。上海市副市
  • Meta盲目扩张致超万人被裁,重金押注元宇宙而前景未明

    图片来源:图虫创意日前,Meta创始人兼CEO 马克·扎克伯发布公开信,宣布Meta计划裁员超11000人,占其员工总数13%。他公开承认了自己的预判失误:“不仅
Top