Microsoft MVP성태의 닷넷 이야기
VC++: 37. XmlCodeGenerator를 C/C++ 코드 생성에 적용 [링크 복사], [링크+제목 복사],
조회: 39836
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 3개 있습니다.)

XmlCodeGenerator를 C/C++ 코드 생성에 적용


XmlCodeGenerator에 대해서는 가끔씩 설명을 드렸지요. (개인적으로, 사실 엄청 잘 쓰고 있는 자작 툴입니다. ^^)

XML/XSLT로 구현하는 매크로 확장 
; https://www.sysnet.pe.kr/2/0/542

XmlCodeGenerator 1.0.0.4 업데이트
; https://www.sysnet.pe.kr/2/0/760

아쉬운 점이 하나 있다면, 이런 CodeGenerator 사용자 정의 툴이 Managed 언어를 다루는 프로젝트에서만 적용된다는 점입니다. 굳이 이런 제한을 둘 필요가 있었을까 싶지요!

오늘은, 문득 이에 대한 해결책이 생각났습니다.

간단합니다. 코드 생성하는 파일들은 그냥 C# 프로젝트에 넣어버리고, C++ 프로젝트에서는 상대 경로를 include하는 것으로 해당 코드들을 그냥 포함해 버리는 것입니다. 이렇게 되면, 심지어 소스 컨트롤을 연결해서 관리하는 것에도 문제가 없습니다. 실제로 한번 따라해 볼까요? ^^




1. XML 파일 생성


자신의 코드에서 복잡하게/반복적으로 사용되는 코드를 간단하게 관리할 수 있는 XML 파일을 만듭니다.

<?xml version="1.0" encoding="utf-8" ?>

<people>
  
  <person name="Tester" age="15" />
  <person name="Administrator" age="17" />
  
</people>

2. XSLT 파일 생성


XML 파일의 내용을 C/C++ 코드로 만들어 줄 XSLT 파일을 만듭니다.

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0"
  xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
  xmlns:xs="http://www.w3.org/2001/XMLSchema"
  xmlns:msxsl="urn:schemas-microsoft-com:xslt"
  xmlns:xcg="https://www.sysnet.pe.kr/WPFClassMemberDef"
>
    <xsl:param name="XCG_CurrentTime"/>
    <xsl:output method="text" encoding="utf-8" indent="yes"></xsl:output>

    <xsl:template match="people">

/*
Created: <xsl:value-of select="$XCG_CurrentTime" />
*/

#if defined (HEADER_DECLARATION)

#if !defined __CPP_ARRAY_H
#define __CPP_ARRAY_H

typedef struct tagPerson
{
  wstring Name;
  int Age;
} Person;

extern vector <xsl:text disable-output-escaping="yes">&lt;</xsl:text>Person *<xsl:text disable-output-escaping="yes">&gt;</xsl:text> g_people;
void InitializePeople();

#endif

#endif

#if defined (IMPLEMENTATION_DEFINITION)

vector <xsl:text disable-output-escaping="yes">&lt;</xsl:text>Person *<xsl:text disable-output-escaping="yes">&gt;</xsl:text> g_people;

void InitializePeople()
{
  <xsl:for-each select="person">
    {
      Person *pItem = new Person();
      pItem-<xsl:text disable-output-escaping="yes">&gt;</xsl:text>Name = L"<xsl:value-of select="@name" />";
      pItem-<xsl:text disable-output-escaping="yes">&gt;</xsl:text>Age = <xsl:value-of select="@age" />;
      g_people.push_back(pItem);
    }
  </xsl:for-each>
}

#endif
  </xsl:template>

</xsl:stylesheet>


3. XmlCodeGenerator 적용


생성한 XML 파일과 XSLT 파일을 C# 프로젝트에 추가하고, XML 파일의 "Custom Tool" 속성을 "XmlCodeGenerator"로 지정해 줍니다.

[그림 1: XmlCodeGenerator 적용]
cpp_xmlcodegen_1.png

