2026년 상식닷컴 선정 식당 & 카페 리스트
최근에 오픈한 호텔을 찾는다면 살펴보세요

솔리디티에서 'delegatecall'과 'call'의 차이는 무엇인가요?

_____
Q1: Solidity에서 `call`과 `delegatecall`이란 무엇인가요?
- `call`: 호출하는 컨트랙트가 외부 주소의 함수나 이더 전송을 실행할 때 사용하는 저수준 함수입니다. 호출 대상 컨트랙트의 컨텍스트(상태, 주소 등)에서 실행됩니다.
- `delegatecall`: 호출하는 컨트랙트가 다른 컨트랙트의 코드를 자신의 컨텍스트에서 실행하게 해주는 저수준 함수입니다. 호출 대상 컨트랙트가 아니라 호출자 컨트랙트의 상태와 주소, 밸런스를 사용합니다.

---

Q2: `call`과 `delegatecall`의 가장 큰 차이점은 무엇인가요?
- `call`은 호출 대상의 컨트랙트 상태를 변경합니다. 즉, 호출 대상 컨트랙트가 자신의 변수들을 사용해 실행됩니다.
- `delegatecall`은 호출자의 상태를 변경합니다. 호출 대상 컨트랙트의 코드를 빌려오지만 호출자 컨트랙트의 저장공간(storage), 주소(msg.sender는 호출자가 아니라 외부 트랜잭션 발신자), 그리고 이더 밸런스를 사용합니다.

---

Q3: `delegatecall`을 쓸 때 주의할 점은 무엇인가요?
- 호출 대상 컨트랙트는 호출자 컨트랙트와 동일한 storage 레이아웃을 가져야 합니다.
- 잘못 쓰면 호출자의 상태가 의도치 않게 변할 수 있고 보안 이슈를 초래할 수 있습니다 (예: 업그레이드 가능한 컨트랙트에서의 storage 충돌).
- `delegatecall`은 호출 대상 코드가 호출자의 상태에 직접 접근하므로 신뢰할 수 있는 코드에만 사용해야 합니다.

---

Q4: `call`과 비교했을 때 `delegatecall`의 주요 사용 사례는 무엇인가요?
- 프록시 패턴에서 주로 사용합니다. 프록시 컨트랙트가 자신의 상태를 유지하고, 다른 컨트랙트(로직 컨트랙트)의 코드를 가져와 실행할 때 `delegatecall`을 씁니다.
- 라이브러리 호출 시에도 라이브러리 코드를 호출자의 컨텍스트로 실행하기 때문에 `delegatecall`을 사용합니다.

---

Q5: `call` 사용 시의 주요 특징은 무엇인가요?
- 독립적인 컨트랙트 호출로, 호출 대상의 상태, 주소, 메시지 관련 전역 변수들이 호출 대상 컨트랙트의 환경에 맞게 작동합니다.
- 호출 대상 컨트랙트에 이더 전송도 가능합니다.
- 호출 실패 여부 확인과 반환값 핸들링이 필요합니다.

---

Q6: 간단히 요약하면 `call`과 `delegatecall`의 차이는 무엇인가요?
- `call`: 대상 컨트랙트에서 코드 실행 → 대상 상태 변경
- `delegatecall`: 호출자 컨트랙트에서 대상 코드 실행 → 호출자 상태 변경

---

예시
```solidity
// call 예시
(bool success, bytes memory data) = target.call(abi.encodeWithSignature("foo(uint256)", 123));

// delegatecall 예시
(bool success, bytes memory data) = target.delegatecall(abi.encodeWithSignature("foo(uint256)", 123));
```

`call`은 target 컨트랙트의 `foo` 함수를 호출하지만,
`delegatecall`은 호출자 컨트랙트(storage, msg.sender 등)를 기준으로 target의 `foo` 코드를 실행합니다.

---

요약하자면, `call`은 완전한 외부 컨트랙트 함수 호출, `delegatecall`은 현재 컨트랙트 상태에서 다른 컨트랙트 코드를 빌려 실행하는 방식입니다. 따라서 각 상황에 맞게 신중히 선택해야 합니다.
Solidity에서 `delegatecall`과 `call`은 스마트 계약 간의 함수 호출을 처리하는 두 가지 방법입니다.

이 두 메서드는 비슷한 목적을 가지고 있지만, 그 동작 방식과 결과는 크게 다릅니다.

아래에서 이 두 메서드의 차이점과 각각의 사용 사례를 자세히 설명하겠습니다.

1. 기본 개념 - call : `call`은 다른 계약의 함수를 호출하는 방법입니다.

이 메서드는 호출된 계약의 컨텍스트에서 실행되며, 호출된 계약의 상태를 변경할 수 있습니다.

즉, `call`을 사용하면 호출된 계약의 상태 변수에 접근하고 수정할 수 있습니다.

- delegatecall : `delegatecall`은 호출하는 계약의 컨텍스트에서 호출된 계약의 함수를 실행합니다.

즉, 호출된 계약의 코드가 호출하는 계약의 상태 변수에 접근하게 됩니다.

이로 인해 호출하는 계약의 상태를 변경할 수 있으며, 호출된 계약의 상태는 변경되지 않습니다.



2. 상태 변수 접근 - call : 호출된 계약의 상태 변수에 접근하고 수정할 수 있습니다.

호출된 계약의 상태가 변경되며, 호출하는 계약의 상태는 영향을 받지 않습니다.

- delegatecall : 호출하는 계약의 상태 변수에 접근하고 수정합니다.

호출된 계약의 상태는 변경되지 않으며, 호출하는 계약의 상태가 변경됩니다.



3. 사용 사례 - call : 일반적으로 다른 계약의 함수를 호출할 때 사용됩니다.

예를 들어, ERC20 토큰 계약에서 다른 계약의 특정 기능을 호출할 때 `call`을 사용할 수 있습니다.

- delegatecall : 주로 프로토콜 업그레이드나 라이브러리 패턴을 구현할 때 사용됩니다.

예를 들어, 프록시 계약에서 `delegatecall`을 사용하여 로직 계약의 함수를 호출하면, 프록시 계약의 상태를 유지하면서 로직 계약의 기능을 사용할 수 있습니다.

이는 스마트 계약의 업그레이드를 용이하게 합니다.



4. 보안 고려사항 - call : `call`을 사용할 때는 호출된 계약의 코드가 신뢰할 수 있는지 확인해야 합니다.

악의적인 계약이 호출될 경우, 예상치 못한 동작을 초래할 수 있습니다.

- delegatecall : `delegatecall`은 특히 보안에 민감합니다.

호출하는 계약의 상태를 변경하기 때문에, 호출된 계약의 코드가 호출하는 계약의 상태를 의도치 않게 변경할 수 있습니다.

따라서, `delegatecall`을 사용할 때는 호출된 계약의 코드가 안전하고 신뢰할 수 있는지 철저히 검토해야 합니다.



5. 예제 코드 아래는 `call`과 `delegatecall`의 간단한 예제입니다.

```solidity // 호출할 계약 contract Target { uint public value; function setValue(uint _value) public { value = _value; } } // 호출하는 계약 contract Caller { uint public value; function callSetValue(address target, uint _value) public { (bool success, ) = target.call(abi.encodeWithSignature("setValue(uint25

6)", _value)); require(success, "Call failed"); } function delegateCallSetValue(address target, uint _value) public { (bool success, ) = target.delegatecall(abi.encodeWithSignature("setValue(uint25

6)", _value)); require(success, "Delegate call failed"); } } ``` 위의 예제에서 `Caller` 계약은 `Target` 계약의 `setValue` 함수를 호출합니다.

`callSetValue`는 `call`을 사용하여 `Target` 계약의 상태를 변경하고, `delegateCallSetValue`는 `delegatecall`을 사용하여 `Caller` 계약의 상태를 변경합니다.

결론 `call`과 `delegatecall`은 Solidity에서 계약 간의 상호작용을 처리하는 중요한 메서드입니다.

각각의 메서드는 특정한 상황에서 유용하게 사용될 수 있으며, 그 동작 방식과 결과가 다르기 때문에 사용 시 주의가 필요합니다.

특히 `delegatecall`은 보안상의 이유로 더욱 신중하게 사용해야 하며, 호출된 계약의 코드가 안전한지 항상 확인해야 합니다.

작성자: 정재현 [비회원] | 작성일자: 1년 전 2024-11-22 19:32:11
조회수: 175 | 댓글: 0 | 좋아요: 0 | 싫어요: 0
내용이 부정확하다면 싫어요를 클릭해주세요.