Perl-데이터베이스 액세스

이 장에서는 Perl 스크립트 내에서 데이터베이스에 액세스하는 방법을 설명합니다. Perl 5부터는 다음을 사용하여 데이터베이스 애플리케이션을 작성하기가 매우 쉬워졌습니다.DBI기준 치수. DBI는Database Independent Interface 즉, DBI가 Perl 코드와 기본 데이터베이스 사이에 추상화 계층을 제공하므로 데이터베이스 구현을 정말 쉽게 전환 할 수 있습니다.

DBI는 Perl 프로그래밍 언어를위한 데이터베이스 액세스 모듈입니다. 사용되는 실제 데이터베이스에 관계없이 일관된 데이터베이스 인터페이스를 제공하는 일련의 메소드, 변수 및 규칙을 제공합니다.

DBI 애플리케이션의 아키텍처

DBI는 백엔드에서 사용 가능한 모든 데이터베이스와 독립적입니다. Oracle, MySQL 또는 Informix 등으로 작업하든 DBI를 사용할 수 있습니다. 이것은 다음 아키텍처 다이어그램에서 분명합니다.

여기서 DBI는 API (즉, 응용 프로그래밍 인터페이스)를 통해 모든 SQL 명령을 가져와 실제 실행을 위해 적절한 드라이버로 보내는 일을 담당합니다. 마지막으로 DBI는 드라이버의 결과를 가져 와서 호출하는 스크립트에 돌려주는 책임이 있습니다.

표기법 및 규칙

이 장에서는 다음 표기법이 사용되며 동일한 규칙을 따르는 것이 좋습니다.

$dsn    Database source name
$dbh    Database handle object
$sth    Statement handle object
$h      Any of the handle types above ($dbh, $sth, or $drh)
$rc     General Return Code  (boolean: true=ok, false=error)
$rv     General Return Value (typically an integer)
@ary    List of values returned from the database.
$rows   Number of rows processed (if available, else -1)
$fh     A filehandle
undef   NULL values are represented by undefined values in Perl
\%attr  Reference to a hash of attribute values passed to methods

데이터베이스 연결

MySQL 데이터베이스로 작업한다고 가정합니다. 데이터베이스에 연결하기 전에 다음 사항을 확인하십시오. MySQL 데이터베이스에서 데이터베이스와 테이블을 만드는 방법을 모르는 경우 MySQL 자습서를 참조 할 수 있습니다.

  • 이름이 TESTDB 인 데이터베이스를 만들었습니다.

  • TESTDB에 이름이 TEST_TABLE 인 테이블을 생성했습니다.

  • 이 테이블에는 FIRST_NAME, LAST_NAME, AGE, SEX 및 INCOME 필드가 있습니다.

  • 사용자 ID "testuser"및 비밀번호 "test123"은 TESTDB에 액세스하도록 설정됩니다.

  • Perl 모듈 DBI가 컴퓨터에 올바르게 설치되었습니다.

  • MySQL 기본 사항을 이해하기 위해 MySQL 자습서를 살펴 보았습니다.

다음은 MySQL 데이터베이스 "TESTDB"와 연결하는 예입니다.

#!/usr/bin/perl

use DBI
use strict;

my $driver = "mysql"; 
my $database = "TESTDB";
my $dsn = "DBI:$driver:database=$database";
my $userid = "testuser";
my $password = "test123";

my $dbh = DBI->connect($dsn, $userid, $password ) or die $DBI::errstr;

데이터 소스와 연결이 설정되면 데이터베이스 핸들이 반환되고 추가 사용을 위해 $ dbh에 저장됩니다. 그렇지 않으면 $ dbh가 undef 값으로 설정 되고 $ DBI :: errstr은 오류 문자열을 반환합니다.

INSERT 작업

테이블에 일부 레코드를 만들려면 INSERT 작업이 필요합니다. 여기에서는 TEST_TABLE 테이블을 사용하여 레코드를 만듭니다. 따라서 데이터베이스 연결이 설정되면 TEST_TABLE에 레코드를 만들 준비가 된 것입니다. 다음은 TEST_TABLE에 단일 레코드를 생성하는 절차입니다. 동일한 개념을 사용하여 원하는만큼 레코드를 만들 수 있습니다.

