当前位置: 首页 > article >正文

JS—call,apply,bind:1分钟掌握三者的区别

个人博客:haichenyi.com。感谢关注

一. 目录

  • 一–目录
  • 二–call
  • 三–apply
  • 四–bind
  • 五–三者对比

二. call

作用: 立即调用函数,显式指定this值,并逐个传递参数。
语法: func.call(thisArg, arg1, arg2, …)
特点:
  1.参数按顺序传递。
  2.适合参数数量固定的场景。
​示例:

const person = { name: 'Alice' };
function greet(greeting) {console.log(`${greeting}, ${this.name}!`);
}
greet.call(person, 'Hello'); // 输出:Hello, Alice!

三. apply

作用: 立即调用函数,显式指定this值,并以数组形式传递参数。
语法: func.apply(thisArg, [argsArray])
特点:
  1.参数以数组或类数组对象传递。
  2.适合参数数量不固定或需动态传递的场景。
​示例:

const numbers = [5, 6, 2, 3, 7];
const max = Math.max.apply(null, numbers); // ES5写法
console.log(max); // 输出:7// ES6可用扩展运算符替代:
const maxES6 = Math.max(...numbers);

四. bind

作用: 返回一个新函数,永久绑定this值和部分参数,不立即执行。
语法: func.bind(thisArg, arg1, arg2, …)
特点:
  1.参数可分阶段传递(绑定时传部分,调用时传剩余)。
  2.适合需要延迟执行或固定上下文的场景(如事件处理)。
​示例:

const car = { brand: 'Toyota' };
function showDetails(year, color) {console.log(`${this.brand} made in ${year}, color: ${color}`);
}const boundFunc = showDetails.bind(car, 2020);
boundFunc('red'); // 输出:Toyota made in 2020, color: red

五. 三者对比

对比表格

方法执行时机参数形式返回值应用场景
call立即执行参数列表 (arg1, …)原函数的返回值明确参数数量,快速调用的场景
apply立即执行参数数组([args])原函数的返回值参数动态或数量不固定的场景
bind返回新函数分阶段传递参数绑定后的新函数延迟执行或固定上下文

关键区别

  1. 执行方式:
    • call/apply:​立即执行函数。
    • bind:返回新函数,​需手动调用。
  2. 参数传递:
    • call:参数逐个传递(如 arg1, arg2)。
    • apply:参数以数组传递(如 [arg1, arg2])。
    • bind:可分阶段传递参数(绑定时传部分,调用时传剩余)。
  3. this绑定
    • bind返回的函数永久绑定this,后续无法修改。
    • call/apply仅对本次调用生效。

  这里我还想说一点,我在鸿蒙开发中遇到的一个问题。关于this指针的问题。

  背景是这样的: 我之前开发Android的时候与H5通信,我写完H5,写Android,通过JS拦截prompt实现通信。简单的来说,就是H5端触发这个方法之后,通过传参告诉我需要执行哪个方法。我知道方法名字,Android里面可以通过反射去获取了之后执行。鸿蒙里面不行啊,没有反射这个概念。鸿蒙是ArkTS是TS的超类,TS是JS的扩展。

  问题就出在执行的方法的这里:

  prompt(param: ESObject): string {let result = JSON.parse(param) as Record<string, string>let callbackId = result.callbackIdlet funcName = result.funcNamelet funcParam = result.funcParamDfhLogUtils.info("DfhWebView", `callbackId:${callbackId},funcName:${funcName},funcParam:${funcParam}`)if (this.iDfhJsFunction == null) {//请先调用initJsFunction初始化iDfhJsFunctionreturn ""}let func = this.iDfhJsFunction.getRegisterJsFunction(funcName)if (!func) {//没有js方法,查看方法名称有没有写错。查找是否已经注册该方法:registerJsFunctionreturn ""}if (!callbackId) {//同步方法return func(param)} else {func(param)//异步方法return ""}}

  我在项目启动的时候,通过容器存储我有哪些方法,说白了就是一个map,key是方法名,value是方法。H5给我方法名的时候,我通过map去获取,获取到了就说明有这个方法,没获取到就说明没有。获取到了之后,就直接加小括号执行就可以了。

  我第一个版本这些方法全都写的静态,静态方法会有一个问题,那就是我如果启动了两个webview,只有一个对象去通信,理论上是有问题。因为,我要是存在局部变量,就会乱。然后,后面遇到了这个问题,我就改成了非静态的。

  静态的变量通过类。对象名去调用。非静态的就是通过this。对象去使用变量。这里就有问题,静态的我就不写了。

  /*** 显示加载对话框* @param param*/showLoading(param: ESObject): void  {this.dfhLoadingBean.showLoading()}

我最开始执行的位置,是没有对象。方法的。而是,直接方法()执行的。这样,就会在执行方法的时候,this指向的不是我实际的对象。
我用bind解决了这个问题。

this.mDfhJsFunctionMap.set("showLoading", this.showLoading.bind(this))

这里还遇到了另一个小插曲,我注册这个容器的方法是在父类中写的,我直接在父类的构造方法里面去初始化容器。而我的这些实现类都是在子类里面。这就有另一个问题,子类构造方法还没有初始化,这里的方法this的指向就不是子类。必须要在子类构造方法初始化了之后,在初始化这个容器。

  constructor(mController: webview.WebviewController) {super(mController)//要放在子类中调用,不要放在父类中调用。父类调用的时候this指向会有问题this.registerJsFunction();}

以上,问题都解决了。但是,ArkTS不推荐使用call,apply,bind方法。那怎么办呢?箭头函数呀。

  /*** 显示加载对话框* @param param*/showLoading = (param: ESObject): void => {this.dfhLoadingBean.showLoading()}

把方法全都写成箭头函数,就解决了这个问题。
this指向的问题,现在基本上都是能用箭头函数就用箭头函数。

相关文章:

JS—call,apply,bind:1分钟掌握三者的区别

个人博客&#xff1a;haichenyi.com。感谢关注 一. 目录 一–目录二–call三–apply四–bind五–三者对比 二. call 作用&#xff1a; 立即调用函数&#xff0c;显式指定this值&#xff0c;并逐个传递参数。 语法&#xff1a; func.call(thisArg, arg1, arg2, …) 特点&…...

Linux TTY设备汇总

目录 1. ‌tty(终端设备统称) 2. ‌ptm(伪终端主设备)与pts(伪终端从设备) 3. ‌ttys(串行端口终端) 4. ‌ttyACM(USB CDC ACM设备) 5. ‌ttyGS(USB Gadget Serial设备) 主要联系‌ ‌典型应用场景‌ TTY_CORE: drivers/tty/tty_io.c:tty_register_driver…...

WPF 与 C# 开发深度剖析

一、引言 在当今的软件开发领域&#xff0c;Windows 平台依旧占据着重要的地位。而 WPF&#xff08;Windows Presentation Foundation&#xff09;作为微软推出的一款强大的用户界面&#xff08;UI&#xff09;框架&#xff0c;为开发者提供了丰富的功能和灵活的设计方式&…...

好消息!软航文档控件(NTKO WebOffice)在Chrome 133版本上提示扩展已停用的解决方案

软航文档控件现有版本依赖Manifest V2扩展技术支持才能正常运行&#xff0c;然而这个扩展技术到2025年6月在Chrome高版本上就彻底不支持了&#xff0c;现在Chrome 133开始的版本已经开始弹出警告&#xff0c;必须手工开启扩展支持才能正常运行。那么如何解决这个技术难题呢&…...

通过仿真确定抗积分饱和策略的最佳系数

通过仿真确定抗积分饱和策略的最佳系数&#xff08;如PID参数 ( K_p, K_i, K_d ) 以及抗饱和参数 ( K_{\text{back}} )、积分限幅值等&#xff09;是一个系统化的过程。以下是具体步骤和示例&#xff1a; — 1. 建立仿真模型 1.1 模型组成 被控对象&#xff1a;例如电机、温…...

消息队列(Kafka及RocketMQ等对比联系)

目录 消息队列 一、为什么使用消息队列&#xff1f;消息队列有什么优点/缺点&#xff1f;介绍下Kafka、ActiveMQ、RabbitMQ、RocketMQ有什么优点缺点&#xff0c;如何取舍&#xff1f; 1.公司业务场景是什么&#xff0c;这个业务场景有什么挑战&#xff0c;如果不用MQ有什么麻…...

GitHub开源的容器管理面板-Dpanel

dpanel Docker安装部署二进制部署 GitHub官网 一块轻量化docker可视化管理面板&#xff0c;由国人开发&#xff0c;个人觉得是比较好用的&#xff0c;功能都很齐全&#xff0c;并且可以通过修改源码&#xff0c;自定义前端样式等。 Docker安装部署 官网 部署环境&#xff1…...

【HarmonyOS Next】三天撸一个BLE调试精灵

【HarmonyOS Next】三天撸一个BLE调试精灵 一、功能介绍 BLE调试精灵APP属于工具类APP&#xff0c;在用户使用的过程中&#xff0c;负责调试BLE设备从机端&#xff0c;比如蓝牙耳机、低功耗设备、带有BLE的空调等设备&#xff0c;可以在页面中清晰看到设备的厂商&#xff0c;…...

java 批量下载doc\excle\pdf

指定图片集合 下载到指定文件夹 import java.io.*; import java.net.HttpURLConnection; import java.net.URL; import java.util.Arrays; import java.util.List;public class OfficeFileDownloader {/*** 需要下载的Office文档URL列表*/private static final List<Strin…...

软件性能效率测试工具有哪些?专业第三方软件检测机构推荐

在软件开发的新时代&#xff0c;软件性能效率测试已经成为每个企业不可或缺的一部分。无论是在竞争激烈的市场中&#xff0c;还是在追求卓越用户体验的过程中&#xff0c;都需要进行有效的性能测试。 一、软件性能效率测试的目标   1、响应时间&#xff1a;确保用户请求的响…...

使用flask_restful快速构建接口

Flask-RESTful 是一个用于快速构建 RESTful API 的 Flask 扩展。它简化了创建、管理和文档化 REST API 的过程。利用 Flask-RESTful&#xff0c;你可以更容易地将你的 Flask 应用程序组织成 RESTful 原则的风格 安装包 pip install flask_restful 快速构建接口 from flask im…...

centos 7 部署FTP 服务用shell 搭建脚本,使用时稍微修改自己所需需求

#!/bin/bash # 检查是否为 root 用户 if [ "$(id -u)" ! "0" ]; then echo "此脚本需要以 root 用户身份运行。" exit 1 fi # 安装 vsftpd yum install vsftpd -y # 备份原始配置文件 cp /etc/vsftpd/vsftpd.conf /etc/vsftpd/vsftpd…...

Hadoop集群搭建(hdfs、yarn)

Hadoop 是 Apache 软件基金会旗下的一个开源项目&#xff0c;是用于处理大数据的分布式系统基础架构&#xff0c;被广泛应用于大数据存储、处理和分析等场景。 一、核心组件 1、Hadoop 分布式文件系统&#xff08;HDFS&#xff09; 具有高容错性&#xff0c;能在低成本硬件上…...

PyArrow 和 Parquet 的组合通过内存优化和高效存储,显著提升大数据处理的性能

PyArrow 和 Apache Parquet 是处理大规模数据的高效工具组合&#xff0c;它们结合了内存优化和列式存储的优势&#xff0c;适用于多种大数据场景。以下是对两者的详细解析及其协同工作的方式&#xff1a; ### 1. **PyArrow 的核心作用** - **Apache Arrow 实现**&#xff1…...

Oracle数据库服务器地址变更与监听配置修改完整指南

一、前言 在企业IT运维中&#xff0c;Oracle数据库服务器地址变更是常见的运维操作。本文将详细介绍如何安全、高效地完成Oracle数据库服务器地址变更及相关的监听配置修改工作&#xff0c;确保数据库服务在迁移后能够正常运行。 二、准备工作 1. 环境检查 确认新旧服务器I…...

2025.03.23【前沿工具】| CellPhoneDB:基因网络分析与可视化的利器

文章目录 1. CellPhoneDB工具简介2. CellPhoneDB的安装方法3. CellPhoneDB常用命令 1. CellPhoneDB工具简介 在单细胞生物学的迅猛发展中&#xff0c;理解细胞间的通讯机制对于揭示组织功能和疾病状态至关重要。CellPhoneDB工具&#xff0c;作为一个专门设计用来分析单细胞转录…...

Keepalived 实现高可用方案

Keepalived简介 ‌Keepalived‌ 是一个基于 ‌VRRP&#xff08;Virtual Router Redundancy Protocol&#xff09;协议‌的高可用性解决方案&#xff0c;主要用于实现‌服务故障自动切换&#xff08;Failover&#xff09;和负载均衡‌。通过管理虚拟 IP&#xff08;VIP&#xf…...

如何使用DeepSeek编写测试用例?

一、DeepSeek在测试用例设计中的定位 DeepSeek作为AI工具,并非直接替代测试设计,而是通过以下方式提升效率: 快速生成基础用例框架(等价类、边界值等) 智能补充易遗漏场景(如特殊字符、异常流) 自动化脚本片段生成(Python/pytest/JUnit等) 测试数据构造建议(符合业务…...

python暴力破解html表单

import requests import time# 目标URL url "http://192.168.3.101/pikachu/vul/burteforce/bf_form.php" # 请替换为实际的目标URL# 已知的用户名 username "admin"# 密码字典文件路径 password_file "passwords.txt"# 伪造请求头&#xff…...

医学图像分割数据集肺分割数据labelme格式6299张2类别

数据集格式&#xff1a;labelme格式(不包含mask文件&#xff0c;仅仅包含jpg图片和对应的json文件) 图像分辨率&#xff1a;1024x1024 图片数量(jpg文件个数)&#xff1a;6299 标注数量(json文件个数)&#xff1a;6299 标注类别数&#xff1a;2 标注类别名称:["leftl…...

C语言复习笔记--函数递归

在学习了函数之后,函数递归是我们必然会接触到的课题,下面就让我们看下函数递归相关的知识. 递归是什么&#xff1f; 递归这个词看着就不那么好理解,那么什么是递归呢?递归其实是⼀种解决问题的⽅法,在C语⾔中,递归就是函数自己调用自己. 写⼀个史上最简单的C语⾔递归代码: …...

Day39 | 724. 寻找数组的中心下标、34. 在排序数组中查找元素的第一个和最后一个位置、922. 按奇偶排序数组 II、35. 搜索插入位置

724. 寻找数组的中心下标 题目链接&#xff1a;724. 寻找数组的中心下标 - 力扣&#xff08;LeetCode&#xff09; 题目难度&#xff1a;简单 代码&#xff1a; class Solution {public int pivotIndex(int[] nums) {int sum0;for(int i0;i<nums.length;i){sumnums[i];}…...

husky的简介以及如果想要放飞自我的解决方案

husky 是一个 Git Hooks 管理工具&#xff0c;它的主要作用是 在 Git 提交&#xff08;commit&#xff09;、推送&#xff08;push&#xff09;等操作时执行自定义脚本&#xff0c;比如代码检查&#xff08;Lint&#xff09;、单元测试&#xff08;Test&#xff09;、格式化代码…...

怎么查看linux是Ubuntu还是centos

要确定你的Linux系统是基于Ubuntu还是CentOS&#xff0c;可以通过几种不同的方法来进行判断。下面是一些常用的方法&#xff1a; 要快速判断 Linux 系统是 Ubuntu 还是 CentOS&#xff0c;可通过以下方法综合验证&#xff1a; 一、查看系统信息文件 1. /etc/os-release 文件…...

侯捷 C++ 课程学习笔记:现代 C++ 中的移动语义与完美转发深度解析

1. 前言&#xff1a;为什么我们需要移动语义&#xff1f; 在侯捷老师的《C11/14/17 新特性详解》课程中&#xff0c;移动语义&#xff08;Move Semantics&#xff09;被称作"C近十年来最重要的革新"。传统C中饱受诟病的深拷贝性能问题&#xff0c;在现代C中通过移动语…...

微服务中的服务发现与注册中心

在微服务架构中&#xff0c;服务实例的数量可能随着流量负载自动扩展或缩减&#xff0c;因此服务之间如何高效地进行通信成为一个重要问题。本篇博客将介绍服务发现的概念&#xff0c;并结合 Consul 和 自定义注册中心 进行实践&#xff0c;帮助开发者在微服务架构下高效管理服…...

23种设计模式-结构型模式-适配器

文章目录 简介场景问题解决方案建立中间转换层关键收益 总结 简介 使接口不兼容的类实现协同工作&#xff0c;通过引入中间层实现客户端接口和服务端接口的兼容。典型场景比如整合第三方类库或遗留系统时保持代码兼容。 场景 假设你正在开发一个股票监控程序。这个程序会下…...

美亚科技业绩波动明显:现金流为负,四起未决诉讼涉金额1700万

《港湾商业观察》施子夫 近期&#xff0c;广东美亚旅游科技集团股份有限公司&#xff08;以下简称&#xff0c;美亚科技&#xff09;披露第二轮审核问询函的回复。从两轮问询函监管层提出的问题来看&#xff0c;有关美亚科技业绩增长的合理性、募投项目的必要性及合理性、经营…...

PyTorch 深度学习实战(21):元强化学习与 MAML 算法

一、元强化学习原理 1. 元学习核心思想 元强化学习&#xff08;Meta-RL&#xff09;旨在让智能体快速适应新任务&#xff0c;其核心是通过任务分布学习共享知识。与传统强化学习的区别在于&#xff1a; 对比维度传统强化学习元强化学习目标解决单一任务快速适应任务分布中的…...

23中设计模式-迭代器(Iterator)设计模式

迭代器设计模式 &#x1f6a9;什么是迭代器设计模式&#xff1f;&#x1f6a9;迭代器设计模式的特点&#x1f6a9;迭代器设计模式的结构&#x1f6a9;迭代器设计模式的优缺点&#x1f6a9;迭代器设计模式的Java实现&#x1f6a9;代码总结&#x1f6a9;总结 &#x1f6a9;什么是…...