草庐IT

4-函数进阶:理解函数调用

cxyfxr 2023-03-28 原文

1. 使用隐式函数参数

函数中的两个隐含的参数:this、arguments
这两个参数可以像函数体内显示声明的参数一样被正常访问。

  • this
    • 表示被调用函数的上下文对象。
  • arguments
    • 表示函数调用过程中传递的所有实参

arguments

  • arguments是函数的所有实参。
  • arguments有数组的特性,但不是真正数组,不可用数组方法,(类数组?NodeList属于类数组)
    • length属性,表示实参的具体个数。
    • arguments[0],可以用数组下标获取实参的值
    • 转换为数组
      • [...arguments]
      • Array.from(arguments)
  • 箭头函数中没有arguments,可以用...params代替
    var s=(...args)=>{ 
        console.log(args)
    } 
    s(1,2,3) // [1, 2, 3]
  • arguments作为函数参数的别名
    • 非严格模式
      • 可以作为行参params的别名,修改行参值
      function init(params){
          arguments[0] = 1;
      }
      init(3,4);  //arguments是一个非数组的结构 3,4
      //params原本为3,arguments[0]改为1,则最后params为1
      
    • 严格模式
      • arguments[index]会打印出对应的参数和行参,
      • 修改arguments[index]不会改变行参对应的params值。打印arguments[index]倒是会改变。
      • 只有params自己可以改变自己。
      • use strict开启严格模式

函数上下文:this

代表函数调用相关联的对象,称之为函数上下文。

  • this的指向
    • 定义函数的方式
    • 定义函数的位置
    • 函数的调用方式!!!!!!!!!
      • 直接调用
      • 对象方法
      • 构造函数
      • apply和call方法

1. 作为函数直接调用

  • 如果一个函数没有作为方法,构造函数、apply、call调用的话,我们称之为函数直接调用。
  • 直接调用的this指向
    • 严格模式下,直接调用的this为unfined
    • 非严格模式下,this => window
  • 直接调用的表现形式
    //函数声明
    function a(){}; 
    a();
    //函数定义
    var b=funcgtion(){}; 
    b();
    //函数表达式
    (function (){} ) ()
    //立即执行函数
    

2. 作为方法调用

  • 作为某个对象的方法|属性被调用。
  • this指向:
    • 当对象是字面量创建,且方法函数为普通函数
      • 事件回调中
        • 方法调用时this应该是当前对象,但在事件回调中,回调函数的this为触发当前事件的元素,所以该回调函数中的方法调用的普通函数被传入隐式this,改变为当前元素。而非对象
          • 此时可以用apply/call/bind来改变回调函数中的this指向
      • 非回调情况下
        • 一般this指向该对象。
    • 当对象是字面量创建,且方法函数为箭头函数时
      • 箭头函数本身没有this,依靠上下文环境
  • 方法调用的表现形式
    function afun(){
        return this;//window
    }; 
    afun();//window
    var obj = {
       b : afun
    }
    obj.b();//obj
    //obj的属性引用了全局变量的函数体,但是是通过方法掉用的,所以this为该对象obj,所以可以在对象的属性中引用函数。
    //所以 函数不看定义,看调用!
    

3. 作为构造函数调用

  • 目的是创建空对象,将其作为this对象传递给构造函数,初始化对象后作为构造函数的返回值。也称构造函数实例化对象。
  • this指向(构造函数实例化对象):
    • 若构造函数显示返回了一个原始值,实例化对象后,返回该对象。
    • 若构造函数显示返回了一个对象,实例化对象后,返回return的对象,当前对象的this失效
function a(){
    return 'a'
}
const ao=new a()
ao //a {}

function v(){
    return {a:1}
}
new v() //{a: 1}

function v(){
    return {
            a:1,
            b:function(){
                    console.log(this)
            }}
}
const obj=new v() 
obj // {a: 1, b: ƒ}
obj.b() // {a: 1, b: ƒ}
  • 构造函数调用的表现形式
    function afun(){
        return 1;//window
    }; 
    afun();//1 构造函数作为普通函数调用,正常返回1,没什么意义
    var obj = new afun(); //obj{} 构造函数实例化对象,返回该对象
    //注意:
    //构造函数显式的返回原始值,对实例化对象没有影响。
    //但显式的返回对象,实例化对象后的变量,返回是return的对象。
    //构造函数只有在实例化对象后,才有意义,将构造函数直接调用没意义。
    
    //构造函数实例化对象时发生了什么:
    //创建空对象
    //this指向为空对象
    //对象实例化并返回,除了上述的构造函数显式的返回了一个对象的情况外。
    
    
  • 构造函数的命名约定
    • 通常以描述所构造的对象的名词命名
    • 大写字母开头,例:Ninja;
    • 但这只是约定。

