>  기사  >  백엔드 개발  >  PHP에서 준비된 명령문과 트랜잭션을 이해하는 데 1분 정도 시간을 투자해 보세요.

PHP에서 준비된 명령문과 트랜잭션을 이해하는 데 1분 정도 시간을 투자해 보세요.

醉折花枝作酒筹
醉折花枝作酒筹앞으로
2021-06-16 14:58:192295검색

이 글에서는 PHP에서 준비된 명령문과 트랜잭션을 소개합니다. 도움이 필요한 친구들이 모두 참고할 수 있기를 바랍니다.

PHP에서 준비된 명령문과 트랜잭션을 이해하는 데 1분 정도 시간을 투자해 보세요.

PHP에서 PDO 연산 배우기 (2) 준비된 명령문과 트랜잭션

오늘 글에서는 PDO에서 준비된 명령문과 트랜잭션의 사용법을 간략하게 알아보겠습니다. 모두 PDO에 있는 객체 아래의 연산입니다. 복잡하지 않으며, 간단한 애플리케이션도 쉽게 구현할 수 있습니다. 하지만 대부분의 경우 모두가 액자를 사용하고 있고, 손글씨를 쓸 기회가 거의 없습니다.

Prepared 문장 기능

Prepared 문장은 실행할 문장을 준비한 후 PDOStatement 객체를 반환하는 것입니다. 일반적으로 우리는 이 명령문을 실행하기 위해 PDOStatement 객체의 excute() 메소드를 사용할 것입니다. 전처리라고 불리는 이유는 무엇입니까? 왜냐하면 이 명령문을 여러 번 호출하고 명령문의 필드 조건을 자리 표시자로 바꿀 수 있기 때문입니다. PDO 개체를 직접 사용하는 query() 또는 exec()에 비해 전처리가 더 효율적이며 클라이언트/서버가 쿼리 및 메타 정보를 캐시할 수 있습니다. 물론 더 중요한 점은 자리 표시자를 적용하면 기본적인 SQL 주입 공격을 효과적으로 방지할 수 있다는 점입니다. SQL 문에 수동으로 따옴표를 추가할 필요가 없으며 전처리를 통해 이 문제를 직접 해결할 수 있다는 사실은 누구나 알 수 있습니다. 면접 시 가장 많이 보는 질문 중 하나이기도 합니다.

// 使用 :name 形式创建一个只进游标的 PDOStatement 对象
$stmt = $pdo->prepare("select * from zyblog_test_user where username = :username", [PDO::ATTR_CURSOR => PDO::CURSOR_FWDONLY]);

var_dump($stmt);
// object(PDOStatement)#2 (1) {
//     ["queryString"]=>
//     string(57) "select * from zyblog_test_user where username = :username"
//   }

$stmt->execute([':username' => 'aaa']);
$aUser = $stmt->fetchAll();

$stmt->execute([':username' => 'bbb']);
$bUser = $stmt->fetchAll();

var_dump($aUser);
// array(1) {
//     [0]=>
//     array(8) {
//       ["id"]=>
//       string(1) "1"
//       [0]=>
//       string(1) "1"
//       ["username"]=>
//       string(3) "aaa"
//       ……

var_dump($bUser);
// array(1) {
//     [0]=>
//     array(8) {
//       ["id"]=>
//       string(1) "2"
//       [0]=>
//       string(1) "2"
//       ["username"]=>
//       string(3) "bbb"
//       ……

prepare() 메소드의 첫 번째 매개변수는 실행해야 하는 SQL 문입니다. 이 코드에서는 :xxx 형식의 자리 표시자를 사용하므로 prepare() 메소드를 호출하여 반환된 PDOStatement 객체에서 Execute() 메서드를 실행하려면 자리 표시자의 값을 지정해야 합니다. 코드에서는 이 SQL 문을 사용하여 서로 다른 자리 표시자 콘텐츠를 대체하여 두 개의 쿼리를 구현합니다.

prepare() 메소드의 두 번째 매개변수는 반환된 PDOStatement 객체에 대해 설정된 속성입니다. 일반적인 사용법은 다음과 같습니다. PDO::ATTR_CURSOR를 PDO::CURSOR_SCROLL로 설정하면 스크롤 가능한 커서가 생성됩니다.

일부 드라이버에는 준비 중에 설정되는 드라이버 수준 옵션이 있습니다. PDO::ATTR_CURSOR는 데이터베이스 커서를 설정하는 타입이고, PDO::CURSOR_FWDONLY는 커서에만 들어가는 PDOStatement 객체를 생성한다는 뜻이다. 이 커서는 PHP에서 가장 빠르고 가장 일반적인 데이터 액세스 모드이기 때문에 이것이 기본 커서 옵션입니다. 데이터베이스 커서에 대한 지식은 관련 내용을 직접 확인하실 수 있습니다.

또한 PDOStatement는 binParam() 메서드를 통해 자리 표시자 데이터를 바인딩할 수도 있습니다. PDOStatement 개체와 관련된 다음 문서에서 계속 학습하겠습니다.

다음으로 ? 자리 표시자를 사용하여 쿼리를 구현하는 방법을 살펴보겠습니다. ? 자리 표시자는 바인딩할 때 아래 첨자 형태로 바인딩됩니다.

