현대 프로그래밍 언어에 매크로가 포함되지 않은 이유는 무엇입니까?


31

C / C ++에서 매우 안전하지 않게 구현 된 것을 알고 있습니다. 더 안전한 방식으로 구현할 수 없습니까? 매크로의 단점은 그들이 제공하는 거대한 힘을 능가하기에 충분히 나쁩니 까?


4
다른 방법으로는 쉽게 달성 할 수없는 매크로가 정확히 어떤 힘을 제공합니까?
Chinmay Kanchi

2
C #에서는 간단한 속성 만들기 및 지원 필드를 하나의 선언으로 래핑하는 데 핵심 언어 확장이 필요했습니다. 그리고 이것은 어휘 매크로 이상을 필요로합니다. 어떻게 Visual Basic의 WithEvents수정 자에 해당하는 기능을 만들 수 있습니까? 시맨틱 매크로와 같은 것이 필요합니다.
Jeffrey Hantin

3
매크로의 문제점은 모든 강력한 메커니즘과 마찬가지로 프로그래머가 다른 사람이 만들거나 만들 것이라는 가정을 깰 수 있다는 것입니다. 가정은 추론의 열쇠이며 논리에 대해 타당하게 추론하는 능력이 없어 진전이 엄청나게 진행됩니다.
dan_waterworth

2
@Chinmay : 매크로가 코드를 생성합니다. 그 힘을 가진 Java 언어에는 기능이 없습니다.
케빈 클라인

1
@Chinmay Kanchi : 매크로를 사용하면 런타임 대신 컴파일 타임에 코드를 실행 (평가) 할 수 있습니다.
Giorgio

답변:


57

