스마트 계약의 구조
페이지 마지막 업데이트됨: 2026년 2월 23일
스마트 컨트랙트는 이더리움 주소 체계 상에서 실행되는 프로그램이며, 거래가 발생할 때 실행되는 데이터와 함수로 구성되어 있습니다. 지금부터 그 구성 요소에 대해 전반적으로 살펴보도록 하겠습니다.
필수 구성 요소
스마트 계약에 대해 먼저 읽어보시기 바랍니다. 자바 스크립트나 파이썬과 같은 프로그래밍 언어에 상당히 익숙하다는 것을 전제합니다.
데이터
모든 계약 데이터는 storage 또는 memory 위치에 할당되어야 합니다. 이 중 스토리지 사용은 비용이 더 발생하므로 여러분은 어떤 것을 활용할 지 미리 고려해야 합니다.
저장 공간
영구적인 데이터는 스토리지로 간주되며 상태 변수 형태로 표현됩니다. 이런 값은 블록체인 상에 영구히 남게 되므로, 컨트랙트 컴파일 시에 스토리지 형태로 사용할 변수를 명확히 구분할 필요가 있습니다.
1// Solidity 예시2contract SimpleStorage {3 uint storedData; // 상태 변수4 // ...5}1# Vyper 예시2storedData: int128여러분이 객체 지향적인 언어를 사용해 보았다면 대부분의 변수 타입에는 익숙할 것입니다. 하지만 이더리움 개발이 처음이라면 address는 생소할 수 있습니다.
address 유형은 이더리움 주소를 저장할 수 있으며, 이는 20바이트 또는 160비트와 같습니다. 또한 16진수로 표기되며 0x로 시작합니다.
기타 유형에는 다음이 포함됩니다:
- 불리언
- 정수
- 고정 소수점 숫자
- 고정 크기 바이트 배열
- 동적 크기 바이트 배열
- 유리수 및 정수 리터럴
- 문자열 리터럴
- 16진수 리터럴
- 열거형
더 많은 설명은 문서를 참고하십시오:
메모리
메모리 변수는 컨트랙스 함수가 실행되는 시간에만 사용이 가능하기 때문에, 블록체인에는 저장되지 않고 비용도 저렴합니다.
Solidity 문서 (opens in a new tab)에서 EVM이 데이터(저장 공간, 메모리, 스택)를 저장하는 방법에 대해 자세히 알아보세요.
환경 변수
여러분이 컨트랙트에 정의한 변수 외에도 특별하게 사용할 수 있는 전역 변수가 있습니다. 그것을 통해 주로 블록체인과 현재 트랜잭션에 대한 정보를 알 수 있습니다.
예시:
| 속성 | 상태 변수 | 설명 |
|---|---|---|
block.timestamp | uint256 | 현재 블록 에포크 타임스탬프 |
msg.sender | 주소 | 메시지의 발신자(현재 호출) |
함수
함수는 거래에 대한 정보를 간단한 방법으로 가져오거나 설정할 수 있습니다.
함수 호출에는 두 가지 방법이 있습니다.
internal– EVM 호출을 생성하지 않습니다- 내부 함수와 상태 변수는 내부에서만 (즉, 현재 계약 또는 그로부터 파생된 계약 내에서만) 접근할 수 있습니다
external– EVM 호출을 생성합니다- 외부 함수는 계약 인터페이스의 일부로, 다른 계약 및 트랜잭션을 통해 호출할 수 있습니다. 외부 함수
f는 내부에서 호출할 수 없습니다(예:f()는 작동하지 않지만this.f()는 작동합니다).
- 외부 함수는 계약 인터페이스의 일부로, 다른 계약 및 트랜잭션을 통해 호출할 수 있습니다. 외부 함수
public 또는 private일 수도 있습니다
public함수는 계약 내에서 내부적으로 또는 메시지를 통해 외부에서 호출할 수 있습니다private함수는 정의된 계약에서만 볼 수 있으며 파생된 계약에서는 볼 수 없습니다
함수와 상태 변수 모두 public 또는 private 설정이 가능합니다.
계약의 상태 변수를 업데이트하는 함수 예제입니다:
1// Solidity 예시2function update_name(string value) public {3 dapp_name = value;4}string유형의 매개변수value가update_name함수에 전달됩니다public으로 선언되어 누구나 접근할 수 있습니다view로 선언되지 않았으므로 계약 상태를 수정할 수 있습니다
View 함수
이 함수들은 컨트랙트 데이터를 변경하지 않습니다. 흔히 "getter" 함수라고도 하며 사용자의 지갑의 잔액을 얻을 때 사용될 때의 예시입니다.
1// Solidity 예시2function balanceOf(address _owner) public view returns (uint256 _balance) {3 return ownerPizzaCount[_owner];4}1dappName: public(string)23@view4@public5def readName() -> string:6 return dappName상태를 수정하는 것으로 간주되는 것:
- 상태 변수에 쓰기
- 이벤트 발생시키기 (opens in a new tab).
- 다른 계약 생성하기 (opens in a new tab).
selfdestruct사용하기.- 호출을 통해 이더 전송
view또는pure로 표시되지 않은 함수 호출하기.- 저수준 호출 사용
- 특정 오프코드를 포함하는 인라인 어셈블리 사용
생성자 함수
constructor 함수는 계약이 처음 배포될 때 한 번만 실행됩니다. 많은 클래스 기반 프로그래밍 언어의 constructor와 마찬가지로, 이 함수는 종종 상태 변수를 지정된 값으로 초기화합니다.
1// Solidity 예시2// 계약의 데이터를 초기화하고, `owner`를3// 계약 생성자의 주소로 설정합니다.4constructor() public {5 // 모든 스마트 계약은 외부 트랜잭션에 의존하여 함수를 실행합니다.6 // `msg`는 보낸 사람의 주소 및 트랜잭션에 포함된 ETH 값과 같은7 // 주어진 트랜잭션의 관련 데이터를 포함하는 전역 변수입니다.8 // 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/units-and-global-variables.html#block-and-transaction-properties9 owner = msg.sender;10}모두 보기1# Vyper 예시23@external4def __init__(_beneficiary: address, _bidding_time: uint256):5 self.beneficiary = _beneficiary6 self.auctionStart = block.timestamp7 self.auctionEnd = self.auctionStart + _bidding_time내장 함수
컨트랙트를 작성할 때 정의하는 변수와 함수 외에 추가적으로 사용할 수 있는 특별한 내장 함수가 있습니다. 대표적인 함수에는 아래가 있으며,
address.send()– 솔리디티send(address)– Vyper
다른 계정으로 ETH를 송금할 때 사용할 수 있습니다.
함수 작성하기
당신의 함수에는 다음이 필요합니다:
- 매개변수 변수 및 타입(매개변수를 받는 경우)
- 내부/외부 선언
- 순수(pure)/뷰(view)/지급(payable) 선언
- 반환 타입(값을 반환하는 경우)
1pragma solidity >=0.4.0 <=0.6.0;23contract ExampleDapp {4 string dapp_name; // 상태 변수56 // 계약이 배포될 때 호출되어 값을 초기화합니다7 constructor() public {8 dapp_name = "My Example dapp";9 }1011 // Get 함수12 function read_name() public view returns(string) {13 return dapp_name;14 }1516 // Set 함수17 function update_name(string value) public {18 dapp_name = value;19 }20}모두 보기완전한 계약은 다음과 같을 수 있습니다. 여기서 constructor 함수는 dapp_name 변수에 대한 초기값을 제공합니다.
이벤트와 로그
이벤트는 스마트 계약이 프론트엔드나 다른 구독 애플리케이션과 통신할 수 있게 합니다. 트랜잭션이 검증되어 블록에 추가되면, 스마트 계약은 이벤트를 방출하고 정보를 로그로 기록할 수 있으며, 프론트엔드는 이를 처리하고 활용할 수 있습니다.
주석이 달린 예시
이것들은 Solidity로 작성된 몇 가지 예시입니다. 코드를 사용해보고 싶다면 Remix (opens in a new tab)에서 상호작용할 수 있습니다.
Hello world
1// 시맨틱 버저닝을 사용하여 Solidity 버전을 지정합니다.2// 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/layout-of-source-files.html#pragma3pragma solidity ^0.5.10;45// `HelloWorld`라는 이름의 계약을 정의합니다.6// 계약은 함수와 데이터(상태)의 모음입니다.7// 배포된 계약은 이더리움 블록체인의 특정 주소에 존재합니다.8// 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/structure-of-a-contract.html9contract HelloWorld {1011 // `string` 유형의 상태 변수 `message`를 선언합니다.12 // 상태 변수는 계약 저장 공간에 영구적으로 저장되는 변수입니다.13 // `public` 키워드를 사용하면 계약 외부에서 변수에 접근할 수 있으며14 // 다른 계약이나 클라이언트가 값을 접근하기 위해 호출할 수 있는 함수를 생성합니다.15 string public message;1617 // 많은 클래스 기반 객체 지향 언어와 마찬가지로, 생성자는18 // 계약 생성 시에만 실행되는 특별한 함수입니다.19 // 생성자는 계약의 데이터를 초기화하는 데 사용됩니다.20 // 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/contracts.html#constructors21 constructor(string memory initMessage) public {22 // 문자열 인수 `initMessage`를 받아 계약의 `message` 저장 공간 변수에23 // 값을 설정합니다.24 message = initMessage;25 }2627 // 문자열 인수를 받아28 // `message` 저장 공간 변수를 업데이트하는 public 함수입니다.29 function update(string memory newMessage) public {30 message = newMessage;31 }32}모두 보기토큰
1pragma solidity ^0.5.10;23contract Token {4 // `address`는 이메일 주소와 비슷하며, 이더리움에서 계정을 식별하는 데 사용됩니다.5 // 주소는 스마트 계약 또는 외부(사용자) 계정을 나타낼 수 있습니다.6 // 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/types.html#address7 address public owner;89 // `mapping`은 기본적으로 해시 테이블 데이터 구조입니다.10 // 이 `mapping`은 부호 없는 정수(토큰 잔액)를 주소(토큰 보유자)에 할당합니다.11 // 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/types.html#mapping-types12 mapping (address => uint) public balances;1314 // 이벤트를 통해 블록체인 활동을 기록할 수 있습니다.15 // 이더리움 클라이언트는 계약 상태 변경에 반응하기 위해 이벤트를 수신할 수 있습니다.16 // 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/contracts.html#events17 event Transfer(address from, address to, uint amount);1819 // 계약 데이터를 초기화하고, `owner`를20 // 계약 생성자의 주소로 설정합니다.21 constructor() public {22 // 모든 스마트 계약은 외부 트랜잭션에 의존하여 함수를 실행합니다.23 // `msg`는 보낸 사람의 주소 및 트랜잭션에 포함된 ETH 값과 같은24 // 주어진 트랜잭션의 관련 데이터를 포함하는 전역 변수입니다.25 // 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/units-and-global-variables.html#block-and-transaction-properties26 owner = msg.sender;27 }2829 // 새로운 토큰을 생성하여 주소로 보냅니다.30 function mint(address receiver, uint amount) public {31 // `require`는 특정 조건을 강제하는 데 사용되는 제어 구조입니다.32 // `require`문이 `false`로 평가되면 예외가 발생하며,33 // 현재 호출 동안 상태에 적용된 모든 변경 사항이 되돌려집니다.34 // 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/control-structures.html#error-handling-assert-require-revert-and-exceptions3536 // 계약 소유자만 이 함수를 호출할 수 있습니다37 require(msg.sender == owner, "당신은 소유자가 아닙니다.");3839 // 토큰의 최대량을 강제합니다40 require(amount < 1e60, "최대 발행량을 초과했습니다");4142 // `receiver`의 잔액을 `amount`만큼 증가시킵니다43 balances[receiver] += amount;44 }4546 // 호출자로부터 주소로 기존 토큰을 보냅니다.47 function transfer(address receiver, uint amount) public {48 // 보내는 사람은 보낼 만큼의 충분한 토큰을 가지고 있어야 합니다49 require(amount <= balances[msg.sender], "잔액이 부족합니다.");5051 // 두 주소의 토큰 잔액을 조정합니다52 balances[msg.sender] -= amount;53 balances[receiver] += amount;5455 // 이전에 정의된 이벤트를 발생시킵니다56 emit Transfer(msg.sender, receiver, amount);57 }58}모두 보기고유 디지털 자산
1pragma solidity ^0.5.10;23// 다른 파일의 심볼을 현재 계약으로 가져옵니다.4// 이 경우 OpenZeppelin의 일련의 헬퍼 계약입니다.5// 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/layout-of-source-files.html#importing-other-source-files67import "../node_modules/@openzeppelin/contracts/token/ERC721/IERC721.sol";8import "../node_modules/@openzeppelin/contracts/token/ERC721/IERC721Receiver.sol";9import "../node_modules/@openzeppelin/contracts/introspection/ERC165.sol";10import "../node_modules/@openzeppelin/contracts/math/SafeMath.sol";1112// `is` 키워드는 외부 계약에서 함수와 키워드를 상속받는 데 사용됩니다.13// 이 경우, `CryptoPizza`는 `IERC721` 및 `ERC165` 계약에서 상속받습니다.14// 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/contracts.html#inheritance15contract CryptoPizza is IERC721, ERC165 {16 // OpenZeppelin의 SafeMath 라이브러리를 사용하여 산술 연산을 안전하게 수행합니다.17 // 자세히 알아보기: https://docs.openzeppelin.com/contracts/2.x/api/math#SafeMath18 using SafeMath for uint256;1920 // Solidity의 상수 상태 변수는 다른 언어와 유사하지만21 // 컴파일 시에 상수인 표현식에서 할당해야 합니다.22 // 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/contracts.html#constant-state-variables23 uint256 constant dnaDigits = 10;24 uint256 constant dnaModulus = 10 ** dnaDigits;25 bytes4 private constant _ERC721_RECEIVED = 0x150b7a02;2627 // Struct 유형을 사용하면 자신만의 유형을 정의할 수 있습니다.28 // 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/types.html#structs29 struct Pizza {30 string name;31 uint256 dna;32 }3334 // Pizza 구조체의 빈 배열을 생성합니다.35 Pizza[] public pizzas;3637 // 피자 ID에서 소유자 주소로의 매핑38 mapping(uint256 => address) public pizzaToOwner;3940 // 소유자 주소에서 소유한 토큰 수로의 매핑41 mapping(address => uint256) public ownerPizzaCount;4243 // 토큰 ID에서 승인된 주소로의 매핑44 mapping(uint256 => address) pizzaApprovals;4546 // 매핑을 중첩할 수 있습니다. 이 예는 소유자를 운영자 승인에 매핑합니다.47 mapping(address => mapping(address => bool)) private operatorApprovals;4849 // 문자열(이름)과 DNA로부터 임의의 피자를 생성하는 내부 함수50 function _createPizza(string memory _name, uint256 _dna)51 // `internal` 키워드는 이 함수가 이 계약과52 // 이 계약을 파생하는 계약 내에서만 보인다는 것을 의미합니다.53 // 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/contracts.html#visibility-and-getters54 internal55 // `isUnique`는 피자가 이미 존재하는지 확인하는 함수 제어자입니다.56 // 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/structure-of-a-contract.html#function-modifiers57 isUnique(_name, _dna)58 {59 // 피자 배열에 피자를 추가하고 id를 얻습니다.60 uint256 id = SafeMath.sub(pizzas.push(Pizza(_name, _dna)), 1);6162 // 피자 소유자가 현재 사용자와 동일한지 확인합니다.63 // 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/control-structures.html#error-handling-assert-require-revert-and-exceptions6465 // note that address(0)은 제로 주소이며,66 // pizza[id]가 아직 특정 사용자에게 할당되지 않았음을 나타냅니다.6768 assert(pizzaToOwner[id] == address(0));6970 // 피자를 소유자에게 매핑합니다.71 pizzaToOwner[id] = msg.sender;72 ownerPizzaCount[msg.sender] = SafeMath.add(73 ownerPizzaCount[msg.sender],74 175 );76 }7778 // 문자열(이름)로부터 임의의 피자를 생성합니다.79 function createRandomPizza(string memory _name) public {80 uint256 randDna = generateRandomDna(_name, msg.sender);81 _createPizza(_name, randDna);82 }8384 // 문자열(이름)과 소유자 주소(생성자)로부터 임의의 DNA를 생성합니다.85 function generateRandomDna(string memory _str, address _owner)86 public87 // `pure`로 표시된 함수는 상태를 읽거나 수정하지 않을 것을 약속합니다.88 // 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/contracts.html#pure-functions89 pure90 returns (uint256)91 {92 // 문자열(이름) + 주소(소유자)로부터 임의의 uint를 생성합니다.93 uint256 rand = uint256(keccak256(abi.encodePacked(_str))) +94 uint256(_owner);95 rand = rand % dnaModulus;96 return rand;97 }9899 // 소유자가 찾은 피자 배열을 반환합니다.100 function getPizzasByOwner(address _owner)101 public102 // `view`로 표시된 함수는 상태를 수정하지 않을 것을 약속합니다.103 // 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/contracts.html#view-functions104 view105 returns (uint256[] memory)106 {107 // `memory` 저장 위치를 사용하여 이 함수 호출의108 // 수명 주기 동안만 값을 저장합니다.109 // 자세히 알아보기: https://solidity.readthedocs.io/en/v0.5.10/introduction-to-smart-contracts.html#storage-memory-and-the-stack110 uint256[] memory result = new uint256[](ownerPizzaCount[_owner]);111 uint256 counter = 0;112 for (uint256 i = 0; i < pizzas.length; i++) {113 if (pizzaToOwner[i] == _owner) {114 result[counter] = i;115 counter++;116 }117 }118 return result;119 }120121 // 피자와 소유권을 다른 주소로 이전합니다.122 function transferFrom(address _from, address _to, uint256 _pizzaId) public {123 require(_from != address(0) && _to != address(0), "유효하지 않은 주소입니다.");124 require(_exists(_pizzaId), "피자가 존재하지 않습니다.");125 require(_from != _to, "같은 주소로 전송할 수 없습니다.");126 require(_isApprovedOrOwner(msg.sender, _pizzaId), "주소가 승인되지 않았습니다.");127128 ownerPizzaCount[_to] = SafeMath.add(ownerPizzaCount[_to], 1);129 ownerPizzaCount[_from] = SafeMath.sub(ownerPizzaCount[_from], 1);130 pizzaToOwner[_pizzaId] = _to;131132 // 가져온 IERC721 계약에 정의된 이벤트를 발생시킵니다.133 emit Transfer(_from, _to, _pizzaId);134 _clearApproval(_to, _pizzaId);135 }136137 /**138 * 주어진 토큰 ID의 소유권을 다른 주소로 안전하게 이전합니다.139 * 대상 주소가 계약인 경우, `onERC721Received`를 구현해야 하며,140 * 이는 안전한 전송 시 호출되고 매직 값141 * `bytes4(keccak256("onERC721Received(address,address,uint256,bytes)"))`을 반환해야 합니다.142 * 그렇지 않으면 전송이 되돌려집니다.143 */144 function safeTransferFrom(address from, address to, uint256 pizzaId)145 public146 {147 // solium-disable-next-line arg-overflow148 this.safeTransferFrom(from, to, pizzaId, "");149 }150151 /**152 * 주어진 토큰 ID의 소유권을 다른 주소로 안전하게 이전합니다.153 * 대상 주소가 계약인 경우, `onERC721Received`를 구현해야 하며,154 * 이는 안전한 전송 시 호출되고 매직 값155 * `bytes4(keccak256("onERC721Received(address,address,uint256,bytes)"))`을 반환해야 합니다.156 * 그렇지 않으면 전송이 되돌려집니다.157 */158 function safeTransferFrom(159 address from,160 address to,161 uint256 pizzaId,162 bytes memory _data163 ) public {164 this.transferFrom(from, to, pizzaId);165 require(_checkOnERC721Received(from, to, pizzaId, _data), "onERC721Received를 구현해야 합니다.");166 }167168 /**169 * 대상 주소에서 `onERC721Received`를 호출하는 내부 함수170 * 대상 주소가 계약이 아닌 경우 호출이 실행되지 않습니다.171 */172 function _checkOnERC721Received(173 address from,174 address to,175 uint256 pizzaId,176 bytes memory _data177 ) internal returns (bool) {178 if (!isContract(to)) {179 return true;180 }181182 bytes4 retval = IERC721Receiver(to).onERC721Received(183 msg.sender,184 from,185 pizzaId,186 _data187 );188 return (retval == _ERC721_RECEIVED);189 }190191 // 피자를 소각합니다 - 토큰을 완전히 파괴합니다.192 // `external` 함수 제어자는 이 함수가193 // 계약 인터페이스의 일부이며 다른 계약이 호출할 수 있음을 의미합니다.194 function burn(uint256 _pizzaId) external {195 require(msg.sender != address(0), "유효하지 않은 주소입니다.");196 require(_exists(_pizzaId), "피자가 존재하지 않습니다.");197 require(_isApprovedOrOwner(msg.sender, _pizzaId), "주소가 승인되지 않았습니다.");198199 ownerPizzaCount[msg.sender] = SafeMath.sub(200 ownerPizzaCount[msg.sender],201 1202 );203 pizzaToOwner[_pizzaId] = address(0);204 }205206 // 주소별 피자 수를 반환합니다.207 function balanceOf(address _owner) public view returns (uint256 _balance) {208 return ownerPizzaCount[_owner];209 }210211 // id로 찾은 피자의 소유자를 반환합니다.212 function ownerOf(uint256 _pizzaId) public view returns (address _owner) {213 address owner = pizzaToOwner[_pizzaId];214 require(owner != address(0), "유효하지 않은 피자 ID입니다.");215 return owner;216 }217218 // 다른 주소가 피자의 소유권을 이전하도록 승인합니다.219 function approve(address _to, uint256 _pizzaId) public {220 require(msg.sender == pizzaToOwner[_pizzaId], "피자 소유자여야 합니다.");221 pizzaApprovals[_pizzaId] = _to;222 emit Approval(msg.sender, _to, _pizzaId);223 }224225 // 특정 피자에 대해 승인된 주소를 반환합니다.226 function getApproved(uint256 _pizzaId)227 public228 view229 returns (address operator)230 {231 require(_exists(_pizzaId), "피자가 존재하지 않습니다.");232 return pizzaApprovals[_pizzaId];233 }234235 /**236 * 주어진 토큰 ID의 현재 승인을 지우는 비공개 함수237 * 주어진 주소가 토큰의 소유자가 아닌 경우 되돌립니다.238 */239 function _clearApproval(address owner, uint256 _pizzaId) private {240 require(pizzaToOwner[_pizzaId] == owner, "피자 소유자여야 합니다.");241 require(_exists(_pizzaId), "피자가 존재하지 않습니다.");242 if (pizzaApprovals[_pizzaId] != address(0)) {243 pizzaApprovals[_pizzaId] = address(0);244 }245 }246247 /*248 * 주어진 운영자의 승인을 설정하거나 해제합니다.249 * 운영자는 보낸 사람을 대신하여 모든 토큰을 이전할 수 있습니다.250 */251 function setApprovalForAll(address to, bool approved) public {252 require(to != msg.sender, "자신의 주소를 승인할 수 없습니다.");253 operatorApprovals[msg.sender][to] = approved;254 emit ApprovalForAll(msg.sender, to, approved);255 }256257 // 주어진 소유자가 운영자를 승인했는지 여부를 알려줍니다.258 function isApprovedForAll(address owner, address operator)259 public260 view261 returns (bool)262 {263 return operatorApprovals[owner][operator];264 }265266 // 피자 소유권을 가져옵니다 - 승인된 사용자만 가능합니다.267 function takeOwnership(uint256 _pizzaId) public {268 require(_isApprovedOrOwner(msg.sender, _pizzaId), "주소가 승인되지 않았습니다.");269 address owner = this.ownerOf(_pizzaId);270 this.transferFrom(owner, msg.sender, _pizzaId);271 }272273 // 피자가 존재하는지 확인합니다.274 function _exists(uint256 pizzaId) internal view returns (bool) {275 address owner = pizzaToOwner[pizzaId];276 return owner != address(0);277 }278279 // 주소가 소유자인지 또는 피자를 이전하도록 승인되었는지 확인합니다.280 function _isApprovedOrOwner(address spender, uint256 pizzaId)281 internal282 view283 returns (bool)284 {285 address owner = pizzaToOwner[pizzaId];286 // 다음으로 인해 solium 검사를 비활성화합니다.287 // https://github.com/duaraghav8/Solium/issues/175288 // solium-disable-next-line operator-whitespace289 return (spender == owner ||290 this.getApproved(pizzaId) == spender ||291 this.isApprovedForAll(owner, spender));292 }293294 // 피자가 고유하고 아직 존재하지 않는지 확인합니다.295 modifier isUnique(string memory _name, uint256 _dna) {296 bool result = true;297 for (uint256 i = 0; i < pizzas.length; i++) {298 if (299 keccak256(abi.encodePacked(pizzas[i].name)) ==300 keccak256(abi.encodePacked(_name)) &&301 pizzas[i].dna == _dna302 ) {303 result = false;304 }305 }306 require(result, "같은 이름의 피자가 이미 존재합니다.");307 _;308 }309310 // 대상 주소가 계약인지 여부를 반환합니다.311 function isContract(address account) internal view returns (bool) {312 uint256 size;313 // 현재 주소에 계약이 있는지 확인하는 더 좋은 방법은 없습니다.314 // 해당 주소의 코드 크기를 확인하는 것보다.315 // 이것이 어떻게 작동하는지에 대한 자세한 내용은 https://ethereum.stackexchange.com/a/14016/36603을316 // 참조하세요.317 // TODO Serenity 릴리스 전에 이것을 다시 확인하세요. 모든 주소가318 // 계약이 될 것이기 때문입니다.319 // solium-disable-next-line security/no-inline-assembly320 assembly {321 size := extcodesize(account)322 }323 return size > 0;324 }325}모두 보기더 읽어보기
스마트 계약에 대한 더 완전한 개요를 원한다면 Solidity와 Vyper의 문서를 확인하세요:
관련 주제
관련 튜토리얼
- 계약 크기 제한에 대응하기 위한 계약 축소 – 스마트 계약의 크기를 줄이기 위한 몇 가지 실용적인 팁입니다.
- 이벤트를 사용하여 스마트 계약에서 데이터 기록하기 – 스마트 계약 이벤트에 대한 소개와 이를 사용하여 데이터를 기록하는 방법입니다.
- Solidity에서 다른 계약과 상호작용하기 – 기존 계약에서 스마트 계약을 배포하고 상호작용하는 방법입니다.