// 使用 ? 形式创建一个只进游标的 PDOStatement 对象
$stmt = $pdo->prepare("select * from zyblog_test_user where username = ?", [PDO::ATTR_CURSOR => PDO::CURSOR_FWDONLY]);

$stmt->execute(['aaa']);
$aUser = $stmt->fetchAll();

var_dump($aUser);
// array(1) {
//     [0]=>
//     array(8) {
//       ["id"]=>
//       string(1) "1"
//       [0]=>
//       string(1) "1"
//       ["username"]=>
//       string(3) "aaa"
//       ……

물론, 이런 종류의 미리 컴파일된 명령문은 쿼리 명령문에만 국한되지 않고 추가, 삭제, 수정이 가능하며 자리표시자도 지원합니다. PHP에서 데이터베이스를 조작하기 위해 준비된 명령문 이 문서에는 자세한 예제가 있습니다.

트랜잭션 기능

모든 사람은 트랜잭션에 대해 어느 정도 이해하고 있어야 하므로 여기서는 구체적인 개념을 소개하지 않고 PDO에서 트랜잭션이 어떻게 구현되는지 살펴보겠습니다. 먼저 거래가 없을 때 어떤 일이 발생하는지 살펴보겠습니다.

$pdo->exec("insert into tran_innodb (name, age) values ('Joe', 12)"); // 成功插入

$pdo->exec("insert into tran_innodb2 (name, age) values ('Joe', 12)"); // 报错停止整个PHP脚本执行
// Fatal error: Uncaught PDOException: SQLSTATE[42S02]: Base table or view not found: 1146 Table 'blog_test.tran_innodb2' doesn't exist

이 두 테이블을 동시에 업데이트해야 하는데 두 번째 문에서 오류를 보고한다고 가정해 보겠습니다. 트랜잭션이 없으면 첫 번째 데이터 조각이 정상적으로 성공적으로 삽입되지만 이는 우리에게 필요한 결과가 아닙니다. 이때 두 테이블을 동시에 성공하거나 실패하게 만들려면 트랜잭션 기능의 도움이 필요합니다.

try {
    // 开始事务
    $pdo->beginTransaction();

    $pdo->exec("insert into tran_innodb (name, age) values ('Joe', 12)");
    $pdo->exec("insert into tran_innodb2 (name, age) values ('Joe', 12)"); // 不存在的表

    // 提交事务
    $pdo->commit();
} catch (Exception $e) {
    // 回滚事务
    $pdo->rollBack();
    // 输出报错信息
    echo "Failed: " . $e->getMessage(), PHP_EOL;
    // Failed: SQLSTATE[42S02]: Base table or view not found: 1146 Table 'blog_test.tran_innodb2' doesn't exist
}

첫 번째는 데이터베이스 자동 제출을 끄고 트랜잭션을 시작하는 데 사용되는 BeginTransaction() 메서드입니다. 이 메서드 이후에는 commit() 또는 RollBack() 메서드를 만난 후에만 트랜잭션이 닫힙니다.

commit() 메소드는 작업 중 사고가 발생하지 않은 경우 BeginTransaction() 이후 모든 데이터 작업을 패키징하여 제출하는 것입니다.

rollBack()은 데이터를 롤백합니다.beginTransaction() 이후의 문이나 코드에 문제가 발생하면 이전 데이터 작업이 롤백되어 startTransaction() 이후의 모든 문이 성공하거나 실패하는지 확인합니다.

이와 같은 세 가지 간단한 기능이 전체 거래 작업을 완료합니다. 트랜잭션에 대한 심층적인 연구는 앞으로 MySQL을 깊이 있게 공부할 때 다루도록 하겠습니다. 여기서 주의할 점은 PDO 객체에서는 예외 발생으로 오류 모드를 지정하는 것이 가장 좋다는 점입니다. 오류 모드를 지정하지 않으면 트랜잭션의 오류가 직접 보고되지 않고 오류 코드가 반환됩니다. 커밋할지 롤백할지 결정하려면 오류 코드를 사용해야 합니다. 이는 예외 메커니즘보다 훨씬 덜 간결하고 직관적입니다.

总结

我们简单的梳理并学习了一下 PDO 中的预处理和事务相关的知识,接下来就要进入 PDOStatement 对象相关内容的学习。PDOStatement 对象就是 PDO 的预处理对象,也就是在日常开发中我们会接触到的最多的数据操作对象。这块可是重点内容,大家可不能松懈了哦!

测试代码:

https://github.com/zhangyue0503/dev-blog/blob/master/php/202008/source/PHP%E4%B8%AD%E7%9A%84PDO%E6%93%8D%E4%BD%9C%E5%AD%A6%E4%B9%A0%EF%BC%88%E4%BA%8C%EF%BC%89%E9%A2%84%E5%A4%84%E7%90%86%E8%AF%AD%E5%8F%A5%E5%8F%8A%E4%BA%8B%E5%8A%A1.php

推荐学习:php视频教程

위 내용은 PHP에서 준비된 명령문과 트랜잭션을 이해하는 데 1분 정도 시간을 투자해 보세요.의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
이 기사는 segmentfault.com에서 복제됩니다. 침해가 있는 경우 admin@php.cn으로 문의하시기 바랍니다. 삭제