Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 3개 있습니다.)

닷넷에서 접근해보는 InterSystems의 IRIS Data Platform 데이터베이스

처음 듣는 데이터베이스입니다. ^^ 아래의 경로를 통해 community 버전을 다운로드 신청할 수 있고,

InterSystems IRIS Community Edition Download Registration
; http://download.intersystems.com/download/register.csp

곧바로 "IRIS_Community-2020.1.0.215.0-win_x64.exe" 식의 파일을 내려받아 설치하면 되는데, 제 경우에는 그냥 기본값으로 진행을 했습니다.

iris_db_install_1.png

이후, "시작" 메뉴를 통해 "Management Portal [IRIS]" 항목을 선택하면 웹 브라우저가 뜨고 관리 페이지로 접속이 됩니다.




DB 설치는 그게 끝이고, 이후 간단한 실습을 위해 테이블을 생성해야 하는데,

InterSystems IRIS Data Platform 2020.2 > Multi-model Data Access > InterSystems SQL Reference > SQL Commands > CREATE TABLE
; https://docs.intersystems.com/irislatest/csp/docbook/DocBook.UI.Page.cls?KEY=RSQL_createtable

어차피 생소한 DB라도 ANSI 규약의 SQL 쿼리 구문은 기본적으로 지원할 것이므로 해당 데이터베이스 문맥에서,

iris_db_install_2.png

실행해 주면 됩니다.

CREATE TABLE Employee (
     EMPID INT NOT NULL,
     NAME   CHAR(100) NOT NULL,
     AGE  INT,
     CONSTRAINT EMPLOYEEPK PRIMARY KEY (EMPID))

한 가지 재미있는 점은, 생성된 테이블을 보면 기본적으로 BIGINT 타입의 "ID"가 있다는 정도!

iris_db_install_3.png




IRIS DB 역시 닷넷으로 프로그래밍하는 방법은, ADO.NET의 사용법을 그대로 따릅니다. DB를 설치한 폴더로부터,

C:\InterSystems\IRIS\dev\dotnet> tree
Folder PATH listing
Volume serial number is D089-B7C9
C:.
└─bin
    ├─Core21
    │  └─Gateway
    │      └─runtimes
    │          └─win
    │              └─lib
    │                  └─netstandard2.0
    ├─CorePackages
    ├─v2.0.50727
    ├─v4.0.30319
    └─v4.5

원하는 패키지를 선택하면 되는데, 이 글에서는 "v4.5"의 것으로 다음과 같이 CRUD 작업을 코딩해 봤습니다.

using InterSystems.Data.IRISClient;
using System;
using System.Data;

namespace ConsoleApp1
{
    class Program
    {
        static void Main(string[] args)
        {
            TestDB();
        }

        static void TestDB()
        {

            // Connecting to the InterSystems Database
            // https://docs.intersystems.com/irislatest/csp/docbook/Doc.View.cls?KEY=BNET_connect#BNET_connect_connect

            using (IRISConnection sqlConnection = new IRISConnection())
            {
                // 기본 포트: 51773
                sqlConnection.ConnectionString = "Server=192.168.0.11; Port=51773; Namespace=USER;Password=SYS;User ID=_SYSTEM;";

                sqlConnection.Open();

                // Create
                IRISCommand insertCommand = new IRISCommand();
                insertCommand.Connection = sqlConnection;
                insertCommand.CommandText = "INSERT INTO Employee(ID, EMPID, NAME, AGE) VALUES (?, ?, ?, ?)";

                insertCommand.Parameters.Add("ID", IRISDbType.BigInt);
                insertCommand.Parameters.Add("EMPID", IRISDbType.Int);
                insertCommand.Parameters.Add("NAME", IRISDbType.NVarChar, 100);
                insertCommand.Parameters.Add("AGE", IRISDbType.Int);

                string nameValue = "Name" + Guid.NewGuid().ToString();
                insertCommand.Parameters[1].Value = (int)DateTime.Now.Ticks;
                insertCommand.Parameters[2].Value = nameValue;
                insertCommand.Parameters[3].Value = 10;

                int affected = insertCommand.ExecuteNonQuery();
                Console.WriteLine("[INSERT] # of affected row: " + affected);

                // Update
                IRISCommand updateCommand = new IRISCommand();
                updateCommand.Connection = sqlConnection;
                updateCommand.CommandText = "UPDATE Employee SET AGE=? WHERE NAME=?";

                updateCommand.Parameters.Add("AGE", IRISDbType.Int);
                updateCommand.Parameters.Add("NAME", IRISDbType.NVarChar, 100);

                updateCommand.Parameters[0].Value = 20;
                updateCommand.Parameters[1].Value = nameValue;

                affected = updateCommand.ExecuteNonQuery();
                Console.WriteLine("[UPDATE] # of affected row: " + affected);

                // Select - ExecuteScalar
                IRISCommand selectCommand = new IRISCommand();
                selectCommand.Connection = sqlConnection;
                selectCommand.CommandText = "SELECT count(*) FROM Employee";

                object result = selectCommand.ExecuteScalar();
                Console.WriteLine("[SELECT] # of records: " + result);

                // Select - DataTable
                DataSet ds = new DataSet();
                IRISDataAdapter da = new IRISDataAdapter("SELECT * FROM Employee", sqlConnection);
                da.Fill(ds, "mytable");

                DataTable dt = ds.Tables["mytable"];
                foreach (DataRow dr in dt.Rows)
                {
                    Console.WriteLine(string.Format("Name = {0}, Age = {1}", dr["NAME"], dr["AGE"]));
                }

                // Delete
                IRISCommand deleteCommand = new IRISCommand();
                deleteCommand.Connection = sqlConnection;
                deleteCommand.CommandText = "DELETE FROM Employee WHERE NAME=?";

                deleteCommand.Parameters.Add("NAME", IRISDbType.NVarChar, 100);
                deleteCommand.Parameters[0].Value = nameValue;

                affected = deleteCommand.ExecuteNonQuery();
                Console.WriteLine("[DELETE] # of affected row: " + affected);
            }
        }

    }
}

