마이크로소프트의 최신 웹 기술의 향연, Remix 2009 행사 공지가 떴습니다.
본사에서 근무하는 시즌이라면 큰 무리 없이 (본사 바로 뒤편의 호텔이라...) 갈 수 있을텐데,
강남에 있다보니 영... 참석이 쉬워보이지가 않군요 ㅜ.ㅜ
* 참석 사전등록 : http://www.visitmix.co.kr/remix09/default.asp
- NoPD -
try { conn.Open(); OracleCommand cmd = new OracleCommand(); cmd.Connection = conn; cmd.CommandText = "SELECT dname FROM dept WHERE deptno = " + textBox1.Text; cmd.CommandType = CommandType.Text; if (dr.Read()) { label1.Text = dr["dname"].ToString(); } } catch (Exception ex) { MessageBox.Show(ex.Message.ToString()); } finally { // 객체 초기화 코드 }이러한 에러 핸들링 방식은 에러 발생시 무리 없이 에러를 처리할 수 있는 구조이긴 하지만, 사용자 친화적인 방식의 에러 처리라고 보기는 어렵습니다. 이러한 코드상에서 에러가 발생하는 경우 아래와 같이 불친절한 메세지가 사용자에게 출력되게 됩니다.
catch (OracleException ex) { switch (ex.Number) { case 1 : MessageBox.Show(" 중복된 데이터를 Insert 하고 있습니다 "); break; case 12545 : MessageBox.Show(" 데이터베이스를 사용할 수 없습니다 "); break; ... ... default : MessageBox.Show(" 처리되지 않은 데이터베이스 에러가 발생했습니다 : " + ex.Message.ToString(); break; } } catch (Exception ex) { MessageBox.Show(ex.Message.ToString()); }보다 상세하게 처리 로직을 추가한 위의 코드를 보면 catch 문이 2개가 사용된 것을 볼 수 있습니다. 오라클과 관련된 에러를 핸들링하는 catch 구문과 일반적인 에러를 핸들링하는 catch 구문이 그것입니다. 이와 같이 에러 메세지를 처리해 주면 사용자에게 보다 편안한 에러 메세지를 제공해 줄 수 있을 것입니다.
* 이 글은 오라클 기술 네트워크(Oracle Technology Network)에 공개된 Joh Paul Cook 의 영문 아티클을 번역 / 의역한 글입니다. 원문은 링크(http://tinyurl.com/m69mvh)를 통하여 확인하실 수 있습니다.
Visual C# 구문
프로젝트에 참조 추가를 한 다음 using 구문을 이용하여 네임스페이스를 선언해 주어야 합니다. 기술적으로 네임스페이스 선언은 해주지 않아도 관계 없지만, 간결한 코드를 위해서는 꼭 선언해 주는 것이 좋습니다. 일반적인 경우와 마찬가지로 네임스페이스나 클래스 선언이 시작되기 전에 using 문을 추가해 주면 됩니다.
using Oracle.DataAccess.Client참조가 정상적으로 추가되었다면 using 문으로 네임스페이스를 선언하는 동안 인텔리센스의 도움을 받으실 수 있었을 것입니다.
XE = (DESCRIPTION = (ADDRESS_LIST = (ADDRESS = (PROTOCOL = TCP)(HOST = DBSVR)(PORT = 1521)) ) (CONNECT_DATA = (SERVICE_NAME = XE) ) )XE 라는 약어는 오라클 클라이언트를 위한 데이터베이스 연결 주소 정보를 정의하고 있습니다. tnsnames.ora 파일에 정의된 위의 약어를 사용하기 위해서 비주얼 스튜디오에서 아래와 같은 연결 문자열 선언을 해야합니다.
string oradb = "Data Source=XE;User ID=scott;Password=tiger;";물론 tnsnames.ora 파일을 사용하지 않기 위하여 연결 문자열을 아래와 같이 변경할 수도 있습니다. tnsnames.ora 파일에 지정된 내용을 그대로 연결 문자열에 정의하면 됩니다.
string oradb = "Data Source=(DESCRIPTION =(ADDRESS_LIST =(ADDRESS = (PROTOCOL = TCP)(HOST = DBSVR)(PORT = 1521))) " + "(CONNECT_DATA =(SERVICE_NAME = XE)));" + "User Id=scott;Password=tiger;";위에서 볼 수 있는 것처럼, 사용자 이름과 패스워드는 연결 문자열 속에 평문으로 저장되어 있습니다. 이 방법은 연결 문자열을 만드는 가장 간단한 방법이지만 보안의 관점에서 봤을 때는 평문으로 저장된 사용자 이름, 패스워드는 좋은 선택이 아닙니다. 특히, 닷넷 어플리케이션은 DLL, EXE 를 막론하고 디스 어셈블러를 통해서 소스 코드의 복원이 가능하기 때문에 보안에 취약합니다. 이와 관련한 내용은 다른 글에서 보다 자세히 다루도록 하겠습니다.
OracleConnection conn = new OracleConnection(XE);연결 문자열의 할당은 오버로드된 객체의 생성자를 통해서 전달되어 할당이 가능합니다. 생성자의 다른 오버로드는 아래와 같은 구문도 사용 가능하도록 정의되어 있습니다.
OracleConnection conn = new OracleConnection(); conn.ConnectionString = oradb;연결 문자열의 할당이 끝나면 이제 남은 일은 커넥션 객체의 Open 메소드를 이용하여 오라클 데이터베이스와의 실제 연결을 만드는 작업을 하면 됩니다. 이 과정에서의 에러 핸들링은 다음 포스트에서 소개하도록 하겠습니다.
conn.Open();Command 객체
string sql = "SELECT dname FROM dept WHERE deptno = 10"; OracleCommand cmd = new OracleCommand(sql, conn); cmd.COmmandType = COmmandType.Text;오버로드된 다른 생성자를 사용하면 구문은 조금 다르게 변할 수 있습니다. Command 객체는 커맨드 텍스트를 실행할 수 있는 메소드를 가지고 있습니다. 이 메소드들은 서로 다른 형태의 SQL 구문에 적절한 메소드 들입니다.
OracleDataReader dr = cmd.ExecuteReader(); dr.Read(); label1.Text= dr["dname"].ToString(); label1.Text = dr.ZGetString(0).ToString(); label1.Text = dr.GetOracleString(0).ToString();위의 예제에서 우리는 DNAME 이라는 문자열 형태의 리턴 값을 가지고 데이터를 핸들링 하고 있습니다. 그러나 DEPTNO (부서번호) 와 같은 문자열이 아닌 데이터를 핸들링 할 떄는 타입 미스매치(Type Mismatch)에 주의해야 합니다. 닷넷 런타임 라이브러리는 암시적으로 적절한 형 변환을 자동으로 시도합니다. 그러나 간혹 데이터 형의 호환성에 문제가 있거나 암시적인 형 변환이 실패하면 예외가 발생하게 됩니다. 암시적인 형 변환이 에러 없이 수행된다 하더라도 필요한 경우에 대하여 명시적으로 형 변환을 하는 것을 권장합니다.
conn.Close(); conn.Dispose();코드를 만들 때 using 문을 이용하여 로직을 구성한 경우에는 명시적으로 Close 나 Dispose 메소드를 호출할 필요가 없습니다. 아래의 코드는 using 문을 이용하여 데이터베이스를 연결하는 예제 코드입니다.
using (OracleConnection conn = new OracleConnection(oradb)) { conn.Open(); OracleCommand cmd = new OracleCOmmand(); cmd.Connection = conn; cmd.CommandText = "select dname from dept where deptno = 10"; cmd.COmmandType = COmmandType.Text; OracleDataReader dr = cmd.ExecteReader(); dr.Read(); label1.Text = dr.GetString(0); }추가적으로 OracleCommand 객체는 Dispose 메소드를 포함하고 있고 OracleDataReader 객체는 Close 와 Dispose 메소드를 포함하고 있습니다. 일반적인 경우에는 객체의 생성과 소멸이 큰 영향을 주지 않을 수 있지만 닷넷 환경에서 시스템의 리소스를 해제하고 어플리케이션의 성능 향상을 위해서 반드시 신경써 주어야 할 부분입니다.
* 이 글은 오라클 기술 네트워크(Oracle Technology Network)에 공개된 Joh Paul Cook 의 영문 아티클을 번역 / 의역한 글입니다. 원문은 링크(http://tinyurl.com/m69mvh)를 통하여 확인하실 수 있습니다.
글을 시작하며
마이크로소프트의 개발 프레임워크인 닷넷 프레임워크의 인기가 높아짐에 따라 많은 개발자들이 닷넷 어플리케이션과 오라클과의 통합을 위한 가장 좋은 방법에 대한 정보를 갈구하고 있습니다. 기본적인 연결에 대한 것 뿐아니라 비주얼 스튜디오 2005 또는 2008을 이용하여 효과적이고 효율적인 개발을 하는 방법에 대한 요구가 무척 많습니다.
이 글을 통해서 오라클 데이터베이스를 사용하는 닷넷 어플리케이션을 만드는 기본적이지만 필수적인 과정에 대한 설명을 하고자 합니다.
여러분들은 비교적 쉬운 것부터 복잡한 것까지 준비된 3개의 예제를 통해 이 같은 내용을 적용해 볼 수 있을 것입니다. 글에 포함되어 있는 모든 스크린 샷은 비주얼 스튜디오 2008의 화면들 입니다만, 비주얼 스튜디오 2005 에서도 크게 다르지 않은 화면을 보게 될 것입니다.
.NET Data Provider
닷넷으로 오라클 연계 어플리케이션을 개발하기 위해서 기본적인 오라클 클라이언트(Oracle Client) 뿐만 아니라 관리되는 데이터 프로바이더(Managed Data Provider)가 필요합니다. 데이터 프로바이더는 우리가 만들게 될 닷넷 어플리케이션 코드와 오라클 클라이언트 연결 소프트웨어 사이에 위치한 계층입니다. 대부분의 경우 닷넷 OLE DB 데이터 프로바이더가 아닌, 개별 데이터베이스 패키지에 최적화된 데이터 프로바이더를 사용할 때 어플리케이션은 최적의 성능을 발휘하게 됩니다.
오라클, 마이크로소프트 뿐만 아니라 서드 파티 벤더들은 오라클에 최적화된 데이터 프로바이더를 제공하고 있습니다. 오라클과 마이크로소프트는 무료로 이같은 데이터 프로바이더를 제공하고 있습니다. (마이크로소프트의 닷넷 프레임워크 2.0에 데이터 프로바이더가 포함되어 있긴 하지만, 여전히 오라클 클라이언트를 필요로 합니다) 이 글에서는, 마이크로소프트가 닷넷 프레임워크에 제공하는 데이터 프로바이더 대신, 오라클이 공급하는 Oracle Data Provider for .NET (이하 ODP.NET) 을 사용할 예정입니다. ODP.NET 은 오라클 데이터베이스 혹은 오라클이 제공하는 별도의 다운로드 경로(http://tinyurl.com/ndouwh)를 통해서 다운로드 받을 수 있습니다.
ODP.NET 은 표준 ADO.NET 데이터 엑세스를 제공할 뿐만 아니라, XML DB, 데이터 엑세스 성능 최적화, RAC(Real Application Clusters) 커넥션 풀링과 같은 오라클만의 특화된 기능들을 사용할 수 있도록 해줍니다.
ODP.NET 과 오라클 클라이언트 소프트웨어가 설치되었다면, 비주얼 스튜디오를 통한 오라클 연계 어플리케이션 개발을 할 준비가 완료되었습니다. 개발을 시작하기 전에 오라클 데이터베이스와의 연결 상태를 확인하는 것은 좋은 생각입니다. 오라클 클라이언트 소프트웨어에 기본적으로 제공되는 SQL*Plus 와 같은 도구를 이용해서 데이터베이스 연결이 잘 되는지 테스트 해보시기 바랍니다.
비주얼 스튜디오 2008 에서 프로젝트 생성하기
지금부터 오라클 데이터베이스에서 데이터를 추출하는 ODP.NET 어플리케이션을 만들어 보도록 하겟습니다. 에러 핸들링과 관련한 부분은 글 후반부에 다루도록 하겠습니다. 비주얼 스튜디오를 실행한 후, 가장 먼저 해야 할 일은 프로젝트를 생성하는 것입니다. File 메뉴의 New > Project 를 선택하여 프로젝트를 생성하겠습니다.
참조 추가 방법 #1
참조 추가 방법 #2
참조 추가 다이얼로그가 출력되면 "Oracle.DataAccess" 컴포넌트를 찾아서 확인 버튼을 누르면 됩니다. 설치된 오라클 클라이언트 버전에 따라 버전 정보, 런타임 정보가 조금 상이할 수 있습니다. 가능하면 최신 버전의 런타임을 사용하는 것이 좋지만, 일반적인 경우 런타임 버전에 따라 생기는 큰 차이점은 없습니다.