首頁  >  文章  >  後端開發  >  php + mysql 分散式事務(xa)

php + mysql 分散式事務(xa)

巴扎黑
巴扎黑原創
2016-11-09 14:07:132768瀏覽

事務(Transaction)是存取並可能更新資料庫中各種資料項目的程式執行單元;

事務應該具有4個屬性:原子性、一致性、隔離性、持續性

原子性(atomicity)。一個事務是一個不可分割的工作單位,事務中包含的諸操作要么都做,要么都不做。

一致性(consistency)。事務必須是使資料庫從一個一致性狀態變成另一個一致性狀態。一致性與原子性是密切相關的。

隔離性(isolation)。一個事務的執行不能被其他事務幹擾。即一個事務內部的操作及使用的資料對並發的其他事務是隔離的,並發執行的各個事務之間不能互相干擾。

持久性(durability)。持續性也稱為永久性(permanence),指一個事務一旦提交,它對資料庫中資料的改變就應該是永久性的。接下來的其他操作或故障不應該對其有任何影響。

分散式事務:分散式事務的參與者、資源管理器、事務管理器等位於不使用的節點上,這些不同的節點相互協作共同完成一個具有邏輯完整性的事務。

修正自己對mysql的一個誤解,mysql從5.0開始支援XA DataSource。 Connector/J 版本要使用5.0版本,5.0以下的不支援。

  XA協定由Tuxedo首先提出的,並交給X/Open組織,作為資源管理器(資料庫)與事務管理器的介面標準。目前,Oracle、Informix、DB2和Sybase等各大資料庫廠商都提供對XA的支援。 XA協定採用兩階段提交方式來管理分散式事務。 XA介面提供資源管理器與事務管理器之間進行通訊的標準介面。 XA協定包含兩套函數,以xa_開頭的及以ax_開頭的。 

  以下的函數使事務管理器可以對資源管理器進行的操作: 

  1)xa_open,xa_close:建立並關閉與資源管理器的連接。 

  2)xa_start,xa_end:開始和結束一個本地事務。 

  3)xa_prepare,xa_commit,xa_rollback:預先提交、提交和回滾一個本地事務。 

  4)xa_recover:回滾一個已進行預先提交的交易。 

  5)ax_開頭的函數可讓資源管理器動態地在事務管理器中進行註冊,並且可以對XID(TRANSACTION IDS)進行操作。 

  6)ax_reg,ax_unreg;允許一個資源管理器在一個TMS(TRANSACTION MANAGER SERVER)中動態註冊或撤銷註冊。

MySQL XA分為兩類,內部XA與外部XA;內部XA用於同一實例下跨多個引擎的事務,由大家熟悉的Binlog作為協調者;外部XA用於跨多MySQL實例的分散式事務,需要應用層介入作為協調者(崩潰時的懸掛事務,全局提交還是回滾,需要由應用層決定,對應用層的實現要求較高);

 Binlog作為內部XA的協調者,在binlog中出現的內部xid,在crash recover時,由binlog負責提交。 (這是因為,binlog不進行prepare, 只進行commit,因此在binlog中出現的內部xid,一定能夠保證其在底層各存儲引擎中已經完成prepare)。

MySQL資料庫外部XA可以用在分散式資料庫代理層,實現對MySQL資料庫的分散式事務支持,例如開源的代理工具:網易的DDB,淘寶的TDDL,B2B的Cobar等等。

範例

public function testAction(){
        $goods_id=1;
        $goods_name = "大西瓜";
        $num = 1;
        $rs_order = $this->test->createorder($goods_id,$goods_name,$num);
        $rs_goods = $this->test->deduction($goods_id,$num);
        if($rs_order['status'] =="success" && $rs_goods['status']=="success"){
            $this->test->commitdb($rs_order['XA']);
            $this->test->commitdb1($rs_goods['XA']);
        }else{
            $this->test->rollbackdb($rs_order['XA']);
            $this->test->rollbackdb1($rs_goods['XA']);
        }
        
        print_r($rs_order);
        echo "<br />";
        print_r($rs_goods);
        die("dddd");
    }
    
    public function createorder($goods_id,$goods_name,$num){
        $XA = uniqid("");
        $this->_db->query("XA START &#39;$XA&#39;");
        $_rs = true;
        try {
            $data = array();
            $data[&#39;order_id&#39;] = "V".date("YmdHis");
            $data[&#39;goods_name&#39;] = $goods_name;
            $data[&#39;goods_num&#39;] = $num;
            $this->_db->insert("temp_orders",$data);
            $rs =  $this->_db->lastInsertId();
            if($rs){
                $_rs = true;
            }else{
                $_rs = false;
            }
        } catch (Exception $e) {
            $_rs = false;
        }
        $this->_db->query("XA END &#39;$XA&#39;");
         if($_rs){
                 $this->_db->query("XA PREPARE &#39;$XA&#39;");
                 return array("status"=>"success","XA"=>$XA);
         }else{
                 return array("status"=>"nosuccess","XA"=>$XA);
         }
    }
    
    public function deduction($id){
        $XA = uniqid("");
        $this->db1->query("XA START &#39;$XA&#39;");
        $last_rs = true;
        try {
                $sql = "select * from temp_goods where id = &#39;$id&#39; and goods_num>0";
                $rs = $this->db1->fetchRow($sql);
                if(!empty($rs)){
                    $sql = "update temp_goods set goods_num = goods_num-1 where id = &#39;$id&#39;";
                    $rd = $this->db1->query($sql);
                    if($rd){
                        $last_rs = true;
                    }else{
                        $last_rs = false;
                    }
                }else{
                        $last_rs = false;;
                }
        } catch (Exception $e) {
             $last_rs = false;;
        }
         $this->db1->query("XA END &#39;$XA&#39;");
         
         if($last_rs){
                 $this->db1->query("XA PREPARE &#39;$XA&#39;");
                 return array("status"=>"success","XA"=>$XA);
         }else{
                 return array("status"=>"nosuccess","XA"=>$XA);
         }
    
    }
    //提交事务!
    public function commitdb($xa){
        return $this->_db->query("XA COMMIT &#39;$xa&#39;");
    }
    
    //回滚事务
    public function rollbackdb($xa){
        return $this->_db->query("XA ROLLBACK &#39;$xa&#39;");
    }
    
    //提交事务!
    public function commitdb1($xa){
        return $this->db1->query("XA COMMIT &#39;$xa&#39;");
    }
    
    //回滚事务
    public function rollbackdb1($xa){
        return $this->db1->query("XA ROLLBACK &#39;$xa&#39;");
    }


陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn