성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] A graphical depiction of the steps ...
[정성태] 질문을 주셔서 출판사 측에 문의를 했습니다. 약 한 달 정도 후...
[Thorondor
] @정성태 개인 블로그인데도 거의 커뮤니티 급 인 것 같아요. 요...
[정성태] Roll A Lisp In C - Reading ; https...
[정성태] Java - How to use the Foreign Funct...
[정성태] 제가 큰 실수를 했군요. ^^; Delegate를 통한 Bein...
[정성태] Working with Rust Libraries from C#...
[정성태] Detecting blocking calls using asyn...
[정성태] 아쉽게도, 커뮤니티는 아니고 개인 블로그입니다. ^^
[정성태] 질문이 잘 이해가 안 됩니다. 우선, 해당 소스코드에서 ILis...
글쓰기
제목
이름
암호
전자우편
HTML
홈페이지
유형
제니퍼 .NET
닷넷
COM 개체 관련
스크립트
VC++
VS.NET IDE
Windows
Team Foundation Server
디버깅 기술
오류 유형
개발 환경 구성
웹
기타
Linux
Java
DDK
Math
Phone
Graphics
사물인터넷
부모글 보이기/감추기
내용
<div style='display: inline'> <h1 style='font-family: Malgun Gothic, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>GetPrivateProfileSection / WritePrivateProfileSection의 C# 버전</h1> <p> 상기시킬 겸, 예전에 쓴 INI 파일의 인코딩 관련 문제를 링크 걸어보고 시작하겠습니다. ^^<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > UTF-8 포맷의 INI 파일에 대한 GetPrivateProfile... API 사용 불가 ; <a target='tab' href='http://www.sysnet.pe.kr/2/0/927'>http://www.sysnet.pe.kr/2/0/927</a> </pre> <br /> 최근 C#에서 INI 파일의 Section 영역을 다루다가 한 가지 더 작은 문제가 생겼습니다.<br /> <br /> 보통 INI 파일의 경우 다음과 같이 정보를 저장하게 되는데요.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > [TestSection] Key1 = Value1 Key2 = Value2 </pre> <br /> 가끔, 해당 Section 내에 정의된 Key 이름을 열람하고 싶을 때가 있는데, 아쉽게도 Win32 API 중에서 그런 용도로 제공되는 함수는 없습니다. 단지, 전체 Key/Value 쌍을 반환하는 함수가 있는데, <a target='tab' href='https://docs.microsoft.com/en-us/windows/win32/api/winbase/nf-winbase-getprivateprofilesection'>GetPrivateProfileSection Win32 API</a>가 그런 역할을 합니다. <br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > DWORD WINAPI GetPrivateProfileSection(LPCTSTR lpAppName, LPTSTR lpReturnedString, DWORD nSize, LPCTSTR lpFileName); </pre> <br /> 이 함수는 각각의 Key/Value 쌍을 '\0' NULL 값으로 구분하고 마지막 항목에서 이중 NULL 값을 채움으로써 목록의 마지막임을 알립니다.<br /> <br /> 즉, 위와 같은 경우에 다음과 같은 문자열 값으로 다뤄지게 됩니다. (읽기 함수에서는 마지막 '\0' NULL 문자가 없지만, 쓰기 함수에서는 마지막 '\0' NULL 문자를 처리해 주어야 합니다.)<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Key1=Value1<span style='color: blue; font-weight: bold'>\0</span>Key2=Value2<span style='color: blue; font-weight: bold'>\0\0</span> </pre> <br /> C/C++의 경우에는 어차피 2번째 인자인 "LPTSTR lpReturnedString" 값이 포인터이다 보니 '\0' NULL 값을 제어하는데 아무런 문제가 없지만, C#에서는 NULL 값 제어를 할 수 없으므로 일반적인 System.Text.StringBuilder 개체로 마샬링을 해서는 안됩니다. (물론, 해도 되지만 그렇게 되면 무조건 첫 번째 Key1=Value1 값만을 반환하게 됩니다.)<br /> <br /> 이 문제를 해결하기 위해서는 다음과 같이 그냥 byte[] 값으로 P/Invoke 구문을 만드는 것이 좋습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > [DllImport(kernel32.dll)] private static extern int GetPrivateProfileSection( String appName, byte [] returnValue, int size, String filePath); </pre> <br /> 그리고, 이를 사용해서 다음과 같이 byte[] 배열을 대상으로 '\0' NULL 처리를 할 수 있는 wrapper 함수를 만들어 두면 되겠고.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > static void Main(string[] args) { Dictionary<string, string> keyValues = GetKeyValuesInSection(@"D:\temp\Test.ini", "TestSection"); } private static Dictionary<string, string> GetKeyValuesInSection(string iniPath, string sectionName) { Dictionary<string, string> keyValues = new Dictionary<string, string>(); <span style='color: blue; font-weight: bold'>byte[] returnValue = new byte[32768 * 2]; int byteCount = GetPrivateProfileSection(sectionName, returnValue, 32768 * 2, iniPath);</span> int pos = 0; while (true) { if (pos >= byteCount) { break; } int count = 0; while (true) { if (returnValue[pos + count] == 0) { break; } count++; } string txt = <span style='color: blue; font-weight: bold'>Encoding.Default.GetString(returnValue, pos, count);</span> int equalPos = txt.IndexOf('='); if (equalPos == -1) { keyValues.Add(txt, ""); } else { keyValues.Add(txt.Substring(0, equalPos), txt.Substring(equalPos + 1)); } pos += count + 1; } return keyValues; } </pre> <br /> 위에서 보면 바이트 배열의 크기를 32767로 했는데요. 왜냐하면 MSDN 문서에 다음과 같이 설명되어 있기 때문입니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > GetPrivateProfileSection function ; <a target='tab' href='https://docs.microsoft.com/en-us/windows/win32/api/winbase/nf-winbase-getprivateprofilesection'>https://docs.microsoft.com/en-us/windows/win32/api/winbase/nf-winbase-getprivateprofilesection</a> </pre> <br /> <div style='BACKGROUND-COLOR: #ccffcc; padding: 10px 10px 5px 10px; MARGIN: 0px 10px 10px 10px; FONT-FAMILY: Malgun Gothic, Consolas, Verdana; COLOR: #005555'> nSize [in] <br /> The size of the buffer pointed to by the lpReturnedString parameter, in characters. <span style='color: blue; font-weight: bold'>The maximum profile section size is 32,767 characters</span> </div><br /> <br /> <hr style='width: 50%' /><br /> <br /> 그런데, 왜? 32767에 곱하기 2를 했을까요? 그 해답은 <a target='tab' href='http://www.sysnet.pe.kr/2/0/927'>이 글의 처음에 링크했던 바와 같이 INI 파일의 포맷이 ANSI와 UNICODE를 함께 지원</a>하기 때문입니다. 그렇게 생각하면 위에서 제가 제공해드린 예제 함수(GetKeyValuesInSection)에는 헛점이 너무나 많습니다. 왜냐하면 유니코드의 경우라면 일반 ASCII 문자에 대해서 '\0' NULL 문자가 올 수 있기 때문에 이를 감안해야 하고, 게다가 Encoding.Default로 문자열을 구해와서도 안됩니다.<br /> <br /> 하지만, 위의 함수는 ANSI와 Unicode INI 파일에 대해서 잘 동작합니다. 왜일까요?<br /> <br /> 이 함수를 통해서 재미있는 사실을 알게 되었는데요. 한글 테스트를 위해 다음과 같은 내용의 UNICODE INI 파일을 생성하고,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > [TestSection] Key1=Value1 한글키=한글값 </pre> <br /> 반환되는 바이트 값을 출력해 보았습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > int byteCount = GetPrivateProfileSection(sectionName, returnValue, 32768 * 2, iniPath); #if DEBUG <span style='color: blue; font-weight: bold'> Console.WriteLine("GetPrivateProfileSection: " + byteCount); Console.WriteLine(BitConverter.ToString(returnValue, 0, byteCount)); Console.WriteLine();</span> #endif </pre> <br /> 결과는 매우 의외였습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > GetPrivateProfileSection: 26 4B-65-79-31-3D-56-61-6C-75-65-31-00-C7-D1-B1-DB-C5-B0-3D-C7-D1-B1-DB-B0-AA-00 분석: 4B-65-79-31-3D-56-61-6C-75-65-31-00 k e y 1 = v a l u e 1 \0 C7-D1-B1-DB-C5-B0-3D-C7-D1-B1-DB-B0-AA-00 한 글 키 = 한 글 값 \0 </pre> <br /> 즉, GetPrivateProfileSection 함수는 WBCS(Wide Byte Character Set) 형식으로 반환하지 않고 MBCS(Multi Byte Character Set) 형식으로 반환하는 것이었습니다. 이렇다는 것은 곧, 만약 위의 INI 파일을 일본어 윈도우에서 실행한다면 정상적으로 값을 구할 수 없음을 뜻합니다. 실제로, <a target='tab' href='http://www.sysnet.pe.kr/2/0/762'>Regional Settings를 영문으로만 바꿔도</a> 다음과 같이 한글값이 '?' 문자로 깨져나옵니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > GetPrivateProfileSection: 20 4B-65-79-31-3D-56-61-6C-75-65-31-00-3F-3F-3F-3D-3F-3F-3F-00 분석: 4B-65-79-31-3D-56-61-6C-75-65-31-00 k e y 1 = v a l u e 1 \0 3F-3F-3F-3D-3F-3F-3F-00 ? ? ? = ? ? ? \0 </pre> <br /> 파일은 유니코드 형식인데, Win32 API가 MBCS 방식이라서 이런 현상이 생기는 것입니다. 허긴... 이에 대해서 마이크로소프트를 탓할 만한 것은 아닌 것 같습니다. 왜냐하면, INI 파일을 다루는 Win32 API들은 현재 16비트 윈도우 응용 프로그램을 위한 호환성만을 목적으로 제공된다고 문서에 명시되어 있기 때문입니다.<br /> <br /> 음... 그래도 그렇지... 가끔은 INI 파일에 설정값 저장하는 것이 은근히 편할 때가 있는데... ^^<br /> <br /> <a target='tab' href='http://www.sysnet.pe.kr/bbs/DownloadAttachment.aspx?fid=693&boardid=331301885'>첨부된 파일은 위의 코드를 포함한 예제 프로젝트</a>입니다. GetPrivateProfileSection과 쌍을 이루는 <a target='tab' href='https://docs.microsoft.com/en-us/windows/win32/api/winbase/nf-winbase-writeprivateprofilesectiona'>WritePrivateProfileSection</a>도 래퍼 함수를 구현해 놓았으니 참고하세요. ^^<br /> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
1271
(왼쪽의 숫자를 입력해야 합니다.)