成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

寫一個(gè)“特殊”的查詢構(gòu)造器 - (六、關(guān)聯(lián))

rainyang / 1090人閱讀

摘要:雖然現(xiàn)在這樣的情況已經(jīng)很少,但是對(duì)于查詢構(gòu)造器而言,還是要提供一個(gè)方便的方法來(lái)對(duì)表前綴進(jìn)行設(shè)置,特別是當(dāng)你沒有權(quán)限修改表名的時(shí)候。所以我們將表前綴作為一個(gè)配置參數(shù)傳入查詢構(gòu)造器,在查詢構(gòu)造器的底層進(jìn)行自動(dòng)前綴添加。

關(guān)聯(lián)查詢是關(guān)系型數(shù)據(jù)庫(kù)典型的查詢語(yǔ)句,根據(jù)兩個(gè)或多個(gè)表中的列之間的關(guān)系,從這些表中查詢數(shù)據(jù)。在 SQL 標(biāo)準(zhǔn)中使用 JOIN 和 ON 關(guān)鍵字來(lái)實(shí)現(xiàn)關(guān)聯(lián)查詢。

Join 子句

join 子句的構(gòu)造并不難,注意事項(xiàng)就是關(guān)聯(lián)查詢的注意事項(xiàng):

寫對(duì)語(yǔ)法和關(guān)聯(lián)的條件

使用 table.field 模式防止字段重名

基類中新建 join() 方法:

// $table 要關(guān)聯(lián)的表
// $one   作為關(guān)聯(lián)條件的一個(gè)表的字段
// $two   作為關(guān)聯(lián)條件的另一個(gè)表的字段
// $type  關(guān)聯(lián)模式 inner、left、right
public function join($table, $one, $two, $type = "INNER")
{
    // 判斷模式是否合法
    if( ! in_array($type, ["INNER", "LEFT", "RIGHT"])) {
        throw new InvalidArgumentException("Error join mode");
    }
    // 構(gòu)建 join 子句字符串
    $this->_join_str .= " ".$type." JOIN ".self::_wrapRow($table).
        " ON ".self::_wrapRow($one)." = ".self::_wrapRow($two);
    return $this;
}

leftJoin() 和 rightJoin() 方法:

public function leftJoin($table, $one, $two)
{
    return $this->join($table, $one, $two, "LEFT");
}

public function rightJoin($table, $one, $two)
{
    return $this->join($table, $one, $two, "RIGHT");
}
注:Sqlite 是不支持 right join 的,所以 rightJoin() 方法在 Sqlite 驅(qū)動(dòng)類中無(wú)效。

構(gòu)建 SELECT student.name, class.name FROM student INNER JOIN class ON student.class_id = class.id;

$results = $driver->table("student")
            ->select("student.name", "class.name")
            ->join("class", "student.class_id", "class.id")
            ->get();
表前綴 為什么要有表前綴

以前很多數(shù)據(jù)表放在一個(gè)數(shù)據(jù)庫(kù)中的時(shí)候,需要表前綴來(lái)區(qū)分功能。雖然現(xiàn)在這樣的情況已經(jīng)很少,但是對(duì)于查詢構(gòu)造器而言,還是要提供一個(gè)方便的方法來(lái)對(duì)表前綴進(jìn)行設(shè)置,特別是當(dāng)你沒有權(quán)限修改表名的時(shí)候。

自動(dòng)添加表前綴的方法

對(duì)于有表前綴的表,我們并不想每次都寫一個(gè)前綴,這樣會(huì)導(dǎo)致前綴更改后,應(yīng)用層要跟著修改。所以我們將表前綴作為一個(gè)配置參數(shù)傳入查詢構(gòu)造器,在查詢構(gòu)造器的底層進(jìn)行自動(dòng)前綴添加。

表前綴的配置,假設(shè)表前綴為 "test_" :

// 以 mysql 為例
$config = [
    "host"        => "localhost",
    "port"        => "3306",
    "user"        => "username",
    "password"    => "password",
    "dbname"      => "dbname",
    "charset"     => "utf8",
    "prefix"      => "test_",
    "timezone"    => "+8:00",
    "collection"  => "utf8_general_ci",
    "strict"      => false,
    // "unix_socket" => "/var/run/mysqld/mysqld.sock",
];
$db = new Mysql($config);

進(jìn)行自動(dòng)添加前綴的方法:

protected function _wrapTable($table)
{
    // 構(gòu)造函數(shù)傳入的配置中有前綴參數(shù)嗎?
    $prefix = array_key_exists("prefix", $this->_config) ?
            $this->_config["prefix"] : "";
    // 拼接前綴
    return $prefix.$table;
}

修改 table() 方法:

public function table($table)
{
    // 自動(dòng)添加前綴
    $this->_table = self::_wrapRow($this->_wrapTable($table));

    return $this;
}

join 子句中也涉及到表,所以修改 join() 方法:

public function join($table, $one, $two, $type = "INNER")
{
    if( ! in_array($type, ["INNER", "LEFT", "RIGHT"])) {
        throw new InvalidArgumentException("Error join mode");
    }
    // 添加表前綴
    $table = $this->_wrapTable($table);
    
    $this->_join_str .= " ".$type." JOIN ".self::_wrapRow($table).
        " ON ".self::_wrapRow($one)." = ".self::_wrapRow($two);
    return $this;
}
table.field 模式的表前綴添加

增加了表前綴后,我們會(huì)發(fā)現(xiàn)一個(gè)問(wèn)題:

使用 table()、join() 方法傳入的表可以自動(dòng)的添加前綴,但是 table.field 格式中的表沒法自動(dòng)添加前綴,如上面的 join("class", "student.class_id", "class.id"),我們總不能每次都寫成 join("class", "test_student.class_id", "test_class.id") 這種 (這樣的話和全部手工添加前綴沒什么兩樣),必須找到一個(gè)自動(dòng)添加前綴的辦法。

觀察 table.field 模式,它出現(xiàn)的位置不定,可能在列、任何一個(gè)子句中出現(xiàn),所以在固定的位置去添加前綴是不大可能的。那么我們反過(guò)來(lái)想一下,如果在 SQL 已經(jīng)構(gòu)造完成但還未執(zhí)行時(shí),這時(shí)已經(jīng)知道有哪些地方使用了這種格式,去一一替換即可。那么如何知道有哪些地方使用了這種格式?

使用正則

我們用正則表達(dá)式找到 table.field 的 table 部分,給 table 加上表前綴即可 (這里不考慮跨庫(kù)查詢時(shí)三個(gè)點(diǎn)的情況)。

基類新增 _wrapPrepareSql() 方法:

// 替換 table.field 為 prefixtable.field
protected function _wrapPrepareSql()
{
    $quote = static::$_quote_symbol;
    $prefix_pattern = "/".$quote."([a-zA-Z0-9_]+)".$quote."(.)".$quote."([a-zA-Z0-9_]+)".$quote."/";
    $prefix_replace = self::_quote($this->_wrapTable("$1"))."$2".self::_quote("$3");

    $this->_prepare_sql = preg_replace($prefix_pattern, $prefix_replace, $this->_prepare_sql);
}

修改 _execute() 方法:

protected function _execute()
{
    try {
        // table.field 模式添加表前綴
        $this->_wrapPrepareSql();
        $this->_pdoSt = $this->_pdo->prepare($this->_prepare_sql);
        $this->_bindParams();
        $this->_pdoSt->execute();
        $this->_reset();
    } catch (PDOException $e) {
        if($this->_isTimeout($e)) { 

            $this->_closeConnection();
            $this->_connect();
            
            try {
                // table.field 模式添加表前綴
                $this->_wrapPrepareSql();
                $this->_pdoSt = $this->_pdo->prepare($this->_prepare_sql);
                $this->_bindParams();
                $this->_pdoSt->execute();
                $this->_reset();
            } catch (PDOException $e) {
                throw $e;
            }
        } else {
            throw $e;
        }
    }

}

最后我們進(jìn)行一個(gè)完整的測(cè)試:

require_once dirname(dirname(__FILE__)) . "/vendor/autoload.php";

use DriversMysql;

$config = [
    "host"        => "localhost",
    "port"        => "3306",
    "user"        => "username",
    "password"    => "password",
    "dbname"      => "database",
    "prefix"      => "test_",
    "charset"     => "utf8",
    "timezone"    => "+8:00",
    "collection"  => "utf8_general_ci",
    "strict"      => false,
];

$driver = new Mysql($config);

$results = $driver->table("student")
    ->select("student.name", "class.name")
    ->join("class", "student.class_id", "class.id")
    ->get();

var_dump($results);

試試看吧!

復(fù)雜語(yǔ)句的構(gòu)造

到目前位置,查詢相關(guān)的 SQL 構(gòu)造方法基本開發(fā)完畢,我們進(jìn)行一些復(fù)雜的 SQL 構(gòu)造吧。

注:這里只是以我的測(cè)試環(huán)境舉例,大家可以按照自己的思路去建表

構(gòu)造語(yǔ)句 SELECT * FROM t_user WHERE username = "Jackie aa" OR ( NOT EXISTS ( SELECT * FROM t_user WHERE username = "Jackie aa" ) AND (username = "Jackie Conroy" OR username = "Jammie Haag")) AND g_id IN ( SELECT id FROM t_user_group) ORDER BY id DESC LIMIT 1 OFFSET 0

$results = $driver->table("user")
            ->where("username", "Jackie aa")
            ->orWhereBrackets(function($query) {
                $query->whereNotExists(function($query) {
                    $query->table("user")->where("username", "Jackie aa");
                })->WhereBrackets(function($query) {
                    $query->where("username", "Jackie Conroy")
                            ->orWhere("username", "Jammie Haag");
                });
            })
            ->whereInSub("g_id", function($query) {
                $query->table("user_group")->select("id");
            })
            ->orderBy("id", "DESC")
            ->limit(0, 1)
            ->get();

