zoukankan      html  css  js  c++  java
  • js原始数据类型和引用数据类型=>callback数据传输原理

    摘要:js的数据类型有种划分方式为 原始数据类型和 引用数据类型。

    原始数据类型 存储在栈(stack)中的简单数据段,也就是说,它们的值直接存储在变量访问的位置。栈区包括了 变量的标识符和变量的值。

     引用数据类型 存储在堆(heap)中的对象,也就是说,存储在变量处的值是一个指针(point),指向存储对象的内存处。放在栈空间中的值是该对象存储在堆中的地址。

    存储在堆和栈中的原始值和引用值

    按值传递 VS. 按引用传递

    按值传递(call by value)是最常用的求值策略:函数的形参是被调用时所传实参的副本。修改形参的值并不会影响实参。

    按引用传递(call by reference)时,函数的形参接收实参的隐式引用,而不再是副本。这意味着函数形参的值如果被修改,实参也会被修改。同时两者指向相同的值。

    按引用传递会使函数调用的追踪更加困难,有时也会引起一些微妙的BUG。

    按值传递由于每次都需要克隆副本,对一些复杂类型,性能较低。两种传值方式都有各自的问题。

    demo实例

    typeof 运算符,返回一个字符串,指示未计算的操作数的类型

        var person,name;
        person = 'kn';
    
        name=person;
        person='黑白';
        console.log(person,name,typeof person)//黑白 kn string

    person的改变没有改变name,说明 string 是按值传递的。赋值时创建一块新的内存空间

    1、基本类型

    • Boolean.  布尔值,true 和 false.
    • null. 一个表明 null 值的特殊关键字。 JavaScript 是大小写敏感的,因此 null 与 NullNULL或其他变量完全不同。
    • undefined.  变量未定义时的属性。
    • Number.  表示数字,例如: 42 或者 3.14159。
    • String.  表示字符串,例如:"Howdy"
    • Symbol ( 在 ECMAScript 6 中新添加的类型).。一种数据类型,它的实例是唯一且不可改变的。

    不可变(immutable)性质

    基本类型是不可变的(immutable),只有对象是可变的(mutable). 有时我们会尝试“改变”字符串的内容,但在JS中,任何看似对string值的”修改”操作,实际都是创建新的string值。任何方法都无法改变一个基本类型的值,  

    1     var str = "abc";
    2     str[0]; // "a"
    3     str[0] = "d";
    4     console.log(str); //abc
    5 
    6     var name = 'jozo';
    7     var upName=name.toUpperCase(); 
    8     console.log(upName,name); // 输出 'JOZO' 'jozo'

    方法操作无法改变一个基本类型的值

    1     var person = 'kn';
    2     person.age = 24;
    3     person.method = function(){};
    4 
    5     console.log(person.age); // undefined
    6     console.log(person.method); // undefined

    上面代码可知,我们不能给基本类型添加属性和方法

    2.引用类型

    引用类型的值是可变的

    
    
    1     var obj = {x : 0};
    2     obj.x = 100;
    3     var o = obj;
    4     o.x = 1;
    5     console.log(obj.x)// 1, 被修改
    6     o = {x:100};  //等同于重新赋值,重新开辟内存,不是修改
    7     console.log(JSON.stringify(obj),JSON.stringify(o))//{"x":1} {"x":100}
    8     obj.x; // 1, 不会因o = {"x":100}改变
    
    

    引用类型的值是同时保存在栈内存和堆内存中的对象

    javascript和其他语言不同,其不允许直接访问内存中的位置,也就是说不能直接操作对象的内存空间,那我们操作啥呢? 实际上,是操作对象的引用,
    所以引用类型的值是按引用访问的。
    准确地说,引用类型的存储需要内存的栈区和堆区(堆区是指内存里的堆内存)共同完成,栈区内存保存变量标识符和指向堆内存中该对象的指针,
    也可以说是该对象在堆内存的地址。
    假如有以下几个对象:

    var person1 = {name:'jozo'};
    var person2 = {name:'xiaom'};
    var person3 = {name:'xiaoq'};

    则这三个对象的在内存中保存的情况如下图:

        var person1 = {};
        var person2 = {};
        console.log(person1 == person2); // false

    引用类型时按引用访问的,换句话说就是比较两个对象的堆内存中的地址是否相同,那很明显,person1和person2在堆内存中地址是不同的。

    两种数据类型在实参和形参中的区别

    首先要明白什么实参什么是形参。

    实参:可以是常量、变量、表达式、函数等, 无论实参是何种类型的量,在进行函数调用时,它们都必须具有确定的值, 以便把这些值传送给形参。 因此应预先用赋值,输入等办法使实参获得确定值。
    形参:全称为“形式参数”是在定义函数名和函数体的时候使用的参数,目的是用来接收调用该函数时传递的参数。
    形参的作用是实现主调函数与被调函数之间的联系,通常将函数所处理的数据,影响函数功能的因素或者函数处理的结果作为形参。
    function addNum(param)  //相对于addNum方法来说 param是形参
    { 
        param+=10; 
         return param; 
    } 
    var num=10;  
    var result=addNum(num); //相对于addNum方法来说 num是实参
    console.log(num);  //10  
    console.log(result);//20

    上面的例子中,当将 num作为实参传入方法 addNum是 param作为形参接受 并在方法体内使用,且num在全局中没用改变,但当实参为引用类型时

    1 function fun(param)  //相对于fun方法来说 param是形参
    2 { 
    3     param[0]=99; 
    4      return param; 
    5 } 
    6 var num=[10];  
    7 var result=fun(num); //相对于fun方法来说 num是实参
    8 console.log(num[0]);  //99 
    9 console.log(result);//[99]

    在方法体内改变 形参将同时改变实参。这个是其他语言中不会存在的,例如php,需要操作引用时,使用 &地址符 在形参的时候声明

     1 <?php
     2 function fun($param)  //相对于fun方法来说 param是形参   function fun(&$param){}
     3 { 
     4     $param[0]=99; 
     5       // return $;
     6 } 
     7 $num=[10];  
     8 fun($num); //相对于fun方法来说 num是实参
     9 var_dump($num);
    10 /*
    11 array (size=1)
    12   0 => int 10
    13   */
    14 
    15 ?>

     思考: 正因为js中 function的形参若为引用类型时,可以影响实参! 推断 callback中参数原理

     1     function fun(data,callback){
     2         var json=[1,2,3];
     3         callback(json)
     4     }
     5 
     6     var data=[];
     7     fun(data,function(result){
     8         data=result;
     9     })
    10     console.log(data)//[1, 2, 3]

    如上例子 在回调函数中修改了 变量 data

    知识点1、 function 是一种数据类型,可以当做参数传递 2、数组是引用类型 3、引用类型的形参会影响实参

     1 <body>
     2     <button onclick='log()'>ajax</button>
     3 </body>
     4 <script>
     5     function fun(data,callback){
     6         setTimeout(function(){
     7             var json=[1,2,3];
     8             callback(json)
     9         },4000)
    10     
    11     }
    12 
    13     var data=[];
    14     fun(data,function(res){
    15         data=res;
    16     })
    17     console.log(data)//[]
    18     function log(){
    19         console.log(data)//[1, 2, 3]  4秒后输出 
    20     }
    21 
    22 </script>

    使用 setTimeout 模拟ajax请求! 

    参考地址: http://www.w3school.com.cn/js/pro_js_value.asp 

    http://www.qdfuns.com/notes/17660/7f82003c5ce92d39d19d6be0403f3f3b.html

     
  • 相关阅读:
    java之集合Collection详解之3
    委托的高级使用
    委托的一般使用
    泛型(Generic)委托
    泛型(Generic)方法(函数,算法)
    泛型(Generic)接口
    泛型(Generic)类的使用原因和使用方式
    C#反射从入门到放弃(这部分遇到的新东西太多了让人接受不能)
    反射应用——依赖注入
    接口的显式实现
  • 原文地址:https://www.cnblogs.com/blog-index/p/6671909.html
Copyright © 2011-2022 走看看