성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] Detecting blocking calls using asyn...
[정성태] 아쉽게도, 커뮤니티는 아니고 개인 블로그입니다. ^^
[정성태] 질문이 잘 이해가 안 됩니다. 우선, 해당 소스코드에서 ILis...
[양승조
] var대신 dinamic으로 선언해서 해결은 했습니다. 맞는 해...
[양승조
] 또 막혔습니다. ㅠㅠ var list = props[i].Ge...
[양승조
] 아. 감사합니다. 어제는 안됐던것 같은데....정신을 차려야겠네...
[정성태] "props[i].GetValue(props[i])" 코드에서 ...
[정성태] 저렇게 조각 코드 말고, 실제로 재현이 되는 예제 프로젝트를 압...
[정성태] Modules 창(Ctrl+Shift+U)을 띄워서, 해당 Op...
[정성태] 만드실 수 있습니다. 단지, Unity 엔진 내의 스크립트와 W...
글쓰기
제목
이름
암호
전자우편
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'>.NET Framework 4.7.1의 새 기능 - Configuration builders</h1> <p> .NET 4.7.1에 추가된 기능 중에,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Announcing the .NET Framework 4.7.1 ; <a target='tab' href='https://blogs.msdn.microsoft.com/dotnet/2017/10/17/announcing-the-net-framework-4-7-1/'>https://blogs.msdn.microsoft.com/dotnet/2017/10/17/announcing-the-net-framework-4-7-1/</a> </pre> <br /> Configuration builders라는 것이 있는데 이에 대한 소개를 해보겠습니다. ^^<br /> <br /> <hr style='width: 50%' /><br /> <br /> 간혹, App.config의 appSettings에 값을 보관하는 경우가 있습니다. 문제는, 이 값이 너무 정적이다는 것입니다. 예를 들어, 다음과 같이 app.config에 값을 정의했을 때,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > <?xml version="1.0" encoding="utf-8" ?> <configuration> <appSettings> <span style='color: blue; font-weight: bold'><add key="Setting1" value="1" /> <add key="Setting2" value="2" /></span> </appSettings> </configuration> </pre> <br /> 각각의 값을 다음과 같은 식으로 AppSettingsReader를 통해 읽을 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > using System; using System.Configuration; namespace ConsoleApp1 { class Program { static void Main(string[] args) { System.Configuration.AppSettingsReader reader = new System.Configuration.AppSettingsReader(); PrintValue(reader, "Setting1"); PrintValue(reader, "Setting2"); PrintValue(reader, "my"); } private static void PrintValue(AppSettingsReader reader, string key) { string txt1 = null; try { txt1 = reader.GetValue(key, typeof(string)) as string; } catch { Console.WriteLine($"'{key}' is not defined"); return; } Console.WriteLine($"'{key}' == {txt1}"); } } } // 출력 결과 /* 'Setting1' == 1 'Setting2' == 'my' is not defined */ </pre> <br /> 그런데 만약, <a target='tab' href='https://www.codeproject.com/Articles/1216422/Modern-Configuration-for-ASP-NET-with-Configuratio'>"Setting1"이라는 값이 "환경 변수"에 정의되어 있다면 app.config이 아닌 환경 변수의 값을 활용하고 싶다</a>면 어떻게 해야 할까요? 물론, 그냥 AppSettingsReader로 값을 읽기 전 환경 변수를 한번 더 체크하면 됩니다.<br /> <br /> .NET 4.7.1부터는 AppSettingsReader에 그런 부가적인 작업을 할 필요 없이 동적으로 app.config의 값을 추가/변경/삭제할 수 있는 기능을 제공하는데 그것이 바로 "Configuration Builder"입니다. 이를 위해 필요한 작업은 app.config에 builder 클래스를 알리는 정도입니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > <?xml version="1.0" encoding="utf-8" ?> <configuration> <span style='color: blue; font-weight: bold'><configSections> <section name="configBuilders" type="System.Configuration.ConfigurationBuildersSection, System.Configuration, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a" restartOnExternalChanges="false" requirePermission="false"/> </configSections></span> <span style='color: blue; font-weight: bold'><configBuilders> <builders> <add name="MyConfigBuilder" type="ConsoleApp1.MyConfigBuilder, ConsoleApp1" /> </builders> </configBuilders></span> <appSettings <span style='color: blue; font-weight: bold'>configBuilders="MyConfigBuilder"</span>> <add key="Setting1" value="May Be Replaced" /> <add key="Setting2" value="May Be Removed" /> <!-- Setting3 could be added by a builder without even being declared here. --> </appSettings> <startup> <supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.7.1" /> </startup> </configuration> </pre> <br /> 물론, Configuration Builder 역할을 하는 사용자 클래스는 구현해야 합니다. 이론 상으로는 해당 클래스가 GAC에 등록한 어셈블리에 있다면 기존 응용 프로그램의 코드는 전혀 변경하지 않아도 되지만, 그렇지 않은 경우라면 자신의 프로젝트에 다음과 같은 클래스를 정의해 제공해야 합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > using System.Collections.Specialized; using System.Configuration; using System.Xml; namespace ConsoleApp1 { public class MyConfigBuilder : ConfigurationBuilder { protected MyConfigBuilder() { } public override void Initialize(string name, NameValueCollection config) { // name == "MyConfigBuilder" base.Initialize(name, config); } public override ConfigurationSection ProcessConfigurationSection(ConfigurationSection configSection) { return base.ProcessConfigurationSection(configSection); } public override XmlNode ProcessRawXml(XmlNode rawXml) { return base.ProcessRawXml(rawXml); } } } </pre> <br /> 위의 클래스는 기본 제공 코드일 뿐, 원한다면 다음과 같이 app.config의 configBuilders 속성이 추가된 XML 노드를 직접 제어할 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > public override XmlNode ProcessRawXml(XmlNode rawXml) { XmlNode node = rawXml.SelectSingleNode("./add[@key = 'Setting1']"); SetAttrValue(node, "value", "Replaced"); node = rawXml.SelectSingleNode("./add[@key = 'Setting2']"); rawXml.RemoveChild(node); AppendAddNode(rawXml, "my", "value"); return base.ProcessRawXml(rawXml); } private void AppendAddNode(XmlNode rawXml, string addName, string addValue) { XmlNode addNode = rawXml.OwnerDocument.CreateNode(XmlNodeType.Element, "add", rawXml.NamespaceURI); XmlAttribute keyNode = rawXml.OwnerDocument.CreateAttribute(rawXml.Prefix, "key", rawXml.NamespaceURI); keyNode.Value = "my"; XmlAttribute valueNode = rawXml.OwnerDocument.CreateAttribute(rawXml.Prefix, "value", rawXml.NamespaceURI); valueNode.Value = "value"; addNode.Attributes.SetNamedItem(keyNode); addNode.Attributes.SetNamedItem(valueNode); rawXml.AppendChild(addNode); } private void SetAttrValue(XmlNode node, string attrName, string attrValue) { XmlNode attrNode = node.Attributes.GetNamedItem(attrName); attrNode.Value = attrValue; } </pre> <br /> 위의 코드에서는 "Setting1"의 값을 변경하고, "Setting2"의 설정은 제거한 후 새롭게 "my"라는 설정을 추가하고 있습니다. 따라서 기존 응용 프로그램을 실행하면 다음과 같은 결과를 볼 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > 'Setting1' == Replaced 'Setting2' is not defined 'my' == value </pre> <br /> (<a target='tab' href='http://www.sysnet.pe.kr/bbs/DownloadAttachment.aspx?fid=1244&boardid=331301885'>첨부 파일은 이 글의 예제 코드를 포함</a>합니다.)<br /> <br /> <hr style='width: 50%' /><br /> <br /> github에 소스 코드 공개 문화가 생기다 보니, 간혹 보면 개발자 PC에서 appSettings에 Facebook 등의 인증 정보를 넣고 테스트하다가 실수로 github 등에 올리는 문제가 있는데요. configBuilders를 잘 활용하면 그런 실수를 미연에 방지할 수 있습니다.<br /> <br /> 마지막으로 살짝 아쉬운 것이 있다면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > With Configuration Builders, applications can apply a custom-defined set of builders <span style='color: blue; font-weight: bold'>to any section of config</span>. </pre> <br /> "any section"을 어떻게 해석해야 할지 모르겠는데 분명한 것은 "all of"는 아니고 "some of"... 도 아니고 "a few of" 정도라는 점입니다. 일단은 appSettings, connectionStrings에 대해 제공하는 것을 확인했고 그 외의 노드는 Visual Studio의 app.config 편집 시에 인텔리센스로 "configBuilders"가 지원되는 지 여부로 쉽게 확인할 수 있습니다.<br /> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
1235
(왼쪽의 숫자를 입력해야 합니다.)