그러면, "그림 1"에서 보는 것처럼 XML 파일 하위에 .cs 파일이 생성됩니다. CS 파일의 내용은 C/C++에서 사용될 수 있는 코드로 다음과 같이 생성됩니다.

/*
Created: 09/13/2009 21:19:05
*/

#if defined (HEADER_DECLARATION)

#if !defined __CPP_ARRAY_H
#define __CPP_ARRAY_H

typedef struct tagPerson
{
  wstring Name;
  int Age;
} Person;

extern vector <Person *> g_people;
void InitializePeople();

#endif

#endif

#if defined (IMPLEMENTATION_DEFINITION)

vector <Person *> g_people;

void InitializePeople()
{
  
    {
      Person *pItem = new Person();
      pItem->Name = L"Tester";
      pItem->Age = 15;
      g_people.push_back(pItem);
    }
  
    {
      Person *pItem = new Person();
      pItem->Name = L"Administrator";
      pItem->Age = 17;
      g_people.push_back(pItem);
    }
  
}

#endif

매크로 상수를 이용해서 하나의 CS 파일 안에 C/C++ 헤더 파일과 코드에서 사용될 수 있도록 하고 있습니다.

C# 프로젝트에서는 위의 CS 파일이 빌드되면 오류가 발생하기 때문에 CS 파일에 대해서는 컴파일 옵션을 "None"으로 해줍니다.

[그림 2: Build Action - None 적용]
cpp_xmlcodegen_2.png

4. 생성된 CS 파일을 C/C++ 프로젝트에서 사용


이제, C/C++ 프로젝트에서 #include를 이용해서 해당 파일들을 포함시켜 줍니다. 간단하게는 stdafx.h/stdafx.cpp 파일에서 각각 다음과 같이 추가시켜주면 되겠지요. ^^

======== stdafx.h ==========

#pragma once

#include "targetver.h"

#include <stdio.h>
#include <tchar.h>

#include <string>
#include <vector>

using namespace std;

#define HEADER_DECLARATION
#include "..\CppTestCodeGenerator\CppArray.cs" 


======== stdafx.cpp ========

#include "stdafx.h"

#undef HEADER_DECLARATION
#define IMPLEMENTATION_DEFINITION
#include "..\CppTestCodeGenerator\CppArray.cs" 




이제부터는 Person이라는 코드를 관리하기 위해 C/C++ 코드를 찾아가서 수정할 필요 없이 해당 XML 파일을 수정함으로써 편리하게 코드에 결과를 반영할 수 있습니다. 위에서는 단순하게 예를 들었지만, 복잡하게/반복적으로 초기화되는 코드를 위의 자동 생성 코드로 치환해 두면 자칫 실수할 수 있는 오류 횟수를 많이 줄일 수 있습니다.

사실... 시간이 지났을 때, C/C++ 코드를 다시 보면서 코드를 추가하는 것보다 XML로 추가하는 것이 더욱 직관적이죠. 초보적인 수준이긴 하지만 이것도 DSL(Domain Specific Language)이라고 할 수 있지 않을까요!

첨부된 파일은 위의 예제를 테스트한 프로젝트입니다.



[이 토픽에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 4/11/2022]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 



2009-11-25 04시23분
[이유진] 프로젝트 실행을 해봤는데 명령 프롬포트 창이 한번 켜졌다 사라지고 끝이네요; 원래 이런가요?^^
[guest]
2009-11-26 09시38분
첨부된 프로젝트 파일을 실행시키셨나요? 사실 그건 별로 의미가 없는데요. ^^ 위의 글이 의미가 있는 것은, 프로젝트에 포함된 XML 파일을 Visual Studio 에서 변경하고 저장했을 때 CS 파일이 덩달아 업데이트 된다는 점입니다. 글의 도입에 소개한 "XML/XSLT 로 구현하는 매크로 확장" 과 "XmlCodeGenerator 1.0.0.4 업데이트" 편을 읽어보시면 이해가 되실 것입니다.
kevin25

