Przykłady transakcji PHP + MySQL

Naprawdę nie znalazłem normalnego przykładu pliku PHP, w którym wykorzystywane są transakcje MySQL. Możesz mi pokazać prosty przykład?

I jeszcze jedno pytanie. Zrobiłem już dużo programowania i nie korzystałem z transakcji. Czy Mogę umieścić funkcję PHP lub coś w header.php, że jeśli jedna mysql_query się nie powiedzie, to pozostałe też się nie powiedzie?
Chyba to rozgryzłem, prawda?:
mysql_query("SET AUTOCOMMIT=0");
mysql_query("START TRANSACTION");

$a1 = mysql_query("INSERT INTO rarara (l_id) VALUES('1')");
$a2 = mysql_query("INSERT INTO rarara (l_id) VALUES('2')");

if ($a1 and $a2) {
    mysql_query("COMMIT");
} else {        
    mysql_query("ROLLBACK");
}
Author: Bill the Lizard, 2010-04-25

9 answers

Pomysł, z którego zazwyczaj korzystam podczas pracy z transakcjami wygląda tak (semi-pseudo-kod):

try {
    // First of all, let's begin a transaction
    $db->beginTransaction();

    // A set of queries; if one fails, an exception should be thrown
    $db->query('first query');
    $db->query('second query');
    $db->query('third query');

    // If we arrive here, it means that no exception was thrown
    // i.e. no query has failed, and we can commit the transaction
    $db->commit();
} catch (Exception $e) {
    // An exception has been thrown
    // We must rollback the transaction
    $db->rollback();
}


Należy zauważyć, że w tym pomyśle, jeśli zapytanie nie powiedzie się, należy wyrzucić wyjątek:

  • PDO może to zrobić, w zależności od tego, jak go skonfigurujesz
  • w przeciwnym razie, z jakimś innym API, może być konieczne przetestowanie wyniku funkcji użytej do wykonania zapytania i rzucenie / align = "left" /


Niestety, nie ma w tym żadnej magii. Nie można po prostu umieścić instrukcji gdzieś i mieć transakcje wykonywane automatycznie: nadal musisz określić, która grupa zapytań ma być wykonana w transakcji.

Na przykład, dość często będziesz miał kilka zapytań przed transakcją (przed begin) i jeszcze kilka zapytań po transakcji (po commit lub rollback) i będziesz chciał te zapytania wykonywane bez względu na to, co się stało (lub nie) w transakcji.

 299
Author: Pascal MARTIN,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2013-03-03 10:25:40

Chyba to rozgryzłem, prawda?:

mysql_query("START TRANSACTION");

$a1 = mysql_query("INSERT INTO rarara (l_id) VALUES('1')");
$a2 = mysql_query("INSERT INTO rarara (l_id) VALUES('2')");

if ($a1 and $a2) {
    mysql_query("COMMIT");
} else {        
    mysql_query("ROLLBACK");
}
 106
Author: good_evening,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2014-03-22 22:32:52
<?php

// trans.php
function begin(){
    mysql_query("BEGIN");
}

function commit(){
    mysql_query("COMMIT");
}

function rollback(){
    mysql_query("ROLLBACK");
}

mysql_connect("localhost","Dude1", "SuperSecret") or die(mysql_error());

mysql_select_db("bedrock") or die(mysql_error());

$query = "INSERT INTO employee (ssn,name,phone) values ('123-45-6789','Matt','1-800-555-1212')";

begin(); // transaction begins

$result = mysql_query($query);

if(!$result){
    rollback(); // transaction rolls back
    echo "transaction rolled back";
    exit;
}else{
    commit(); // transaction is committed
    echo "Database transaction was successful";
}

?>
 37
Author: Gedzberg Alex,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2015-04-08 09:26:08

Ponieważ jest to pierwszy wynik w google dla "transakcji PHP mysql", pomyślałem, że dodam odpowiedź, która wyraźnie pokazuje, jak to zrobić z mysqli (jak oryginalny autor chciał przykładów). Oto uproszczony przykład transakcji z PHP / mysqli:

// let's pretend that a user wants to create a new "group". we will do so
// while at the same time creating a "membership" for the group which
// consists solely of the user themselves (at first). accordingly, the group
// and membership records should be created together, or not at all.
// this sounds like a job for: TRANSACTIONS! (*cue music*)

$group_name = "The Thursday Thumpers";
$member_name = "EleventyOne";
$conn = new mysqli($db_host,$db_user,$db_passwd,$db_name); // error-check this

// note: this is meant for InnoDB tables. won't work with MyISAM tables.