4. 使用apply/call调用

显式的修改函数的上下文,将this赋给指定对象。我们可以使每个函数上用apply和call完成。
二者唯一区别是,参数列表的形式。

  • apply方法调用
    • 传参2个:function.apply(指定this的对象,传参数组)
  • call调用
    • 传参n个:function.call(指定this的对象,params1,params2,params2)
p75例子,构造函数实例化对象后,对象方法在调用时,修改不了对象的属性。
原因是对象方法是在[事件回调函数中被调用的,this被改变为触发事件的元素]。(对象方法是普通函数定义)

注意:p80实现foreach迭代方法展示如何设置函数上下文例子

【构造函数实例化对象】后,对象方法若是箭头函数定义。
箭头函数没有单独的this,在定义时与上下文this一致。
故,构造函数中的箭头函数在被实例化后,this被定义为实例化后的对象。
无论怎么调用箭头函数的这个方法,都不会被传入和改变this对象。

注意区分,【对象方法】而非【构造函数实例化对象】,中定义箭头函数时,此时的this为window

5. 解决函数this上下文的问题

  • 使用箭头函数
    • 箭头函数没有单独的this,在定义时与上下文this一致。
    • 调用箭头函数时,不会隐式传入this
    • 箭头函数的this
      • 对象中,window
      • 实例化的对象,是对象。且不会隐式传入this
  • 使用bind
    • 创建并返回一个与原函数相同的新函数,并指定新函数的this上下文。
      • button.click.bind(button)将button.click方法copy一份,将this改为button对象。
        `

回到本章的问题,为一个或多个按钮设置是否单击的状态。【单击意味着事件回调,会隐式的传入this】

  • 一个 =>【对象】
    • 对象的方法要用普通函数,而不是箭头函数,箭头函数的this不知道是否可以被改变,因为他不会隐式传入this
    • 对象的方法用普通函数,会被事件回调更改this指向,可以用apply、call、bind改变this指向。
  • 多个 =>【构造函数】
    • 箭头函数,因为实例化时被定义好了this,并且不会被隐式传入this,可以规避被事件回调改变this的问题。
    • 普通函数,可以用apply、call、bind改变this指向。
      • apply、call改变原来函数的this,bind创建一个与原来一样的函数,并指定this对象

有关4-函数进阶:理解函数调用的更多相关文章

  1. ruby - 在没有 sass 引擎的情况下使用 sass 颜色函数 - 2

    我想在一个没有Sass引擎的类中使用Sass颜色函数。我已经在项目中使用了sassgem,所以我认为搭载会像以下一样简单:classRectangleincludeSass::Script::FunctionsdefcolorSass::Script::Color.new([0x82,0x39,0x06])enddefrender#hamlengineexecutedwithcontextofself#sothatwithintemlateicouldcall#%stop{offset:'0%',stop:{color:lighten(color)}}endend更新:参见上面的#re

  2. ruby-on-rails - 在 ruby​​ 中使用 gsub 函数替换单词 - 2

    我正在尝试用ruby​​中的gsub函数替换字符串中的某些单词,但有时效果很好,在某些情况下会出现此错误?这种格式有什么问题吗NoMethodError(undefinedmethod`gsub!'fornil:NilClass):模型.rbclassTest"replacethisID1",WAY=>"replacethisID2andID3",DELTA=>"replacethisID4"}end另一个模型.rbclassCheck 最佳答案 啊,我找到了!gsub!是一个非常奇怪的方法。首先,它替换了字符串,所以它实际上修改了

  3. 使用 ACL 调用 upload_file 时出现 Ruby S3 "Access Denied"错误 - 2

    我正在尝试编写一个将文件上传到AWS并公开该文件的Ruby脚本。我做了以下事情:s3=Aws::S3::Resource.new(credentials:Aws::Credentials.new(KEY,SECRET),region:'us-west-2')obj=s3.bucket('stg-db').object('key')obj.upload_file(filename)这似乎工作正常,除了该文件不是公开可用的,而且我无法获得它的公共(public)URL。但是当我登录到S3时,我可以正常查看我的文件。为了使其公开可用,我将最后一行更改为obj.upload_file(file

  4. ruby - 在 Ruby 中有条件地定义函数 - 2

    我有一些代码在几个不同的位置之一运行:作为具有调试输出的命令行工具,作为不接受任何输出的更大程序的一部分,以及在Rails环境中。有时我需要根据代码的位置对代码进行细微的更改,我意识到以下样式似乎可行:print"Testingnestedfunctionsdefined\n"CLI=trueifCLIdeftest_printprint"CommandLineVersion\n"endelsedeftest_printprint"ReleaseVersion\n"endendtest_print()这导致:TestingnestedfunctionsdefinedCommandLin

  5. c# - 如何在 ruby​​ 中调用 C# dll? - 2

    如何在ruby​​中调用C#dll? 最佳答案 我能想到几种可能性:为您的DLL编写(或找人编写)一个COM包装器,如果它还没有,则使用Ruby的WIN32OLE库来调用它;看看RubyCLR,其中一位作者是JohnLam,他继续在Microsoft从事IronRuby方面的工作。(估计不会再维护了,可能不支持.Net2.0以上的版本);正如其他地方已经提到的,看看使用IronRuby,如果这是您的技术选择。有一个主题是here.请注意,最后一篇文章实际上来自JohnLam(看起来像是2009年3月),他似乎很自在地断言RubyCL

  6. java - 从 JRuby 调用 Java 类的问题 - 2

    我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www

  7. ruby - 调用其他方法的 TDD 方法的正确方法 - 2

    我需要一些关于TDD概念的帮助。假设我有以下代码defexecute(command)casecommandwhen"c"create_new_characterwhen"i"display_inventoryendenddefcreate_new_character#dostufftocreatenewcharacterenddefdisplay_inventory#dostufftodisplayinventoryend现在我不确定要为什么编写单元测试。如果我为execute方法编写单元测试,那不是几乎涵盖了我对create_new_character和display_invent

  8. ruby - 在 Ruby 中按名称传递函数 - 2

    如何在Ruby中按名称传递函数?(我使用Ruby才几个小时,所以我还在想办法。)nums=[1,2,3,4]#Thisworks,butismoreverbosethanI'dlikenums.eachdo|i|putsiend#InJS,Icouldjustdosomethinglike:#nums.forEach(console.log)#InF#,itwouldbesomethinglike:#List.iternums(printf"%A")#InRuby,IwishIcoulddosomethinglike:nums.eachputs在Ruby中能不能做到类似的简洁?我可以只

  9. 【鸿蒙应用开发系列】- 获取系统设备信息以及版本API兼容调用方式 - 2

    在应用开发中,有时候我们需要获取系统的设备信息,用于数据上报和行为分析。那在鸿蒙系统中,我们应该怎么去获取设备的系统信息呢,比如说获取手机的系统版本号、手机的制造商、手机型号等数据。1、获取方式这里分为两种情况,一种是设备信息的获取,一种是系统信息的获取。1.1、获取设备信息获取设备信息,鸿蒙的SDK包为我们提供了DeviceInfo类,通过该类的一些静态方法,可以获取设备信息,DeviceInfo类的包路径为:ohos.system.DeviceInfo.具体的方法如下:ModifierandTypeMethodDescriptionstatic StringgetAbiList​()Obt

  10. C51单片机——实现用独立按键控制LED亮灭(调用函数篇) - 2

    说在前面这部分我本来是合为一篇来写的,因为目的是一样的,都是通过独立按键来控制LED闪灭本质上是起到开关的作用,即调用函数和中断函数。但是写一篇太累了,我还是决定分为两篇写,这篇是调用函数篇。在本篇中你主要看到这些东西!!!1.调用函数的方法(主要讲语法和格式)2.独立按键如何控制LED亮灭3.程序中的一些细节(软件消抖等)1.调用函数的方法思路还是比较清晰地,就是通过按下按键来控制LED闪灭,即每按下一次,LED取反一次。重要的是,把按键与LED联系在一起。我打算用K1来作为开关,看了一下开发板原理图,K1连接的是单片机的P31口,当按下K1时,P31是与GND相连的,也就是说,当我按下去时

随机推荐