ValueType 值类型
布尔类型:bool test= (20>19);
整数类型:( int 、 uint)
支持4种类型的运算比较:
- 比较大小: Comparisons: <=, <, ==, !=, >=, >
- 二进制运算:Bit operators: &, |, ^ (bitwise exclusive or), ~ (bitwise negation)
- 位移运算:Shift operators: << (left shift), >> (right shift)
- 算术运算:Arithmetic operators: +, -, unary -, *, /, % (modulo), ** (exponentiation)
Address: 见Address
Contract Types
定长字节数组 Fixed-size byte arrays:bytes1, bytes2, bytes3,byte32
- .length 得到byte数组的固定长度
- 同样支持 比较大小、位移、与二进制运算
- 下标访问:如果x是 byteI类型,那么 x[k] (0<= k < I),返回第k个byte (只读)
- String Literals and Types 字符串:
- 1个字符就是一个字节,也就是solidity中最多可以有255种不同的字符
- 因为1个字符就是一个字节,8位,那么字符就可以进行16进制转换,比如 huang,5个字符,5个字节,可以用10位的十六进制表示
- 字符串是特殊的动态长度字节数组
- 字符串不能够字节的修改长度和内容,需要转换为bytes动态字节数组
- 字符串在solidity中使用了UTF8格式来存储,所以汉字占了3个字节,英文和特殊字符占了一个字节
- 直接通过bytes把string 输出为16进制的数字了
string public name="huangzhenshi";
function getLength() public returns(bytes){
return bytes(name);
}
- 其它
- 不定长字节数组 Dynamically-sized byte array: bytes 和 string
- 枚举类型 Enums
- 函数类型 Function Types
byte类型
- byte既可以表示数字,也可以表示字符串
- byte在数据结构上是8位,所以byte1的上限是255, 256的话,编译报错。如果bytes1表示负数的话,需要一个符号位,那么最小能表示的负数 -128
- byte如果表示16进制的数据的话,正好是2位16进制数,因为 2**4=16,比如 bytes1 num= 0xff,所以用bytes 20来标识 uint160的数字,也就是 0x40位的地址,所以比特币的地址是一串 长度为40的十六进制数字。
- byte表示字符串的时候,一个字符就是一个byte,比如 bytes5="huang"; 也就是说在solidity中一个字符就是一个字节,8位存储,也就是最多可以有255种不同的字符
Address
- 20 字节长度的值(以太坊的地址),160位(2进制),所以可以用一个uint160编码,uint160 和 address可以相互转换
- 地址分为两种状态,普通状态和 address payable状态,只有payable状态的地址才可以接收外部赠与的Eth
- 地址既可以表示一个用户的账户AccountAddress,也可以表示一个合约的地址,例如 msg.sender; 表示发起该调用的调用者的address,如果A用户调用 a合约,那么在a合约中 msg.sender=Account(A).address; 如果A用户调用a合约,并且a合约通过call调用了b合约,那么在b合约中:msg.sender=a合约.address
- solidity中的this 代表当前账户的 address
- address.balance 获取账号余额
- address.transfer 和address.send类似,一个转账失败会抛出异常,send会返回false
- zhangsanAddress.transfer(10Eth) 表示当前账户向 zhangsanAddress转账10Eth, 从合约发起方向某个地址转入以太币(默认单位是wei),地址无效或者合约发起方余额不足时,代码将抛出异常并停止转账。
- this.send 从合约发起方向某个地址转入以太币(单位是wei),地址无效或者合约发起方余额不足时,send不会抛出异常,而是直接返回false。
//获取当前账户余额
function getBalance() returns (uint){
return this.balance;
}
function deposit() payable{
address Account2 = 0x14723a09acff6d2a60dcdf7aa4aff308fddc160c;
Account2.transfer(msg.value);
}
function deposit() payable returns (bool){
address Account2 = 0x14723a09acff6d2a60dcdf7aa4aff308fddc160c;
return Account2.send(msg.value);
}
send()方法执行时有一些风险
- 调用递归深度不能超1024。
- 如果gas不够,执行会失败。
所以transfer相对send较安全,transfer失败会异常,而send则必须通过check 返回值来判断是非交易成功
address 支持的函数
<address>.balance (uint256):balance of the Address in Wei
<address payable>.transfer(uint256 amount):
send given amount of Wei to Address, reverts on failure, forwards 2300 gas stipend, not adjustable<address payable>.send(uint256 amount) returns (bool):send given amount of Wei to Address, returns false on failure, forwards 2300 gas stipend, not adjustable
<address>.call(bytes memory) returns (bool, bytes memory):
issue low-level CALL with the given payload, returns success condition and return data, forwards all available gas, adjustable<address>.delegatecall(bytes memory) returns (bool, bytes memory):
issue low-level DELEGATECALL with the given payload, returns success condition and return data, forwards all available gas, adjustable<address>.staticcall(bytes memory) returns (bool, bytes memory):
issue low-level STATICCALL with the given payload, returns success condition and return data, forwards all available gas, adjustable
Call函数
在Solidity中,可以使用call、delegatecall、callcode三个函数实现跨合约的函数调用功能。
如果A用户调用a合约,并且a合约里面的函数通过call调用了b合约的方法
- 那么在a合约中 msg.sender=Account(A).address;
- 那么在b合约中:msg.sender=Contract(a).address
pragma solidity ^0.4.0;
contract A {
address public temp1;
uint256 public temp2;
function fcall(address addr) public {
temp1 = msg.sender;
temp2 = 100;
addr.call(bytes4(keccak256("test()")));
}
}
contract B {
address public temp1;
uint256 public temp2;
function test() public {
temp1 = msg.sender;
temp2 = 200;
}
}
参考
英文版官网
https://solidity.readthedocs.io/en/latest/types.html
中文版翻译
https://www.cnblogs.com/StephenWu/p/7096550.html