理解JavaScript关键字this

本文详细解释了JavaScript中this关键字的使用方式及其在不同上下文中的表现,包括对象方法调用、函数未绑定调用及事件处理器等情况,并提供了解决方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

原文:Understanding JavaScript's this keyword

    几天前,我写了一篇关于面向方面的JavaScript的文章,说明了如何利用包装器来替换给定的函数或方法。为了全面地理解这一概念,正确理解JavaScript对其关键字“this”的处理是很有帮助的。如何在JavaScript各种各样的上下文中确定this指的是什么,本文将给出一个简单概要。

    让我们从下面 Bob 的例子开始:

var Bob = {
  name: "Bob",
  greet: function() {
    alert( "Hi, my name is " + this.name );
  }
};

    在这里,我们创建了一个拥有两个成员的Object实例,一个简单的字符name,和一个包含对name引用的方法greet,当执行 Bob.greet()方法时,greet()函数用Bob作为this的对象。现在,让我们来看看同样的Bob.greet()在新的对象上执行会如何:

var Alice = {
  name: "Alice",
  greet: Bob.greet
};

    考虑下面哪一个才是Alice.greet()的输出呢?

   1.  "Hi, my name is Bob", or
   2. "Hi, my name is Alice" 

    如果你选择的是“2”,对了!Bob的greet函数中“this.name”的直到函数真正执行时才会被解析,所以,当Alice.greet()被调用时,函数greet()的this被设置为了Alice。但是,如果调用函数本身,而不是作为某个对象的方法,会发生什么呢?看看下面这个:

var unboundGreet = Bob.greet;
unboundGreet();

    你预测函数unboundGreet()会输出什么呢?答案是“"Hi, my name is”。这是因为当一个未绑定的函数被调用时,“this”被设置为了全局对象window,而window.name并没有定义,所以就产生空字符。

    下面的例子你将会看到隐含的未绑定会令调试很头痛:

setTimeout( Bob.greet, 1000 ); // Call Bob.greet 1 second from now

    开发者可能希望Bob.greet()会在一秒后调用,它绑定到了Bob上就会输出“Hi, my name is Bob”,但是这种预期并没有出现,相反,这个greet()方法只能称为未绑定,其输出结果同unboundGreet()是一样的。下面这个等效的代码或许能更清楚地说明其中的原由:

var unboundGreet = Bob.greet;
setTimeout( unboundGreet, 1000 );

    在这里,很明显,在执行时间开始(一秒后)时,函数greet()是一个未绑定的。当你把一个对象的方法设为一个事件处理器时,相同情况也会发生。看看下面的HTML:

<a id="clickme" href="">Greetings!</a>

    添加如下的JavaScript:

document.getElementById("clickme").onclick = Bob.greet;

    可能我们预期这会触发链接输出“Hi, my name is Bob”,但当点击它时,结果又和调用unboundGreet()是一样的。
    解决隐含未绑定的问题方法就是创建一个函数,用它来包装无范围的方法(the bound method)。大多数JavaScript库都包含了这样的助手,多数时候是通过向Function.prototype添加一个bind方法来完成的。
    这里就不用这种方式,下面用一个简单的例子说明如何建仓这样的包装器(wrapper)。

var bobsGreet = function() {
  Bob.greet();
};

setTimeout( bobsGreet, 1000 );

    或者更简单一点,

setTimeout( function(){

  Bob.greet();

}, 1000 );

    这就是本文的全部内容了。

1. 用户与身体信息管理模块 用户信息管理: 注册登录:支持手机号 / 邮箱注册,密码加密存储,提供第三方快捷登录(模拟) 个人资料:记录基本信息(姓名、年龄、性别、身高、体重、职业) 健康目标:用户设置目标(如 “减重 5kg”“增肌”“维持健康”)及期望周期 身体状态跟踪: 体重记录:定期录入体重数据,生成体重变化曲线(折线图) 身体指标:记录 BMI(自动计算)、体脂率(可选)、基础代谢率(根据身高体重估算) 健康状况:用户可填写特殊情况(如糖尿病、过敏食物、素食偏好),系统据此调整推荐 2. 膳食记录与食物数据库模块 食物数据库: 基础信息:包含常见食物(如米饭、鸡蛋、牛肉)的名称、类别(主食 / 肉类 / 蔬菜等)、每份重量 营养成分:记录每 100g 食物的热量(kcal)、蛋白质、脂肪、碳水化合物、维生素、矿物质含量 数据库维护:管理员可添加新食物、更新营养数据,支持按名称 / 类别检索 膳食记录功能: 快速记录:用户选择食物、输入食用量(克 / 份),系统自动计算摄入的营养成分 餐次分类:按早餐 / 午餐 / 晚餐 / 加餐分类记录,支持上传餐食照片(可选) 批量操作:提供常见套餐模板(如 “三明治 + 牛奶”),一键添加到记录 历史记录:按日期查看过往膳食记录,支持编辑 / 删除错误记录 3. 营养分析模块 每日营养摄入分析: 核心指标计算:统计当日摄入的总热量、蛋白质 / 脂肪 / 碳水化合物占比(按每日推荐量对比) 微量营养素分析:检查维生素(如维生素 C、钙、铁)的摄入是否达标 平衡评估:生成 “营养平衡度” 评分(0-100 分),指出摄入过剩或不足的营养素 趋势分析: 周 / 月营养趋势:用折线图展示近 7 天 / 30 天的热量、三大营养素摄入变化 对比分析:将实际摄入与推荐量对比(如 “蛋白质摄入仅达到推荐量的 70%”) 目标达成率:针对健
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值