... 61  62  63  64  65  66  [67]  68  69  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12264정성태7/9/202028290오류 유형: 628. docker: Error response from daemon: Conflict. The container name "..." is already in use by container "...".
12261정성태7/9/202019494VS.NET IDE: 148. 윈도우 10에서 .NET Core 응용 프로그램을 리눅스 환경에서 실행하는 2가지 방법 - docker, WSL 2 [5]
12260정성태7/8/202017314.NET Framework: 926. C# - ETW를 이용한 ThreadPool 스레드 감시파일 다운로드1
12259정성태7/8/202016480오류 유형: 627. nvlddmkm.sys의 BAD_POOL_HEADER BSOD 문제 [1]
12258정성태7/8/202020328기타: 77. DataDog APM 간략 소개
12257정성태7/7/202016626.NET Framework: 925. C# - ETW를 이용한 Monitor Enter/Exit 감시파일 다운로드1
12256정성태7/7/202017767.NET Framework: 924. C# - Reflection으로 변경할 수 없는 readonly 정적 필드 [4]
12255정성태7/6/202018543.NET Framework: 923. C# - ETW(Event Tracing for Windows)를 이용한 Finalizer 실행 감시파일 다운로드1
12254정성태7/2/202017056오류 유형: 626. git - REMOTE HOST IDENTIFICATION HAS CHANGED!
12253정성태7/2/202019550.NET Framework: 922. C# - .NET ThreadPool의 Local/Global Queue파일 다운로드1
12252정성태7/2/202021319.NET Framework: 921. C# - I/O 스레드를 사용한 비동기 소켓 서버/클라이언트파일 다운로드2
12251정성태7/1/202020084.NET Framework: 920. C# - 파일의 비동기 처리 유무에 따른 스레드 상황 [1]파일 다운로드2
12250정성태6/30/202022256.NET Framework: 919. C# - 닷넷에서의 진정한 비동기 호출을 가능케 하는 I/O 스레드 사용법 [1]파일 다운로드1
12249정성태6/29/202017767오류 유형: 625. Microsoft SQL Server 2019 RC1 Setup - 설치 제거 시 Warning 26003 오류 발생
12248정성태6/29/202015575오류 유형: 624. SQL 서버 오류 - service-specific error code 17051
12247정성태6/29/202017485.NET Framework: 918. C# - 불린 형 상수를 반환값으로 포함하는 3항 연산자 사용 시 단축 표현 권장(IDE0075) [2]파일 다운로드1
12246정성태6/29/202018654.NET Framework: 917. C# - USB 관련 ETW(Event Tracing for Windows)를 이용한 키보드 입력을 감지하는 방법
12245정성태6/24/202019087.NET Framework: 916. C# - Task.Yield 사용법 (2) [2]파일 다운로드1
12244정성태6/24/202019085.NET Framework: 915. ETW(Event Tracing for Windows)를 이용한 닷넷 프로그램의 내부 이벤트 활용 [1]파일 다운로드1
12243정성태6/23/202015529VS.NET IDE: 147. Visual C++ 프로젝트 - .NET Core EXE를 "Debugger Type"으로 지원하는 기능 추가
12242정성태6/23/202017072오류 유형: 623. AADSTS90072 - User account '...' from identity provider 'live.com' does not exist in tenant 'Microsoft Services'
12241정성태6/23/202020242.NET Framework: 914. C# - Task.Yield 사용법파일 다운로드1
12240정성태6/23/202021786오류 유형: 622. 소켓 바인딩 시 "System.Net.Sockets.SocketException: An attempt was made to access a socket in a way forbidden by its access permissions" 오류 발생
12239정성태6/21/202019868Linux: 30. (윈도우라면 DLL에 속하는) .so 파일이 텍스트로 구성된 사례 [1]
12238정성태6/21/202018204.NET Framework: 913. C# - SharpDX + DXGI를 이용한 윈도우 화면 캡처 라이브러리
12237정성태6/20/202017759.NET Framework: 912. 리눅스 환경의 .NET Core에서 "test".IndexOf("\0")가 0을 반환
... 61  62  63  64  65  66  [67]  68  69  70  71  72  73  74  75  ...