(첨부 파일은 이 글의 예제 코드를 포함합니다.)

보는 바와 같이 일반적인 ADO.NET과 사용법은 같지만, SQL Server만 다루던 개발자라면 Parameterized Query에 이름을 지정할 수 없다는 것을 유의해야 합니다. 예를 들어, 본문에서 "INSERT INTO Employee(ID, EMPID, NAME, AGE) VALUES (?, ?, ?, ?)" 쿼리를 보였는데, 따라서 Parameters 컬렉션에 (이름이 없으므로) 들어가는 순서대로 "?"에 매핑이 된다는 것을 신경 써야 합니다.




한 가지 좀 안 좋은 면이 있는데요, IRISConnection 단위로 이전에 수행되지 않은 쿼리인 경우 언제나 "System.ArgumentNullException" 예외가 InterSystems.Data.IRISClient.CachedPrepareCollection.GetStatement 메서드에서 발생합니다. (그러니까, 현실적으로 거의 항상 발생한다고 보면 됩니다.)

internal CachedPrepare GetStatement(string sqlText, IRISCommand cmd)
{
    CachedPrepare prepare2;
    this.rwl.EnterReadLock();
    try
    {
        object obj2 = this.sqlToId[sqlText]; // 수행된 적이 없는 쿼리는 obj2 == null
        CachedPrepare prepare = (CachedPrepare) base[obj2]; // System.ArgumentNullException 예외 발생
        if (prepare != null)
        {
            if (prepare.ownedBy == null)
            {
                prepare.ownedBy = cmd;
                this.unownedCnt--;
                prepare.refCount++;
            }
            else
            {
                return null;
            }
        }
        prepare2 = prepare;
    }
    catch (Exception)
    {
        prepare2 = null;
    }
    finally
    {
        this.rwl.ExitReadLock();
    }
    return prepare2;
}

try/catch가 되어 있기 때문에 실행에 지장은 주지 않지만, Debug 창에 System.ArgumentNullException 오류가 끊임없이 나오는 것은 별로 좋은 사용자 경험은 아닙니다. 게다가 만약 해당 Exception을 "Break When Thrown"으로 지정해 놓았다면 매번 비주얼 스튜디오는 다음의 예외 창을 보여줍니다. (따라서 반드시 해당 옵션을 꺼두어야 합니다.)

System.ArgumentNullException
  HResult=0x80004003
  Message=Key cannot be null.
Parameter name: key
  Source=mscorlib
  StackTrace:
   at System.Collections.Hashtable.get_Item(Object key)

  This exception was originally thrown at this call stack:
    System.Collections.Hashtable.this[object].get(object)




참고로, 아래의 문서를 보면,

Using Cache ADO.NET Managed Provider Classes
; https://cedocs.intersystems.com/latest/csp/docbook/Doc.View.cls?KEY=GBMP_ado#GBMP_ado_intro

Installing Cache
; http://www.hardhats.org/projects/New/InstallCache.html#download-and-install-cach%C3%A9-on-microsoft-windows