構(gòu)造語(yǔ)句 SELECT t_user.username, t_user_group.groupname FROM t_user LEFT JOIN t_user_group ON t_user.g_id = t_user_group.id WHERE username = "Jackie aa" OR ( NOT EXISTS ( SELECT * FROM t_user WHERE username = "Jackie aa" ) AND username = "Jackie Conroy" )

$results = $driver->table("user")
            ->select("user.username", "user_group.groupname")
            ->leftJoin("user_group", "user.g_id", "user_group.id")
            ->where("user.username", "Jackie aa")
            ->orWhereBrackets(function($query) {
                $query->whereNotExists(function($query) {
                    $query->table("user")->where("username", "Jackie aa");
                })->where("user.username", "Jackie Conroy");
            })
            ->get();

更多例子參考 WorkerF 單元測(cè)試 - PDODQL

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/30825.html

相關(guān)文章

  • 一個(gè)特殊查詢構(gòu)造器 - (前言)

    摘要:而在項(xiàng)目開發(fā)中,我們想要的是一個(gè)更好用的可維護(hù)的工具,此時(shí),對(duì)代碼的封裝模塊化就顯得尤為重要,于是出現(xiàn)了兩種方案查詢構(gòu)造器,對(duì)象關(guān)系映射。典型環(huán)境下按照一般的查詢構(gòu)造器處理就行。 文章目錄 寫一個(gè)特殊的查詢構(gòu)造器 - (前言) 寫一個(gè)特殊的查詢構(gòu)造器 - (一、程序結(jié)構(gòu),基礎(chǔ)封裝) 寫一個(gè)特殊的查詢構(gòu)造器 - (二、第一條語(yǔ)句) 寫一個(gè)特殊的查詢構(gòu)造器 - (三、條件查詢) 寫一個(gè)特殊...

    GitChat 評(píng)論0 收藏0
  • 一個(gè)特殊查詢構(gòu)造器 - (二、第一條語(yǔ)句)

    摘要:注在常駐內(nèi)存單例模式下,這種多次用一個(gè)類進(jìn)行查詢的情形很常見。斷線重連對(duì)于典型環(huán)境而言,一次的查詢已經(jīng)隨著的請(qǐng)求而結(jié)束,的垃圾回收功能會(huì)回收一次請(qǐng)求周期內(nèi)的數(shù)據(jù)。但在常駐內(nèi)存的環(huán)境下,尤其是單例模式下,數(shù)據(jù)庫(kù)驅(qū)動(dòng)類可能一直在內(nèi)存中不被銷毀。 構(gòu)造、執(zhí)行第一條語(yǔ)句 上一篇完成了代碼結(jié)構(gòu)的搭建和 PDO 的基礎(chǔ)封裝,這一篇我們來(lái)講如何構(gòu)造一個(gè)最基本的 SQL 語(yǔ)句,并執(zhí)行得到結(jié)果。 que...

    dadong 評(píng)論0 收藏0
  • Java多線程學(xué)習(xí)()Lock鎖使用

    摘要:返回與此鎖相關(guān)聯(lián)的給定條件等待的線程數(shù)的估計(jì)。查詢是否有線程正在等待獲取此鎖。為公平鎖,為非公平鎖線程運(yùn)行了獲得鎖定運(yùn)行結(jié)果公平鎖的運(yùn)行結(jié)果是有序的。 系列文章傳送門: Java多線程學(xué)習(xí)(一)Java多線程入門 Java多線程學(xué)習(xí)(二)synchronized關(guān)鍵字(1) java多線程學(xué)習(xí)(二)synchronized關(guān)鍵字(2) Java多線程學(xué)習(xí)(三)volatile關(guān)鍵字 ...

    Caicloud 評(píng)論0 收藏0
  • Spring Boot 2.x():優(yōu)雅統(tǒng)一返回值

    摘要:下面我們來(lái)測(cè)試一下,訪問(wèn)我們經(jīng)過(guò)修改后的編寫的接口這里我將返回值統(tǒng)一為,以便數(shù)據(jù)存入,實(shí)際類型應(yīng)是接口的返回類型。如果沒有返回值的話,那就可以一個(gè)對(duì)象直接通過(guò)構(gòu)造方法賦值即可。 為什么要統(tǒng)一返回值 在我們做后端應(yīng)用的時(shí)候,前后端分離的情況下,我們經(jīng)常會(huì)定義一個(gè)數(shù)據(jù)格式,通常會(huì)包含code,message,data這三個(gè)必不可少的信息來(lái)方便我們的交流,下面我們直接來(lái)看代碼 ReturnV...

    shaonbean 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<