성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
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'> <div style='font-family: 맑은 고딕, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>Pet Shop 4.0 - SQL Server Compact Edition Version - 두 번째 이야기</div><br /> <br /> 간단하게 테스트 해볼 수 있는 ASP.NET 예제로써 Pet Shop을 선택했습니다. 이것은 자바 버전의 JPetStore와의 성능 비교를 위해 .NET 버전으로 만들어진 것으로 다음의 경로에서 다운로드 받을 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > Microsoft .NET Pet Shop 4: Migrating an ASP.NET 1.1 Application to 2.0 ; http://msdn.microsoft.com/en-us/library/aa479070.aspx 설치 버전 ; http://download.microsoft.com/download/8/0/1/801ff297-aea6-46b9-8e11-810df5df1032/Microsoft%20.NET%20Pet%20Shop%204.0.msi </pre> <br /> 참고로, Pet Shop 4.0부터 .NET 2.0 버전이고, 그 이전 버전은 .NET 1.x 기반으로 만들어져 있습니다.<br /> <br /> <hr style='width: 50%' /><br /> <br /> 그런데, 간단한 예제에 비해서 SQL Express 설치까지 요구하기 때문에 .NET을 모르는 이들에게는 너무 불편한 것이 사실입니다. 그래서, DB를 SQL Express를 사용하지 않고 파일 기반의 SQL Server Compact Edition을 이용하는 것으로 예제를 변경해야겠다는 생각이 들었습니다.<br /> <br /> 그래서 MDF 파일을 SDF로 변경했던 작업이 지난번에 쓴 글이었지요. ^^<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > Pet Shop 4.0 - SQL Server Compact Edition Version ; <a target='_tab' href='/2/0/812'>http://www.sysnet.pe.kr/2/0/812</a> </pre> <br /> 그 이후에는 Pet Shop 솔루션의 소스 코드를 변경해 주어야 했는데요. 대략 다음과 같은 변경 사항이 있었습니다.<br /> <br /> <div style='font-family: 맑은 고딕, Consolas; font-size: 12pt; color: #2211AA; text-align: left; font-weight: bold'>1. web.config의 DB 연결 문자열 변경</div><br /> <br /> MDF로부터 마이그레이션 해놓은 SDF 파일을 App_Data 폴더에 저장하고, Pet Shop 예제의 web.config에서 연결 문자열을 그에 맞게 조정해야 했습니다. 기본적으로, Pet Shop 예제는 설치 시에 DB 연결을 위한 정보를 묻고, 그에 따라 암호화된 연결 문자열을 다음과 같이 생성하는데,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > <connectionStrings configProtectionProvider="RsaProtectedConfigurationProvider"> <EncryptedData Type="..." xmlns="..."> <EncryptionMethod Algorithm="http://www.w3.org/2001/04/xmlenc#tripledes-cbc"/> <KeyInfo xmlns="http://www.w3.org/2000/09/xmldsig#"> <EncryptedKey xmlns="http://www.w3.org/2001/04/xmlenc#"> <EncryptionMethod Algorithm="http://www.w3.org/2001/04/xmlenc#rsa-1_5"/> <KeyInfo xmlns="http://www.w3.org/2000/09/xmldsig#"> <KeyName>Rsa Key</KeyName> </KeyInfo> <CipherData> <CipherValue>...</CipherValue> </CipherData> </EncryptedKey> </KeyInfo> <CipherData> <CipherValue>...</CipherValue> </CipherData> </EncryptedData> </connectionStrings> </pre> <br /> 위의 정보를 싹 지우고, 아래의 연결 문자열로 대체합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > <connectionStrings> <add name="SQLConnString1" connectionString="<b style='COLOR: blue'>Data Source='|DataDirectory|MSPetShop4.sdf</b>';LCID=;Password=;Encrypt=FALSE;SSCE:Max Database Size=4091;" /> <add name="SQLConnString2" connectionString="Data Source='|DataDirectory|MSPetShop4.sdf';LCID=;Password=;Encrypt=FALSE;SSCE:Max Database Size=4091;" /> <add name="SQLConnString3" connectionString="Data Source='|DataDirectory|MSPetShop4.sdf';LCID=;Password=;Encrypt=FALSE;SSCE:Max Database Size=4091;" /> <add name="SQLProfileConnString" connectionString="D<b style='COLOR: blue'>ata Source='|DataDirectory|MSPetShop4Profile.sdf</b>';LCID=;Password=;Encrypt=FALSE;SSCE:Max Database Size=4091;" /> </connectionStrings> </pre> <br /> <div style='font-family: 맑은 고딕, Consolas; font-size: 12pt; color: #2211AA; text-align: left; font-weight: bold'>2. System.Data.SqlClient에서 제공되는 타입을 System.Data.SqlServerCe에서 제공되는 타입으로 교체</div><br /> <br /> 예를 들면, SqlCommand, SqlConnection... 등의 타입을 각각 SqlCeCommand, SqlCeConnection으로 바꿔주어야 합니다. 바꾸는 김에 가능한 인터페이스를 사용하는 식으로 조정해 주는 것도 좋겠지요. 이를 테면, 아래와 같은 식의 코드입니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > <b style='COLOR: blue'>IDbCommand</b> cmd = new <b style='COLOR: blue'>SqlCeCommand</b>(); <b style='COLOR: blue'>IDbConnection</b> conn = new <b style='COLOR: blue'>SqlCeConnection</b>(connectionString); </pre> <br /> 참고로, 꽤 많은 부분을 바꿔야 합니다.<br /> <br /> <div style='font-family: 맑은 고딕, Consolas; font-size: 12pt; color: #2211AA; text-align: left; font-weight: bold'>3. ASP.NET에서 SQL Server 2005 CE 사용 가능함을 표시</div><br /> <br /> 일단, 위의 2단계까지 변경하고 나서 실행하면 다음과 같은 예외가 발생합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > <b style='COLOR: blue'>SQL Server Compact is not intended for ASP.NET development.</b> Line 119: { Line 120: IDbCommand cmd = new SqlCeCommand(); Line 121: IDbConnection conn = new SqlCeConnection(connectionString); </pre> <br /> 이 대목에서 좌절할 수도 있는데 ^^ 다행히 방법을 찾을 수 있었습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > SQL Server 2005 Compact Edition and ASP.NET ; <a target='_tab' href='http://vidmar.net/weblog/archive/2007/04/19/SQL-Server-2005-Compact-Edition-and-ASP.NET.aspx'>http://vidmar.net/weblog/archive/2007/04/19/SQL-Server-2005-Compact-Edition-and-ASP.NET.aspx</a> </pre> <br /> 그래서, Global.asax에서 다음과 같이 코드를 추가했습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > protected void Application_Start(object sender, EventArgs e) { AppDomain.CurrentDomain.SetData("<b style='COLOR: blue'>SQLServerCompactEditionUnderWebHosting</b>", true); } </pre> <br /> 엄밀히,<br /> 가상 응용 프로그램은 실행되면서 AppDomain을 새롭게 생성해서 매핑시키고, 그 안에서 HttpRuntime이 운영되면서 n개의 HttpApplication 개체가 생성되고 사라지기 때문에 위의 Application_Start는 한 번 이상 수행될 수 있습니다. 따라서, "단 한 번"의 실행을 보장하려면 static 생성자에 놓는 것도 좋겠습니다.<br /> <br /> <div style='font-family: 맑은 고딕, Consolas; font-size: 12pt; color: #2211AA; text-align: left; font-weight: bold'>4. SqlCacheDependency 비활성</div><br /> <br /> Pet Shop 내부에서 SqlCacheDependency를 사용하는데, 기본적으로 CE 버전에는 맞지 않기 때문에 이 부분을 비활성화 시켰습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > TableCacheDependency / TableDependency.cs protected TableDependency(string configKey) { string dbName = ConfigurationManager.AppSettings["CacheDatabaseName"]; string tableConfig = ConfigurationManager.AppSettings[configKey]; string[] tables = tableConfig.Split(configurationSeparator); // 주석 처리 // foreach (string tableName in tables) // dependency.Add(new SqlCacheDependency(dbName, tableName)); } </pre> <br /> <div style='font-family: 맑은 고딕, Consolas; font-size: 12pt; color: #2211AA; text-align: left; font-weight: bold'>5. VarChar ==> NVarChar 변경</div><br /> SQL Server CE 버전은 "N" 버전의 형식만을 지원합니다. 따라서 Char, Text, VarChar는 각각 NChar, NText, NVarChar로 변경해 주어야 합니다. Pet Shop 예제에서는 그중에서 VarChar만 사용하고 있으므로 그것만 NVarChar로 변경해 주면 됩니다.<br /> <br /> 예를 들어, 아래와 같은 SqlCeParameter의 인자들이 그 대상이 됩니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > IDbDataParameter[] parms = { new SqlCeParameter("@Username", <b style='COLOR: blue'>SqlDbType.NVarChar</b>, 256), new SqlCeParameter("@ApplicationName", <b style='COLOR: blue'>SqlDbType.NVarChar</b>, 256), new SqlCeParameter("@LastActivityDate", SqlDbType.DateTime), new SqlCeParameter("@LastUpdatedDate", SqlDbType.DateTime), new SqlCeParameter("@IsAnonymous", SqlDbType.Bit)}; </pre> <br /> <div style='font-family: 맑은 고딕, Consolas; font-size: 12pt; color: #2211AA; text-align: left; font-weight: bold'>5. 다중 SQL 실행문 분리</div><br /> SQL Server CE 버전의 SqlCeCommand는 다중 SQL 문 실행을 지원하지 않습니다. 다행히, Pet Shop에서 이것이 문제되는 경우는 딱 하나였는데,<br /> <br /> <SPAN style='BACKGROUND-COLOR: #ccffcc; FONT-STYLE: italic; MARGIN: 10px 0px 10px 10px; WIDTH: 800px; FONT-FAMILY: 맑은 고딕, Consolas, Verdana; COLOR: #005555'><br /> SQLProfileDAL / PetShopProfileProvider.cs<br /> <br /> public int CreateProfileForUser(string userName, bool isAuthenticated, string appName) <br /> {<br /> string sqlInsert = "INSERT INTO Profiles (Username, ApplicationName, LastActivityDate, LastUpdatedDate, IsAnonymous) Values(@Username, @ApplicationName, @LastActivityDate, @LastUpdatedDate, @IsAnonymous); <b style='COLOR: blue'>SELECT @@IDENTITY;</b>";<br /> }</span><br /><br /> <br /> 위의 마지막 "SELECT @@IDENTITY"를 분리해내야 합니다. 그래서, 다음과 같이 두번 실행하도록 변경했습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > string sqlInsert = "INSERT INTO Profiles (Username, ApplicationName, LastActivityDate, LastUpdatedDate, IsAnonymous) Values(@Username, @ApplicationName, @LastActivityDate, @LastUpdatedDate, @IsAnonymous);"; IDbDataParameter[] parms = { new SqlCeParameter("@Username", SqlDbType.NVarChar, 256), new SqlCeParameter("@ApplicationName", SqlDbType.NVarChar, 256), new SqlCeParameter("@LastActivityDate", SqlDbType.DateTime), new SqlCeParameter("@LastUpdatedDate", SqlDbType.DateTime), new SqlCeParameter("@IsAnonymous", SqlDbType.Bit)}; parms[0].Value = userName; parms[1].Value = appName; parms[2].Value = DateTime.Now; parms[3].Value = DateTime.Now; parms[4].Value = !isAuthenticated; int uniqueID = 0; <b style='COLOR: blue'>SqlHelper.ExecuteNonQuery(SqlHelper.ConnectionStringProfile, CommandType.Text, sqlInsert, parms);</b> string sqlSelect = "SELECT MAX(UniqueID) FROM Profiles"; <b style='COLOR: blue'>string result = SqlHelper.ExecuteScalar(SqlHelper.ConnectionStringProfile, CommandType.Text, sqlSelect, null).ToString();</b> int.TryParse(result, out uniqueID); return uniqueID; } </pre> <br /> 좀 더 정확하게는 트랜잭션으로 묶어야 하겠지만 어차피 실제 서비스로 사용될 예제가 아니기 때문에 더 이상의 변경은 하지 않았습니다.<br /> <br /> <hr style='width: 50%' /><br /> <br /> 여기까지 반영해서 <a target='_tab' href='http://www.sysnet.pe.kr/bbs/DownloadAttachment.aspx?wid=815&boardid=331301885'>소스 코드를 변경한 솔루션을 첨부 파일</a>로 올렸으니 필요하신 분들은 다운로드 하세요. ^^<br /> <br /><br /><hr /><span style='color: Maroon'>[이 토픽에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
1369
(왼쪽의 숫자를 입력해야 합니다.)