"Cache" 접두사로 시작하는 Data Provider도 있었던 것 같습니다.




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 7/28/2020]

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

비밀번호

댓글 작성자
 




... 61  62  63  64  65  66  67  68  69  70  71  72  [73]  74  75  ...
NoWriterDateCnt.TitleFile(s)
11822정성태2/20/201911645오류 유형: 517. docker에 설치한 MongoDB 서버로 연결이 안 되는 경우
11821정성태2/20/201912098오류 유형: 516. Visual Studio 2019 - This extension uses deprecated APIs and is at risk of not functioning in a future VS update. [1]
11820정성태2/20/201915241오류 유형: 515. 윈도우 10 1809 업데이트 후 "User Profiles Service" 1534 경고 발생
11819정성태2/20/201913913Windows: 158. 컴퓨터와 사용자의 SID(security identifier) 확인 방법
11818정성태2/20/201912838VS.NET IDE: 131. Visual Studio 2019 Preview의 닷넷 프로젝트 빌드가 20초 이상 걸리는 경우 [2]
11817정성태2/17/20199991오류 유형: 514. WinDbg Preview 실행 오류 - Error : DbgX.dll : WindowsDebugger.WindowsDebuggerException: Could not load dbgeng.dll
11816정성태2/17/201912494Windows: 157. 윈도우 스토어 앱(Microsoft Store App)을 명령행에서 직접 실행하는 방법
11815정성태2/14/201911386오류 유형: 513. Visual Studio 2019 - VSIX 설치 시 "The extension cannot be installed to this product due to prerequisites that cannot be resolved." 오류 발생
11814정성태2/12/201910205오류 유형: 512. VM(가상 머신)의 NT 서비스들이 자동 시작되지 않는 문제
11813정성태2/12/201911812.NET Framework: 809. C# - ("Save File Dialog" 등의) 대화 창에 확장 속성을 보이는 방법
11812정성태2/11/20199632오류 유형: 511. Windows Server 2003 VM 부팅 후 로그인 시점에 0xC0000005 BSOD 발생
11811정성태2/11/201913423오류 유형: 510. 서버 운영체제에 NVIDIA GeForce Experience 실행 시 wlanapi.dll 누락 문제
11810정성태2/11/201911485.NET Framework: 808. .NET Profiler - GAC 모듈에서 GAC 비-등록 모듈을 참조하는 경우의 문제
11809정성태2/11/201912997.NET Framework: 807. ClrMD를 이용해 메모리 덤프 파일로부터 특정 인스턴스를 참조하고 있는 소유자 확인
11808정성태2/8/201914053디버깅 기술: 123. windbg - 닷넷 응용 프로그램의 메모리 누수 분석
11807정성태1/29/201912360Windows: 156. 가상 디스크의 용량을 복구 파티션으로 인해 늘리지 못하는 경우 [4]
11806정성태1/29/201912114디버깅 기술: 122. windbg - 덤프 파일로부터 PID와 환경 변수 등의 정보를 구하는 방법
11805정성태1/28/201913960.NET Framework: 806. C# - int []와 object []의 차이로 이해하는 제네릭의 필요성 [4]파일 다운로드1
11804정성태1/24/201912013Windows: 155. diskpart - remove letter 이후 재부팅 시 다시 드라이브 문자가 할당되는 경우
11803정성태1/10/201911488디버깅 기술: 121. windbg - 닷넷 Finalizer 스레드가 멈춰있는 현상
11802정성태1/7/201912877.NET Framework: 805. 두 개의 윈도우를 각각 실행하는 방법(Windows Forms, WPF)파일 다운로드1
11801정성태1/1/201913913개발 환경 구성: 427. Netsh의 네트워크 모니터링 기능 [3]
11800정성태12/28/201813196오류 유형: 509. WCF 호출 오류 메시지 - System.ServiceModel.CommunicationException: Internal Server Error
11799정성태12/19/201813995.NET Framework: 804. WPF(또는 WinForm)에서 UWP UI 구성 요소 사용하는 방법 [3]파일 다운로드1
11798정성태12/19/201813233개발 환경 구성: 426. vcpkg - "Building vcpkg.exe failed. Please ensure you have installed Visual Studio with the Desktop C++ workload and the Windows SDK for Desktop C++"
11797정성태12/19/201810582개발 환경 구성: 425. vcpkg - CMake Error: Problem with archive_write_header(): Can't create '' 빌드 오류
... 61  62  63  64  65  66  67  68  69  70  71  72  [73]  74  75  ...