solidity数据位置-memory,storage和calldata

时间:2023-03-08 17:27:19
solidity数据位置-memory,storage和calldata

有三种类型,memory,storage和calldata,一般只有外部函数的参数(不包括返回参数)被强制指定为calldata。这种数据位置是只读的,不会持久化到区块链

storage存储或memory内存
memory存储位置同我们普通程序的内存类似,即分配,即使用,动态分配,越过作用域即不可被访问,等待被回收。
而对于storage的变量,数据将永远存在于区块链上。

总结¶
强制指定的数据位置:
    •    外部函数的参数(不包括返回参数): calldata,效果跟 memory 差不多
    •    状态变量: storage
默认数据位置:
    •    函数参数(包括返回参数): memory
    •    所有其它局部变量: storage

下面举例说明赋值行为:

1.memory = storage (值传递,互不影响)

pragma solidity ^0.4.24;

contract Person {

    int public _age;

    constructor (int age) public {
      _age = age;
    }     function f() public view{
      modifyAge(_age);
    }     function modifyAge(int age) public pure{
      age = 100;
    }
}

solidity数据位置-memory,storage和calldata

在这里一开始deploy合约时,传入的age值为30,此时_age的值为30

然后运行f()函数,在这里使用了为storage类型的_age作为函数modifyAge的参数,相当于创建了一个临时变量age(memory类型),将storage类型的变量_age赋值给memory类型的变量age,是值传递,所以在modifyAge函数中,age变量的值的变化并不会影响到_age变量的值

所以再查看_age的值,还是为30

2.storage = memory

当storage是状态变量(即全局变量时),为值传递

当storage为局部变量时,该赋值会出错,解决方法是将storage的局部变量声明为memory即可

1)当storage为局部变量时:

如下面的例子:

pragma solidity ^0.4.;

contract Person {

    string public  _name;

    constructor() public {
_name = "liyuechun";
} function f() public view{ modifyName(_name);
} function modifyName(string name) public pure{ string memory name1 = name;
bytes(name1)[] = 'L';
}
}

调用f()函数,将storage类型的状态变量_name作为参数赋值给函数modifyName(string) memory类型的name形参,为memory = storage,为值传递

然后在函数modifyName(string)中,还将memory类型的name形参赋值给memory类型的name1局部变量,memory = memory,为引用传递,改变一个另一个也跟着改变,但是因为先是进行了值传递,name与_name之间已经互不影响了,所以不会跟着改变_name

solidity数据位置-memory,storage和calldata

2)当storage为状态变量时:

pragma solidity ^0.4.;

contract Person {

    string public  _name;
string public changedName; constructor() public {
_name = "liyuechun";
} function f() public{//不能在声明为view modifyName(_name);
} function modifyName(string name) public{//不能在声明为view changedName = name;
bytes(name)[] = 'L';
}
}
warning:function declared as view,but this expression(potentially) modifies the state and thus requires non-payable(the default) or payable.

因为函数modifyName(string)改变了值changedName的状态,所以不能声明为view了

solidity数据位置-memory,storage和calldata

调用f()函数,将storage类型的状态变量_name作为参数赋值给函数modifyName(string) memory类型的name形参,为memory = storage,为值传递

然后memory类型的name形参赋值给storage类型的状态变量changedName,storage = memory,为值传递,因此name的值的改变不会导致changedName的值的改变,更不要说_name了

调用f()后为:

solidity数据位置-memory,storage和calldata

3.storage = storage

是引用传递,所以一个值的变化一定会导致另一个值的变化

pragma solidity ^0.4.;

contract Person {

    string public  _name;

    constructor() public {
_name = "liyuechun";
} function f() public{ modifyName(_name);
} function modifyName(string storage name) internal { string storage name1 = name;
bytes(name1)[] = 'L';
}
}

⚠️:如果modifyName(string)函数不声明为internal会报错:

TypeError:Location has to be memory for publicly visible functions(remove the "storage" keyword)

这是因为形参是默认为memory类型的,这里声明为storage,那么函数的类型就必须声明为internal或者private

调用f()函数,首先会将为storage类型的_name变量赋值给modifyName(string)函数storage类型的name形参,storage = storage,为引用传递

然后在modifyName(string)函数中,将storage类型的name变量赋值给storage类型的name1变量,storage = storage,为引用传递

都为引用传递,所以最后name1值的变化会导致_name的值的变化

solidity数据位置-memory,storage和calldata

调用f()后:

solidity数据位置-memory,storage和calldata

其实在这里如果将modifyName(string)函数改成如下,也是能够成功的,因为其实没必要进行两次引用传递:

    function modifyName(string storage name) internal {

        bytes(name)[] = 'L';
}

4.memory = memory

是引用传递,所以一个值的变化一定会导致另一个值的变化

pragma solidity ^0.4.;

contract Person {

    function modifyName(string name) public pure returns(string){

        string memory name1 = name;
bytes(name1)[] = 'L';
return name;
}
}

这里调用modifyName(string)函数,将memory类型的形参赋值给memory类型的局部变量name1,memory = memory,为引用传递

这时候改变name1的值,从return 的name可以看到,它的值也随之改变

solidity数据位置-memory,storage和calldata