以太坊中的地址的长度为20
字节,一字节等于8位,一共160
位,所以address
其实亦可以用uint160
来声明。
备注:以太坊钱包地址是以16进制的形式呈现,我们知道一个十六进制的数字占4
位,160 / 4 = 40
,所以钱包地址ca35b7d915458ef540ade6068dfe2f44e8fa733c
的长度为40。
不可不知的几个常识
合约拥有者
msg.sender
就是当前调用方法时的发起人,一个合约部署后,通过钱包地址操作合约的人很多,但是如何正确判断谁是合约的拥有者,判断方式很简单,就是第一次部署合约时,谁出的gas
,谁就对合约具有拥有权。
合约地址
一个合约部署后,会有一个合约地址,这个合约地址就代表合约自己。
this
this
在合约中到底是msg.sender
还是合约地址
,我们写一个栗子验证一下。
pragma solidity ^0.4.0;
contract addressBalance{
function getContractAddress() constant returns (address){
return this;
}
function getMsgSenderAddrress() constant returns (address){
return msg.sender;
}
}
由打印结果可以看出,this
代表的是合约地址。同时也可以看出msg.sender
即是调用合约的账户地址
支持的运算符
-
<=
,<
,==
,!=
,>=
和>
成员变量和函数
一、 balance
如果我们需要查看一个地址的余额,我们可以使用balance
属性进行查看。
pragma solidity ^0.4.0;
contract addressBalance{
function getBalance(address addr) constant returns (uint){
return addr.balance;
}
}
二、this 查看当前合约地址余额
上面说了this
代表当前合约地址,那么如果在代码中我们只是想简单查询当前合约余额,我们可以直接通过this.balance
进行查询。
pragma solidity ^0.4.0;
contract addressBalance{
function getBalance() constant returns (uint){
return this.balance;
}
function getBalance(address addr) constant returns (uint){
return addr.balance;
}
}
三、transfer
transfer:从合约发起方向某个地址转入以太币,地址无效或者合约发起方余额不足时,代码将抛出异常并停止转账。
pragma solidity ^0.4.0;
contract PayableKeyword{
pragma solidity ^0.4.0;
contract PayableKeyword{
// 从合约发起方 向`0x14723a09acff6d2a60dcdf7aa4aff308fddc160c`地址转入 msg.value 个以太币
function deposit() payable{
address Account2 = 0x14723a09acff6d2a60dcdf7aa4aff308fddc160c;
Account2.transfer(msg.value);
}
// 读取 0x14723a09acff6d2a60dcdf7aa4aff308fddc160c 地址的余额
function getAccount2Balance() constant returns (uint) {
address Account2 = 0x14723a09acff6d2a60dcdf7aa4aff308fddc160c;
return Account2.balance;
}
// 读取合约发起方的余额
function getOwnerBalance() constant returns (uint) {
address Owner = msg.sender;
return Owner.balance;
}
}
}
四、send
send:send
相对transfer
方法较底层,不过使用方法和transfer
相同,都是从合约发起方向某个地址转入以太币,地址无效或者合约发起方余额不足时,send
不会抛出异常,而是直接返回false
。
pragma solidity ^0.4.4;
contract PayableKeyword{
function deposit() payable returns (bool){
address Account2 = 0x14723a09acff6d2a60dcdf7aa4aff308fddc160c;
return Account2.send(msg.value);
}
function getAccount2Balance() constant returns (uint) {
address Account2 = 0x14723a09acff6d2a60dcdf7aa4aff308fddc160c;
return Account2.balance;
}
function getOwnerBalance() constant returns (uint) {
address Owner = msg.sender;
return Owner.balance;
}
}
send()方法执行时有一些风险
- 调用递归深度不能超1024。
- 如果gas不够,执行会失败。
- 所以使用这个方法要检查成功与否。或为保险起见,货币操作时要使用一些最佳实践。
transfer
相对send
较安全
五、call(),callcode()和delegatecall()这三个方法略复杂且底层,以后专门介绍
地址字面量(Address Literals)
十六进制的字符串,凡是能通过地址合法性检查(address checksum test),就会被认为是地址,如0xdCad3a6d3569DF655070DEd06cb7A1b2Ccd1D3AF
。需要注意的是39到41位没有通过地址合法性检查的,会提示一个警告,但会被视为普通的有理数字面量。