try {

    $conn->autocommit(FALSE); // i.e., start transaction

    // assume that the TABLE groups has an auto_increment id field
    $query = "INSERT INTO groups (name) ";
    $query .= "VALUES ('$group_name')";
    $result = $conn->query($query);
    if ( !$result ) {
        $result->free();
        throw new Exception($conn->error);
    }

    $group_id = $conn->insert_id; // last auto_inc id from *this* connection

    $query = "INSERT INTO group_membership (group_id,name) ";
    $query .= "VALUES ('$group_id','$member_name')";
    $result = $conn->query($query);
    if ( !$result ) {
        $result->free();
        throw new Exception($conn->error);
    }

    // our SQL queries have been successful. commit them
    // and go back to non-transaction mode.

    $conn->commit();
    $conn->autocommit(TRUE); // i.e., end transaction
}
catch ( Exception $e ) {

    // before rolling back the transaction, you'd want
    // to make sure that the exception was db-related
    $conn->rollback(); 
    $conn->autocommit(TRUE); // i.e., end transaction   
}

Należy również pamiętać, że PHP 5.5 ma nową metodę mysqli:: begin_transaction . Jednak nie zostało to jeszcze udokumentowane przez zespół PHP, a ja nadal utknąłem w PHP 5.3, więc nie mogę tego skomentować.

 34
Author: EleventyOne,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2013-07-25 00:34:10

Proszę sprawdzić, którego silnika pamięci używasz. Jeśli jest to MyISAM, to Transaction('COMMIT','ROLLBACK') nie będzie obsługiwane, ponieważ tylko silnik przechowywania InnoDB, a nie MyISAM, obsługuje transakcje.

 7
Author: dinesh,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2016-09-15 09:49:27

Zrobiłem funkcję, aby uzyskać wektor zapytań i zrobić transakcję, może ktoś się dowie, że jest przydatna:

function transaction ($con, $Q){
        mysqli_query($con, "START TRANSACTION");

        for ($i = 0; $i < count ($Q); $i++){
            if (!mysqli_query ($con, $Q[$i])){
                echo 'Error! Info: <' . mysqli_error ($con) . '> Query: <' . $Q[$i] . '>';
                break;
            }   
        }

        if ($i == count ($Q)){
            mysqli_query($con, "COMMIT");
            return 1;
        }
        else {
            mysqli_query($con, "ROLLBACK");
            return 0;
        }
    }
 5
Author: Marco,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2014-05-22 13:03:38

Podczas korzystania z połączenia PDO:

$pdo = new PDO('mysql:host=localhost;dbname=mydb;charset=utf8', $user, $pass, [
    PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION, // this is important
]);

Często używam następującego kodu do zarządzania transakcjami:

function transaction(Closure $callback)
{
    global $pdo; // let's assume our PDO connection is in a global var

    // start the transaction outside of the try block, because
    // you don't want to rollback a transaction that failed to start
    $pdo->beginTransaction(); 
    try
    {
        $callback();
        $pdo->commit(); 
    }
    catch (Exception $e) // it's better to replace this with Throwable on PHP 7+
    {
        $pdo->rollBack();
        throw $e; // we still have to complain about the exception
    }
}

Przykład użycia:

transaction(function()
{
    global $pdo;

    $pdo->query('first query');
    $pdo->query('second query');
    $pdo->query('third query');
});

W ten sposób Kod zarządzania transakcjami nie jest powielany w całym projekcie. Co jest dobre, bo sądząc po innych odpowiedziach w tym wątku łatwo popełnić w nim błędy. Najczęstsze z nich to zapominanie o rethrow wyjątku i rozpoczynanie transakcji wewnątrz bloku try.

 4
Author: Danila Piatov,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2017-09-02 00:34:21

Miałem to, ale nie jestem pewien, czy to prawda. Może wypróbować również to.

mysql_query("START TRANSACTION");
$flag = true;
$query = "INSERT INTO testing (myid) VALUES ('test')";

$query2 = "INSERT INTO testing2 (myid2) VALUES ('test2')";

$result = mysql_query($query) or trigger_error(mysql_error(), E_USER_ERROR);
if (!$result) {
$flag = false;
}

$result = mysql_query($query2) or trigger_error(mysql_error(), E_USER_ERROR);
if (!$result) {
$flag = false;
}

if ($flag) {
mysql_query("COMMIT");
} else {        
mysql_query("ROLLBACK");
}

Pomysł stąd: http://www.phpknowhow.com/mysql/transactions/

 3
Author: nodeffect,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2014-01-30 01:44:08

Jeszcze jeden przykład stylu proceduralnego z mysqli_multi_query, zakłada, że $query jest wypełniona instrukcjami oddzielonymi średnikami.

mysqli_begin_transaction ($link);

for (mysqli_multi_query ($link, $query);
    mysqli_more_results ($link);
    mysqli_next_result ($link) );

! mysqli_errno ($link) ?
    mysqli_commit ($link) : mysqli_rollback ($link);
 1
Author: guest,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2016-02-11 21:22:34