레코드 생성은 다음 단계를 따릅니다.

  • INSERT 문으로 SQL 문을 준비합니다. 이것은 사용하여 수행됩니다prepare() API.

  • SQL 쿼리를 실행하여 데이터베이스에서 모든 결과를 선택합니다. 이것은 사용하여 수행됩니다execute() API.

  • Stattement 핸들을 해제합니다. 이것은 사용하여 수행됩니다finish() API.

  • 모든 것이 잘되면 commit 그렇지 않으면이 작업을 수행 할 수 있습니다. rollback완전한 거래. 커밋 및 롤백은 다음 섹션에서 설명합니다.

my $sth = $dbh->prepare("INSERT INTO TEST_TABLE
                       (FIRST_NAME, LAST_NAME, SEX, AGE, INCOME )
                         values
                       ('john', 'poul', 'M', 30, 13000)");
$sth->execute() or die $DBI::errstr;
$sth->finish();
$dbh->commit or die $DBI::errstr;

바인드 값 사용

입력 할 값을 미리 제공하지 않는 경우가 있습니다. 따라서 런타임에 필요한 값을 취하는 바인드 변수를 사용할 수 있습니다. Perl DBI 모듈은 실제 값 대신 물음표를 사용하고 실제 값은 런타임에 execute () API를 통해 전달됩니다. 다음은 예입니다-

my $first_name = "john";
my $last_name = "poul";
my $sex = "M";
my $income = 13000;
my $age = 30;
my $sth = $dbh->prepare("INSERT INTO TEST_TABLE
                        (FIRST_NAME, LAST_NAME, SEX, AGE, INCOME )
                          values
                        (?,?,?,?)");
$sth->execute($first_name,$last_name,$sex, $age, $income) 
          or die $DBI::errstr;
$sth->finish();
$dbh->commit or die $DBI::errstr;

읽기 작업

모든 데이터베이스에 대한 읽기 작업은 데이터베이스에서 유용한 정보, 즉 하나 이상의 테이블에서 하나 이상의 레코드를 가져 오는 것을 의미합니다. 따라서 데이터베이스 연결이 설정되면이 데이터베이스에 쿼리를 작성할 준비가 된 것입니다. 다음은 AGE가 20보다 큰 모든 레코드를 쿼리하는 절차입니다. 이는 4 단계를 거치게됩니다.

  • 필수 조건에 따라 SQL SELECT 쿼리를 준비합니다. 이것은 사용하여 수행됩니다prepare() API.

  • SQL 쿼리를 실행하여 데이터베이스에서 모든 결과를 선택합니다. 이것은 사용하여 수행됩니다execute() API.

  • 모든 결과를 하나씩 가져와 그 결과를 인쇄합니다. fetchrow_array() API.

  • Stattement 핸들을 해제합니다. 이것은 사용하여 수행됩니다finish() API.

my $sth = $dbh->prepare("SELECT FIRST_NAME, LAST_NAME
                        FROM TEST_TABLE 
                        WHERE AGE > 20");
$sth->execute() or die $DBI::errstr;
print "Number of rows found :" + $sth->rows;
while (my @row = $sth->fetchrow_array()) {
   my ($first_name, $last_name ) = @row;
   print "First Name = $first_name, Last Name = $last_name\n";
}
$sth->finish();

바인드 값 사용

사전에 조건이 주어지지 않은 경우가있을 수 있습니다. 따라서 런타임에 필요한 값을 사용하는 바인드 변수를 사용할 수 있습니다. Perl DBI 모듈은 실제 값 대신 물음표를 사용하고 실제 값은 런타임에 execute () API를 통해 전달됩니다. 다음은 예입니다-

$age = 20;
my $sth = $dbh->prepare("SELECT FIRST_NAME, LAST_NAME
                        FROM TEST_TABLE
                        WHERE AGE > ?");
$sth->execute( $age ) or die $DBI::errstr;
print "Number of rows found :" + $sth->rows;
while (my @row = $sth->fetchrow_array()) {
   my ($first_name, $last_name ) = @row;
   print "First Name = $first_name, Last Name = $last_name\n";
}
$sth->finish();

업데이트 작업

모든 데이터베이스에 대한 UPDATE 작업은 데이터베이스 테이블에서 이미 사용 가능한 하나 이상의 레코드를 업데이트하는 것을 의미합니다. 다음은 SEX가 'M'인 모든 레코드를 업데이트하는 절차입니다. 여기서 우리는 모든 남성의 나이를 1 년 늘릴 것입니다. 세 단계가 필요합니다.

  • 필수 조건에 따라 SQL 쿼리를 준비합니다. 이것은 사용하여 수행됩니다prepare() API.

  • SQL 쿼리를 실행하여 데이터베이스에서 모든 결과를 선택합니다. 이것은 사용하여 수행됩니다execute() API.

  • Stattement 핸들을 해제합니다. 이것은 사용하여 수행됩니다finish() API.

  • 모든 것이 잘되면 commit 그렇지 않으면이 작업을 수행 할 수 있습니다. rollback완전한 거래. 커밋 및 롤백 API는 다음 섹션을 참조하세요.

my $sth = $dbh->prepare("UPDATE TEST_TABLE
                        SET   AGE = AGE + 1 
                        WHERE SEX = 'M'");
$sth->execute() or die $DBI::errstr;
print "Number of rows updated :" + $sth->rows;
$sth->finish();
$dbh->commit or die $DBI::errstr;

바인드 값 사용

사전에 조건이 주어지지 않은 경우가있을 수 있습니다. 따라서 런타임에 필요한 값을 사용하는 바인드 변수를 사용할 수 있습니다. Perl DBI 모듈은 실제 값 대신 물음표를 사용하고 실제 값은 런타임에 execute () API를 통해 전달됩니다. 다음은 예입니다-

$sex = 'M';
my $sth = $dbh->prepare("UPDATE TEST_TABLE
                        SET   AGE = AGE + 1
                        WHERE SEX = ?");
$sth->execute('$sex') or die $DBI::errstr;
print "Number of rows updated :" + $sth->rows;
$sth->finish();
$dbh->commit or die $DBI::errstr;

어떤 경우에는 미리 주어지지 않은 값을 설정하고 싶을 때 다음과 같이 바인딩 값을 사용할 수 있습니다. 이 예에서 모든 남성의 소득은 10000으로 설정됩니다.

$sex = 'M';
$income = 10000;
my $sth = $dbh->prepare("UPDATE TEST_TABLE
                        SET   INCOME = ?
                        WHERE SEX = ?");
$sth->execute( $income, '$sex') or die $DBI::errstr;
print "Number of rows updated :" + $sth->rows;
$sth->finish();

작업 삭제

데이터베이스에서 일부 레코드를 삭제하려면 DELETE 작업이 필요합니다. 다음은 AGE가 30 인 TEST_TABLE에서 모든 레코드를 삭제하는 절차입니다.이 작업은 다음 단계를 수행합니다.

  • 필수 조건에 따라 SQL 쿼리를 준비합니다. 이것은 사용하여 수행됩니다prepare() API.

  • 데이터베이스에서 필요한 레코드를 삭제하기 위해 SQL 쿼리를 실행합니다. 이것은 사용하여 수행됩니다execute() API.

  • Stattement 핸들을 해제합니다. 이것은 사용하여 수행됩니다finish() API.

  • 모든 것이 잘되면 commit 그렇지 않으면이 작업을 수행 할 수 있습니다. rollback 완전한 거래.

$age = 30;
my $sth = $dbh->prepare("DELETE FROM TEST_TABLE
                         WHERE AGE = ?");
$sth->execute( $age ) or die $DBI::errstr;
print "Number of rows deleted :" + $sth->rows;
$sth->finish();
$dbh->commit or die $DBI::errstr;

do 문 사용

UPDATE, INSERT 또는 DELETE를 수행하는 경우 데이터베이스에서 돌아 오는 데이터가 없으므로이 작업을 수행하는 지름길이 있습니다. 당신이 사용할 수있는do 다음과 같이 명령을 실행합니다.

$dbh->do('DELETE FROM TEST_TABLE WHERE age =30');

do성공하면 참 값을 반환하고 실패하면 거짓 값을 반환합니다. 실제로 성공하면 영향을받는 행의 수를 반환합니다. 이 예에서는 실제로 삭제 된 행 수를 반환합니다.

COMMIT 작업

커밋은 변경 사항을 완료하기 위해 데이터베이스에 녹색 신호를 제공하는 작업이며이 작업 후에는 변경 사항을 원래 위치로 되돌릴 수 없습니다.

다음은 호출하는 간단한 예입니다. commit API.

$dbh->commit or die $dbh->errstr;

ROLLBACK 작업

모든 변경 사항이 만족스럽지 않거나 작업 사이에 오류가 발생하면 해당 변경 사항을 되돌려 사용할 수 있습니다. rollback API.

다음은 호출하는 간단한 예입니다. rollback API.

$dbh->rollback or die $dbh->errstr;

거래 시작

많은 데이터베이스가 트랜잭션을 지원합니다. 즉, 데이터베이스를 수정하는 전체 쿼리를 만들 수 있지만 실제로 변경된 사항은 없습니다. 그런 다음 마지막에 특수 SQL 쿼리를 실행합니다.COMMIT, 모든 변경이 동시에 이루어집니다. 또는 쿼리 ROLLBACK을 실행할 수 있습니다.이 경우 모든 변경 사항이 삭제되고 데이터베이스는 변경되지 않습니다.

Perl DBI 모듈 제공 begin_work다음 호출까지 커밋 또는 롤백 할 때까지 트랜잭션을 활성화하는 API (AutoCommit 해제) 다음 커밋 또는 롤백 후에 자동 커밋이 자동으로 다시 켜집니다.

$rc  = $dbh->begin_work  or die $dbh->errstr;

자동 커밋 옵션

트랜잭션이 단순하다면 많은 커밋을 실행해야하는 수고를 덜 수 있습니다. 연결 호출을 할 때 다음을 지정할 수 있습니다.AutoCommit쿼리가 성공할 때마다 자동 커밋 작업을 수행하는 옵션입니다. 다음과 같이 생겼습니다.

my $dbh = DBI->connect($dsn, $userid, $password,
              {AutoCommit => 1}) 
              or die $DBI::errstr;

여기서 AutoCommit은 값 1 또는 0을 취할 수 있습니다. 여기서 1은 AutoCommit이 켜져 있음을 의미하고 0은 AutoCommit이 꺼져 있음을 의미합니다.

자동 오류 처리

연결 호출을 할 때 자동으로 오류를 처리하는 RaiseErrors 옵션을 지정할 수 있습니다. 오류가 발생하면 DBI는 오류 코드를 반환하는 대신 프로그램을 중단합니다. 오류 발생시 프로그램을 중단하는 것만으로도 편리 할 수 ​​있습니다. 다음과 같이 생겼습니다.

my $dbh = DBI->connect($dsn, $userid, $password,
              {RaiseError => 1})
              or die $DBI::errstr;

여기서 RaiseError는 값 1 또는 0을 취할 수 있습니다.

데이터베이스 연결 해제

데이터베이스 연결을 끊으려면 다음을 사용하십시오. disconnect 다음과 같이 API-

$rc = $dbh->disconnect  or warn $dbh->errstr;

disconnect 메소드의 트랜잭션 동작은 슬프게도 정의되지 않았습니다. 일부 데이터베이스 시스템 (예 : Oracle 및 Ingres)은 미해결 변경 사항을 자동으로 커밋하지만 다른 데이터베이스 시스템 (예 : Informix)은 미해결 변경 사항을 롤백합니다. 자동 커밋을 사용하지 않는 응용 프로그램은 연결 해제를 호출하기 전에 명시 적으로 커밋 또는 롤백을 호출해야합니다.

NULL 값 사용

정의되지 않은 값 또는 undef는 NULL 값을 나타내는 데 사용됩니다. NULL이 아닌 값과 마찬가지로 NULL 값으로 열을 삽입하고 업데이트 할 수 있습니다. 이 예제는 NULL 값으로 열 나이를 삽입하고 업데이트합니다.

$sth = $dbh->prepare(qq {
         INSERT INTO TEST_TABLE (FIRST_NAME, AGE) VALUES (?, ?)
       });
$sth->execute("Joe", undef);

여기 qq{} 인용 된 문자열을 반환하는 데 사용됩니다. prepareAPI. 그러나 WHERE 절에서 NULL 값을 사용하려고 할 때는주의해야합니다. 고려-

SELECT FIRST_NAME FROM TEST_TABLE WHERE age = ?

undef (NULL)를 자리 표시 자에 바인딩하면 NULL 연령이있는 행이 선택되지 않습니다! 최소한 SQL 표준을 준수하는 데이터베이스 엔진의 경우. 이에 대한 이유는 데이터베이스 엔진의 SQL 매뉴얼이나 SQL 책을 참조하십시오. NULL을 명시 적으로 선택하려면 "WHERE age IS NULL"이라고 말해야합니다.

일반적인 문제는 런타임에 정의되거나 undef (NULL 또는 NULL이 아님) 될 수있는 값을 코드 조각에서 처리하도록하는 것입니다. 간단한 기술은 필요에 따라 적절한 문을 준비하고 NULL이 아닌 경우 자리 표시자를 대체하는 것입니다.

$sql_clause = defined $age? "age = ?" : "age IS NULL";
$sth = $dbh->prepare(qq {
         SELECT FIRST_NAME FROM TEST_TABLE WHERE $sql_clause
       });
$sth->execute(defined $age ? $age : ());

기타 DBI 기능

available_drivers

@ary = DBI->available_drivers;
@ary = DBI->available_drivers($quiet);

@INC의 디렉터리를 통해 DBD :: * 모듈을 검색하여 사용 가능한 모든 드라이버 목록을 반환합니다. 기본적으로 일부 드라이버가 이전 디렉토리에서 동일한 이름의 다른 드라이버에 의해 숨겨지면 경고가 표시됩니다. $ quiet에 대해 참 값을 전달하면 경고가 표시되지 않습니다.

설치된 _ 드라이버

%drivers = DBI->installed_drivers();

현재 프로세스에 '설치된'(로드 된) 모든 드라이버에 대한 드라이버 이름 및 드라이버 핸들 쌍 목록을 반환합니다. 드라이버 이름에는 'DBD ::'접두사가 포함되지 않습니다.

데이터 소스

@ary = DBI->data_sources($driver);

명명 된 드라이버를 통해 사용할 수있는 데이터 소스 (데이터베이스) 목록을 반환합니다. $ driver가 비어 있거나 undef이면 DBI_DRIVER 환경 변수의 값이 사용됩니다.

인용문

$sql = $dbh->quote($value);
$sql = $dbh->quote($value, $data_type);

문자열 내에 포함 된 특수 문자 (예 : 인용 부호)를 이스케이프하고 필요한 유형의 외부 인용 부호를 추가하여 SQL 문에서 리터럴 값으로 사용할 문자열 리터럴을 인용하십시오.

$sql = sprintf "SELECT foo FROM bar WHERE baz = %s",
                $dbh->quote("Don't");

대부분의 데이터베이스 유형에서 quote는 'Don''t'(외부 인용 부호 포함)를 반환합니다. quote () 메소드가 원하는 문자열로 평가되는 SQL 표현식을 리턴하는 것은 유효합니다. 예를 들면-

$quoted = $dbh->quote("one\ntwo\0three")

may produce results which will be equivalent to

CONCAT('one', CHAR(12), 'two', CHAR(0), 'three')

모든 핸들에 공통적 인 방법

오류

$rv = $h->err;
or
$rv = $DBI::err
or
$rv = $h->err

마지막으로 호출 된 드라이버 메서드에서 네이티브 데이터베이스 엔진 오류 코드를 반환합니다. 코드는 일반적으로 정수이지만이를 가정해서는 안됩니다. 이것은 $ DBI :: err 또는 $ h-> err와 동일합니다.

errstr

$str = $h->errstr;
or
$str = $DBI::errstr
or
$str = $h->errstr

마지막으로 호출 된 DBI 메서드에서 네이티브 데이터베이스 엔진 오류 메시지를 반환합니다. 이것은 위에서 설명한 "err"메소드와 동일한 수명 문제를 가지고 있습니다. 이는 $ DBI :: errstr 또는 $ h-> errstr과 동일합니다.

$rv = $h->rows;
or
$rv = $DBI::rows

이는 이전 SQL 문에 의해 영향을 받고 $ DBI :: rows에 해당하는 행 수를 반환합니다.

자취

$h->trace($trace_settings);

DBI는 수행중인 작업에 대한 런타임 추적 정보를 생성 할 수있는 매우 유용한 기능을 갖추고 있으므로 DBI 프로그램에서 이상한 문제를 추적하려고 할 때 시간을 크게 절약 할 수 있습니다. 다른 값을 사용하여 추적 레벨을 설정할 수 있습니다. 이 값은 0에서 4까지 다양합니다. 값 0은 추적 사용 안함을 의미하고 4는 완전한 추적 생성을 의미합니다.

보간 된 진술은 금지됩니다.

다음과 같이 보간 된 문을 사용하지 않는 것이 좋습니다.

while ($first_name = <>) {
   my $sth = $dbh->prepare("SELECT * 
                          FROM TEST_TABLE 
                          WHERE FIRST_NAME = '$first_name'");
   $sth->execute();
   # and so on ...
}

따라서 보간 된 문을 대신 사용하지 마십시오. bind value 동적 SQL 문을 준비합니다.