나는 주된 이유는 매크로가 어휘 적이라고 생각합니다 . 몇 가지 결과가 있습니다.

  • 컴파일러는 매크로가 의미 론적으로 닫혀 있는지 확인하는 방법이 없습니다. 즉, 매크로가 함수처럼 "의미있는 단위"를 나타냅니다. (고려 #define TWO 1+1하는 것은 무엇 TWO*TWO입니까? 3.)

  • 매크로는 함수처럼 입력되지 않습니다. 컴파일러는 매개 변수 및 리턴 유형이 의미가 있는지 확인할 수 없습니다. 매크로를 사용하는 확장 식만 확인할 수 있습니다.

  • 코드가 컴파일되지 않으면 컴파일러는 오류가 매크로 자체에 있는지 또는 매크로가 사용되는 장소에 있는지 알 수있는 방법이 없습니다. 컴파일러는 시간의 절반을 잘못된 장소에보고하거나 둘 중 하나라도 괜찮더라도보고해야합니다. (고려 #define min(x,y) (((x)<(y))?(x):(y)): 유형 xy일치하지 않거나 구현하지 않으면 컴파일러는 어떻게해야 operator<합니까?)

  • 자동화 된 도구는 의미 적으로 유용한 방식으로 작업 할 수 없습니다. 특히 함수처럼 작동하지만 표현식으로 확장되는 매크로에는 IntelliSense와 같은 것을 사용할 수 없습니다. (다시 min예)

  • 매크로의 부작용은 함수와 같이 명시 적이 지 않으므로 프로그래머에게 혼란을 줄 수 있습니다. ( min예를 다시 고려하십시오. 함수 호출에서에 대한 표현식 x은 한 번만 평가 된다는 것을 알고 있지만 여기서는 매크로를 보지 않으면 알 수 없습니다.)

내가 말했듯이, 이것들은 모두 매크로가 어휘 적이라는 사실의 결과입니다. 더 적절한 것으로 바꾸려고하면 함수와 상수가 생깁니다.


32
모든 매크로 언어가 어휘적인 것은 아닙니다. 예를 들어 Scheme 매크로는 구문 적이며 C ++ 템플릿은 의미 론적입니다. 어휘 매크로 시스템은 구문을 인식하지 않고도 어떤 언어로든 고정시킬 수 있다는 차이점이 있습니다.
Jeffrey Hantin

8
@Jeffrey :“사람들이 일반적으로 어휘 매크로를 생각하는 프로그래밍 언어로 매크로를 참조 할 때”“매크로가 어휘 적”이라고 말하는 것은 정말 짧은 축약 인 것 같습니다. Scheme이 근본적으로 다른 무언가에이로드 된 용어를 사용한다는 것은 불행한 일입니다. 그러나 C ++ 템플릿은 완전히 어휘 적이 지 않기 때문에 매크로라고 널리 알려져 있지 않습니다.
Timwi

25
Scheme의 용어 매크로 사용은 Lisp로 거슬러 올라갑니다. 아마도 아마도 대부분의 다른 용도보다 우선합니다. IIRC C / C ++ 시스템은 원래 전 처리기 라고 불렀습니다 .
Bevan

16
@Bevan이 맞습니다. 매크로가 어휘 적이라고 말하는 것은 가장 친숙한 새가 펭귄이기 때문에 새가 날 수 없다고 말하는 것과 같습니다. 즉, 당신이 제기하는 대부분의 (그러나 전부는 아님) 포인트는 구문 매크로에도 적용되지만, 그 정도는 적습니다.
Laurence Gonsalves

13

그러나 C / C ++보다 매크로 더 잘 설계하고 구현할 있습니다.

매크로의 문제점은 실제로 코드를 다른 것으로 재 작성 하는 언어 구문 확장 메커니즘이라는 점입니다.

  • C / C ++의 경우 기본 위생 검사가 없습니다. 조심하면 문제가 없습니다. 실수를하거나 매크로를 과도하게 사용하면 큰 문제가 생길 수 있습니다.

    여기에 (C / C ++ 스타일) 매크로로 수행 할 수있는 많은 간단한 작업을 다른 언어로 다른 방법으로 수행 할 수 있습니다.

  • 다양한 리스프 (Lisp) 방언과 같은 다른 언어에서는 매크로가 핵심 언어 구문과 더 잘 통합되지만 매크로 "누출"의 선언에는 여전히 문제가 발생할 수 있습니다. 이것은 위생적인 매크로에 의해 해결됩니다 .


간략한 역사적 맥락

매크로 (매크로 명령어의 줄임말)는 어셈블리 언어의 맥락에서 처음 등장했습니다. Wikipedia 에 따르면 , 1950 년대에 일부 IBM 어셈블러에서 매크로를 사용할 수있었습니다.

원래 LISP에는 매크로가 없었지만 1960 년대 중반에 처음으로 MacLisp에 도입되었습니다. https://stackoverflow.com/questions/3065606/when-did-the-idea-of-macros-user-defined-code -변환이 나타납니다 . http://www.csee.umbc.edu/courses/331/resources/papers/Evolution-of-Lisp.pdf . 그 전에 "fexprs"는 매크로와 유사한 기능을 제공했습니다.

초기 C 버전에는 매크로가 없었습니다 ( http://cm.bell-labs.com/cm/cs/who/dmr/chist.html ). 이들은 전처리기를 통해 1972-73 년경에 추가되었습니다. 그 전에 C는 #include및 만 지원했습니다 #define.

M4 매크로 전처리 기는 1977 년경에 시작되었습니다.

보다 최근의 언어는 작동 모델이 텍스트가 아닌 구문 적 인 매크로를 구현합니다.

따라서 누군가 "매크로"라는 용어의 특정 정의의 우선 순위 에 대해 이야기 할 때 그 의미가 시간이 지남에 따라 진화했음을 주목하는 것이 중요합니다.


9
C ++은 전 처리기 및 템플릿 확장이라는 두 가지 매크로 시스템을 갖추고 있습니다.
Jeffrey Hantin

나는 템플릿 확장이 매크로라고 주장하면서 매크로의 정의를 확장하고 있다고 생각합니다. 대부분의 현대 언어에는 실제로 (귀하의 정의에 따라) 매크로가 있기 때문에 정의를 사용하면 원래의 질문이 의미가없고 명백히 잘못됩니다. 그러나 대다수의 개발자로 매크로를 사용하면 좋은 질문이됩니다.
Dunk

@Dunk, Lisp에서와 같이 매크로의 정의는 한심한 C 프리 프로세서에서와 ​​같이 "현대적인"바보 같은 이해를 앞섭니다.
SK-logic

@SK : "기술적으로"정확하고 대화를 난독 처리하는 것이 더 좋습니까, 아니면 이해하는 것이 더 낫습니까?
Dunk

1
@ 덩크, 용어는 무지한 무리가 소유하지 않습니다. 그들의 무지를 고려해서는 안됩니다. 그렇지 않으면 컴퓨터 과학의 전체 영역을 무너 뜨릴 것입니다. 누군가가 "매크로"를 C 전처리기에 대한 참조로만 이해한다면 그것은 무지에 지나지 않습니다. 나는 사무실에서 VBA "매크로"인 Lisp 나 심지어 pardonnez mon français에 대해 들어 본 적이없는 사람들의 비율이 의심 스럽다.
SK-logic

12

Scott이 지적한 것처럼 매크로를 사용하면 논리를 숨길 수 있습니다. 물론 함수, 클래스, 라이브러리 및 기타 여러 일반적인 장치도 마찬가지입니다.

그러나 강력한 매크로 시스템은 한층 더 발전하여 일반적으로 사용되지 않는 구문과 구조를 설계하고 활용할 수 있습니다. 도메인 언어, 코드 생성기 등 단일 언어 환경에서 편안하게 사용할 수있는 훌륭한 도구입니다.

그러나 악용 될 수 있습니다. 코드를 읽기, 이해 및 디버깅하기 어렵게 만들고, 새로운 프로그래머가 코드베이스에 익숙해지는 데 필요한 시간을 늘리고, 비용과 실수로 인해 지연 될 수 있습니다.

따라서 프로그래밍 을 단순화 하려는 언어 (예 : Java 또는 Python)의 경우 이러한 시스템이 끔찍합니다.


3
그리고 자바는 foreach, 주석, 어설 션, 제네릭별로 삭제하여 ...
Jeffrey Hantin

2
@Jeffrey : 많은 타사 코드 생성기를 잊지 마십시오 . 다시 한번 생각 해보자.
Shog9

4
Java가 단순하지 않고 단순했던 방법의 또 다른 예.
Jeffrey Hantin

3
@ JeffreyHantin : foreach의 문제점은 무엇입니까?
Casebash

1
@ 덩크 (Dunk)이 비유는 팽팽하지만 ... 언어 확장 성은 플루토늄과 비슷하다고 생각합니다. 구현 비용이 많이 들고, 원하는 모양으로 가공하기가 매우 어렵고, 잘못 사용하면 매우 위험하지만 잘 적용하면 놀랍도록 강력합니다.
Jeffrey Hantin

6

매크로는 일부 상황에서 매우 안전하게 구현 될 수 있습니다. 예를 들어 Lisp에서 매크로는 변환 된 코드를 데이터 구조 (s- 표현식)로 반환하는 함수일뿐입니다. 물론 Lisp는 호모 닉 이라는 사실과 "코드는 데이터" 라는 사실에서 큰 이점을 얻습니다 .

매크로가 쉬운 방법의 예는 예외의 경우에 사용할 기본값을 지정하는이 Clojure 예입니다.

(defmacro on-error [default-value code]
  `(try ~code (catch Exception ~'e ~default-value)))

(on-error 0 (+ nil nil))               ;; would normally throw NullPointerException
=> 0                                   ;l; but we get the default value

그러나 Lisps에서도 일반적인 조언은 "필요하지 않으면 매크로를 사용하지 마십시오"입니다.

호모 닉 언어를 사용하지 않으면 매크로가 훨씬 까다로워지고 다양한 옵션에 모두 함정이 있습니다.

  • 텍스트 기반 매크로 ( 예 : C 프리 프로세서)는 구현이 간단하지만 구문상의 문제를 포함하여 텍스트 형식으로 올바른 소스 구문을 생성해야하기 때문에 올바르게 사용하기가 매우 까다 롭습니다.
  • 매크로 기반 DSLS- 예를 들어 C ++ 템플릿 시스템. 복잡하고 자체적으로 까다로운 구문을 생성 할 수 있으며, 언어 구문 및 시맨틱에 새로운 복잡성을 도입하기 때문에 컴파일러 및 도구 작성자가 올바르게 처리하기가 매우 복잡 할 수 있습니다.
  • AST / 바이트 코드 조작 API ( 예 : Java 리플렉션 / 바이트 코드 생성)는 이론적으로 매우 유연하지만 매우 장황 할 수 있습니다. 매우 간단한 작업을 수행하려면 많은 코드가 필요할 수 있습니다. 3 줄 함수와 동등한 기능을 생성하는 데 10 줄의 코드가 필요한 경우 메타 프로그래밍 노력으로 많은 것을 얻지 못했습니다 ...

더욱이, 매크로가 할 수있는 모든 것은 궁극적으로 튜링 완전한 언어로 다른 방식으로 달성 될 수 있습니다 (이것이 많은 상용구를 작성하는 것을 의미하더라도). 이러한 까다로운 결과로 많은 언어에서 매크로가 실제로 구현하려는 모든 노력의 가치가 없다고 결정하는 것은 놀라운 일이 아닙니다.


어. 더 이상 "코드는 데이터가 좋은 것"쓰레기가 아닙니다. 코드는 코드이고, 데이터는 데이터이며, 둘을 올바르게 분리하지 못하면 보안상의 허점이됩니다. 현존하는 가장 큰 취약성 클래스 중 하나 인 SQL Injection의 출현으로 인해 코드와 데이터를 한 번에 쉽게 교환 할 수 있다는 아이디어가 불신했을 것입니다.
메이슨 휠러

10
@Mason-코드 데이터 개념을 이해하지 못한다고 생각합니다. 모든 C 프로그램 소스 코드도 데이터입니다. 텍스트 형식으로 표시됩니다. Lisp는 컴파일하기 전에 매크로에 의해 조작되고 변환 될 수있는 실제 중간 데이터 구조 (s- 표현)로 코드를 표현한다는 점을 제외하면 동일합니다. 두 경우 모두 신뢰할 수없는 입력을 컴파일러에 보내는 것은 보안상의 허점이 될 수 있지만 실수로하기가 어렵고 컴파일러가 아닌 어리석은 일을하는 것은 당신의 잘못입니다.
mikera

녹은 또 다른 흥미로운 안전한 매크로 시스템입니다. C / C ++ 어휘 매크로와 관련된 문제를 피하기 위해 엄격한 규칙 / 의미가 있으며, DSL을 사용하여 입력 표현식의 일부를 나중에 삽입 할 매크로 변수로 캡처합니다. 입력에서 함수를 호출하는 Lisps의 기능만큼 강력하지는 않지만 다른 매크로에서 호출 할 수있는 유용한 조작 매크로를 제공합니다.
zstewart

어. 더 이상 보안 쓰레기가 아닙니다 . 그렇지 않으면, 폰 노이만 아키텍처가 내장 된 모든 시스템, 예를 들어 자체 수정 코드 기능이있는 모든 IA-32 프로세서를 비난하십시오. 난 당신이 물리적으로 구멍을 채울 수 있는지 여부를 의심 ... 어쨌든, 당신은 일반적으로 사실을 직시해야 : 코드와 데이터 사이의 동형 사상은 세계의 본성 여러 가지 방법 . 그리고 아마도 보안 요구 사항의 불변성을 유지해야 할 의무가있는 프로그래머는 아마도 당신입니다. 아마도 조기 분리를 인위적으로 적용하는 것과 다릅니다.
FrankHB

4

질문에 대답하려면 주로 어떤 매크로가 사용되는지 (경고 : 뇌 컴파일 코드)에 대해 생각해보십시오.

  • 기호 상수를 정의하는 데 사용되는 매크로 #define X 100

이것은 쉽게 다음으로 대체 될 수 있습니다. const int X = 100;

  • 인라인 유형에 관계없이 함수를 정의하는 데 사용되는 매크로 #define max(X,Y) (X>Y?X:Y)

함수 오버로딩을 지원하는 모든 언어에서 올바른 형식의 오버로드 된 함수를 사용하거나 제네릭을 지원하는 언어에서 제네릭 함수를 사용하여 훨씬 더 안전한 형식으로 에뮬레이션 할 수 있습니다. 매크로는 컴파일 할 수있는 포인터 나 문자열을 포함하여 어떤 것도 비교하려고 시도 하지만, 원하는 것은 아닙니다. 반면에 매크로를 안전한 유형으로 만들면 오버로드 된 기능에 비해 이점이나 편의성이 제공되지 않습니다.

  • 자주 사용하는 요소에 대한 바로 가기를 지정하는 데 사용되는 매크로입니다. #define p printf

이것은 p()같은 일 을하는 기능으로 쉽게 대체됩니다 . 이것은 C ( va_arg()함수 패밀리 를 사용해야 함 )와 관련이 있지만 가변 수의 함수 인수를 지원하는 다른 많은 언어에서는 훨씬 간단합니다.

특수 매크로 언어가 아닌 언어 에서 이러한 기능을 지원하는 것이 더 간단하고 오류가 적으며 코드를 읽는 다른 사람들과 혼동되지 않습니다. 실제로 다른 방법으로는 쉽게 복제 할 수없는 매크로 의 단일 사용 사례를 생각할 수 없습니다. 그들이 같은 조건부 컴파일 구조에 연결하는 경우 매크로가 진정으로 유용한 장소입니다 #if(등).

그 시점에서, 나는 대중 언어의 조건부 컴파일에 대한 비 전처리 프로세서 솔루션이 매우 번거 롭다고 믿기 때문에 당신과 논쟁하지 않을 것입니다 (Java의 바이트 코드 주입과 같은). 그러나 D와 같은 언어는 전처리 기가 필요없고 전 처리기 조건을 사용하는 것보다 더 번거롭지 않고 오류가 발생하기 쉬운 솔루션을 제시했습니다.


1
최대 #define 해야하는 경우 매개 변수 주위에 괄호를 넣으십시오. 따라서 연산자 우선 순위에서 예기치 않은 영향이 없습니다 ... # define max (X, Y) ((X)> (Y)? (X) :( Y))
foo

당신은 그것이 단지 예일 뿐이라는 것을 알고 있습니다.
Chinmay Kanchi

이 문제를 체계적으로 다루기 위해 +1. 조건부 컴파일은 쉽게 악몽이 될 수 있다고 덧붙이고 싶습니다. "unifdef"(??)라는 프로그램이 포스트 프로세싱 후 남은 것을 볼 수 있도록하는 프로그램이 있다는 것을 기억합니다.
Ingo

7
In fact, I can't think of a single use-case for macros that can't easily be duplicated in another way: 적어도 C에서는 매크로를 사용하지 않고 토큰 연결을 사용하여 식별자 (변수 이름)를 구성 할 수 없습니다.
Charles Salvia

매크로를 사용하면 특정 코드 및 데이터 구조가 "병렬"로 유지 될 수 있습니다. 예를 들어, 관련 메시지가 적은 조건이 있고 간결한 형식으로 조건을 유지해야하는 enum경우, 조건을 정의하기 위해 및 문자열을 정의하는 상수 문자열 배열을 사용하려고하면 문제가 발생할 수 있습니다. 열거 형과 배열이 동기화되지 않습니다. 매크로를 사용하여 모든 (enum, string) 쌍을 정의한 다음 매번 범위에서 다른 적절한 정의와 함께 해당 매크로를 두 번 포함하면 각 열거 형 값을 문자열 옆에 놓을 수 있습니다.
supercat

2

매크로에서 가장 큰 문제는 매크로를 많이 사용하면 찾기 쉽고 유지하기 어려운 매크로에서 논리를 숨길 수 있기 때문에 코드를 읽고 유지하기가 매우 어렵다는 것입니다. ).


매크로를 문서화해서는 안됩니까?
Casebash

@Casebash : 물론 매크로는 다른 소스 코드와 마찬가지로 문서화해야하지만 실제로는 거의 보지 못했습니다.
Scott Dorman

3
문서를 디버그 한 적이 있습니까? 잘못 작성된 코드를 처리 할 때는 충분하지 않습니다.
JeffO

OOP에도 이런 문제가 있습니다.
aoeu256

2

C / C ++의 MACRO는 매우 제한적이며 오류가 발생하기 쉬우 며 실제로는 유용하지 않다는 점을 염두에두고 시작하지 마십시오.

LISP 또는 z / OS 어셈블러 언어로 구현 된 MACRO는 신뢰할 수 있고 매우 유용 할 수 있습니다.

그러나 C의 제한된 기능을 남용했기 때문에 그들은 나쁜 평판을 얻었습니다. 따라서 아무도 더 이상 매크로를 구현하지 않습니다. 대신 사용되는 간단한 작업 매크로 중 일부를 수행하는 템플릿과 Java의 주석과 같은 작업을 수행하여 더 복잡한 작업을 수행합니다.

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.