理解 DataSet(數(shù)據(jù)集)和 DataTable(數(shù)據(jù)表)

2018-02-24 15:41 更新

理解 DataSet(數(shù)據(jù)集)和 DataTable(數(shù)據(jù)表)

PHPUnit 的數(shù)據(jù)庫擴(kuò)展模塊的核心概念是 DataSet(數(shù)據(jù)集)和 DataTable(數(shù)據(jù)表)。為了掌握如何使用 PHPUnit 進(jìn)行測試,需要試著去了解這些簡單的概念。DataSet(數(shù)據(jù)集)和 DataTable(數(shù)據(jù)表)是圍繞著數(shù)據(jù)庫表、行、列的抽象層。通過一套簡單的API,底層數(shù)據(jù)庫內(nèi)容被隱藏在對(duì)象結(jié)構(gòu)之下,同時(shí),這個(gè)對(duì)象結(jié)構(gòu)也可以用其他非數(shù)據(jù)庫數(shù)據(jù)源來實(shí)現(xiàn)。

為了能比較實(shí)際內(nèi)容和預(yù)期內(nèi)容,這種抽象是必須的。預(yù)期內(nèi)容可以用諸如 XML、 YAML、 CSV 文件或者 PHP 數(shù)組等方式來表達(dá)。DataSet 和 DataTable 接口以語義相似的方式模擬關(guān)系數(shù)據(jù)庫存儲(chǔ),從而能夠?qū)@些概念上完全不同的數(shù)據(jù)源進(jìn)行比較。

在測試中,數(shù)據(jù)庫斷言的工作流由以下三個(gè)簡單的步驟組成:

  • 用表名稱來指定數(shù)據(jù)庫中的一個(gè)或多個(gè)表(實(shí)際上是指定了一個(gè)數(shù)據(jù)集)

  • 用你喜歡的格式(YAML、XML等等)來指定預(yù)期數(shù)據(jù)集

  • 斷言這兩個(gè)數(shù)據(jù)集陳述是彼此相等的。

在 PHPUnit 的數(shù)據(jù)庫擴(kuò)展中,斷言并非唯一使用 DataSet 和 DataTable 的情形。就像上一節(jié)中所展示的那樣,它們也用于描述數(shù)據(jù)庫的初始內(nèi)容。數(shù)據(jù)庫 TestCase 類強(qiáng)制要求定義一個(gè)基境數(shù)據(jù)集,隨后用它來:

  • 根據(jù)此數(shù)據(jù)集所指定的所有表名,將數(shù)據(jù)庫中對(duì)應(yīng)表內(nèi)的行全部刪除。

  • 將數(shù)據(jù)集內(nèi)數(shù)據(jù)表中的所有行寫入數(shù)據(jù)庫。

可用的各種實(shí)現(xiàn)

有三種不同類型的 DataSet/DataTable:

  • 基于文件的 DataSet 和 DataTable

  • 基于查詢的 DataSet 和 DataTable

  • 篩選與組合 DataSet 和 DataTable

基于文件的數(shù)據(jù)集和表一般用于初始化基境或描述數(shù)據(jù)庫的預(yù)期狀態(tài)。

Flat XML DataSet (平直 XML 數(shù)據(jù)集)

最常見的一種數(shù)據(jù)集名叫 Flat XML。這是一種非常簡單的 XML 格式,根節(jié)點(diǎn)為 <dataset>,根節(jié)點(diǎn)下的每個(gè)標(biāo)簽就代表數(shù)據(jù)庫中的一行數(shù)據(jù)。標(biāo)簽的名稱就等于表名,而每個(gè)屬性代表一個(gè)列。一個(gè)簡單的留言本應(yīng)用程序的例子大致上可能是這樣:


<?xml version="1.0" ?>
<dataset>
    <guestbook id="1" content="Hello buddy!" user="joe" created="2010-04-24 17:15:23" />
    <guestbook id="2" content="I like it!" user="nancy" created="2010-04-26 12:14:20" />
</dataset>

顯然,這非常易于編寫。在這里,<guestbook> 是表名,這個(gè)表內(nèi)有兩行記錄,每行有四個(gè)列:“id”、“content”、“user” 和 “created”,以及各自的值。

不過,這種簡單性是有代價(jià)的。

從上面這個(gè)例子里不太容易看出該如何指定一個(gè)空表。其實(shí)可以插入一個(gè)沒有屬性值的標(biāo)簽,以空表的名字作為標(biāo)簽名??盏?guestbook 表所對(duì)應(yīng)的 Flat XML 文件大致上可能是這樣:


<?xml version="1.0" ?>
<dataset>
    <guestbook />
</dataset>

在 Flat XML DataSet 中,要處理 NULL 值會(huì)非常煩。在幾乎所有數(shù)據(jù)庫中(Oracle 是個(gè)例外),NULL 值和空字符串值是有區(qū)別的,這一點(diǎn)在 Flat XML 格式中很難表述。可以在數(shù)據(jù)行的表述中省略掉對(duì)應(yīng)的屬性來表示NULL值。假定上面這個(gè)留言本通過在 user 列使用 NULL 值的方式來允許匿名留言,那么 guestbook 表的內(nèi)容可能是這樣:


<?xml version="1.0" ?>
<dataset>
    <guestbook id="1" content="Hello buddy!" user="joe" created="2010-04-24 17:15:23" />
    <guestbook id="2" content="I like it!" created="2010-04-26 12:14:20" />
</dataset>

在這個(gè)例子里第二個(gè)條目是匿名發(fā)表的。但是這為列的識(shí)別帶來了一個(gè)非常嚴(yán)重的問題。在數(shù)據(jù)集相等斷言的判定過程中,每個(gè)數(shù)據(jù)集都需要指明每個(gè)表擁有哪些列。如果有一個(gè)列在數(shù)據(jù)表的所有行里其值都是 NULL,那么數(shù)據(jù)庫擴(kuò)展模塊又該從何得知表中包含這個(gè)列呢?

在這里,F(xiàn)lat XML DataSet 做了一個(gè)關(guān)鍵假設(shè):一個(gè)表的列信息由此表第一行的屬性定義決定。在上面這個(gè)例子里,這意味著 guestbook 有 “id”、“content”、“user” 和 “created” 這幾個(gè)列。第二行中 “user” 列沒有定義,因此將向數(shù)據(jù)庫中插入 NULL 值。

如果從數(shù)據(jù)集中刪掉第一行,因?yàn)闆]有指定 “user”,guestbook 表擁有的列就只剩下 “id”、“content” 和 “created”。

要在有 NULL 值的情況下有效地使用 Flat XML Dataset,就必須保證每個(gè)表的第一行不包含 NULL 值,只有后繼的那些行才能省略屬性。這就有點(diǎn)棘手,因?yàn)閿?shù)據(jù)行的排列順序也是數(shù)據(jù)斷言的一個(gè)相關(guān)因素。

反過來,如果在 Flat XML Dataset 中只指明了實(shí)際表中所有列的某個(gè)子集,那么所有省略掉的列都會(huì)設(shè)為它們的的默認(rèn)值。如果某個(gè)省略掉的列的定義是 “NOT NULL DEFAULT NULL”,就會(huì)出現(xiàn)錯(cuò)誤。

總的來說,建議只在不需要 NULL 值的情況下使用 Flat XML Dataset。

可以在數(shù)據(jù)庫 TestCase 中調(diào)用 createFlatXmlDataSet($filename) 方法來創(chuàng)建 Flat XML Dataset 實(shí)例:

<?php
class MyTestCase extends PHPUnit_Extensions_Database_TestCase
{
    public function getDataSet()
    {
        return $this->createFlatXmlDataSet('myFlatXmlFixture.xml');
    }
}
?>

XML DataSet (XML 數(shù)據(jù)集)

有另外一種更加結(jié)構(gòu)化的 XML DataSet,它寫起來有點(diǎn)冗長,但是規(guī)避了 Flat XML DataSet 所存在的 NULL 問題。在根節(jié)點(diǎn) <dataset> 內(nèi),可以指定 <table>、<column><row>、<value><null /> 標(biāo)簽。和上面用 Flat XML 所定義的留言本數(shù)據(jù)集等價(jià)的 XML DataSet 如下:


<?xml version="1.0" ?>
<dataset>
    <table name="guestbook">
        <column>id</column>
        <column>content</column>
        <column>user</column>
        <column>created</column>
        <row>
            <value>1</value>
            <value>Hello buddy!</value>
            <value>joe</value>
            <value>2010-04-24 17:15:23</value>
        </row>
        <row>
            <value>2</value>
            <value>I like it!</value>
            <null />
            <value>2010-04-26 12:14:20</value>
        </row>
    </table>
</dataset>

所定義的每個(gè) <table> 都有一個(gè)名稱,并且必須有對(duì)所有列及其名稱的定義。其下可以包含零個(gè)或任意正整數(shù)個(gè) <row> 元素。沒有定義 <row> 意味著這是個(gè)空表。<value><null /> 標(biāo)簽必須按照之前給定 <column> 元素的順序來指定。<null /> 標(biāo)簽顯然意味著這個(gè)值為 NULL。

可以在數(shù)據(jù)庫 TestCase 中調(diào)用 createXmlDataSet($filename) 方法來創(chuàng)建 XML DataSet 實(shí)例:

<?php
class MyTestCase extends PHPUnit_Extensions_Database_TestCase
{
    public function getDataSet()
    {
        return $this->createXMLDataSet('myXmlFixture.xml');
    }
}
?>

MySQL XML DataSet (MySQL XML 數(shù)據(jù)集)

這種新的 XML 格式是 MySQL 數(shù)據(jù)庫服務(wù)器專用的。PHPUnit 3.5 加入了對(duì)這種格式的支持??梢杂?mysqldump 工具來生成這種格式的文件。與同樣為 mysqldump 所支持的 CSV 數(shù)據(jù)集不同,這種 XML 格式可以在單個(gè)文件中包含多個(gè)表的數(shù)據(jù)。要生成這種格式的文件,可以這樣調(diào)用 mysqldump

mysqldump --xml -t -u [username] --password=[password] [database] > /path/to/file.xml

可以在數(shù)據(jù)庫 TestCase 中調(diào)用 createMySQLXMLDataSet($filename) 方法來使用這個(gè)文件:

<?php
class MyTestCase extends PHPUnit_Extensions_Database_TestCase
{
    public function getDataSet()
    {
        return $this->createMySQLXMLDataSet('/path/to/file.xml');
    }
}
?>

YAML DataSet (YAML 數(shù)據(jù)集)

也可以用 YAML DataSet 來寫這個(gè)留言本的例子:


guestbook:
  -
    id: 1
    content: "Hello buddy!"
    user: "joe"
    created: 2010-04-24 17:15:23
  -
    id: 2
    content: "I like it!"
    user:
    created: 2010-04-26 12:14:20

簡單方便,同時(shí)還解決了和它類似的 FLat XML DataSet 所具有的 NULL 問題。在 YAML 中,只有列名而沒有指定值就表示 NULL??瞻鬃址畡t這樣指定:column1: ""。

目前,數(shù)據(jù)庫 TestCase 中沒有 YAML DataSet 的工廠方法,因此需要手工進(jìn)行實(shí)例化:

<?php
class YamlGuestbookTest extends PHPUnit_Extensions_Database_TestCase
{
    protected function getDataSet()
    {
        return new PHPUnit_Extensions_Database_DataSet_YamlDataSet(
            dirname(__FILE__)."/_files/guestbook.yml"
        );
    }
}
?>

CSV DataSet (CSV 數(shù)據(jù)集)

另外一種基于文件的 DataSet 是基于 CSV 文件的。數(shù)據(jù)集中的每個(gè)表用一個(gè)單獨(dú)的 CSV 文件表示。對(duì)于留言本的例子,可以這樣定義 guestbook-table.csv 文件:


id,content,user,created
1,"Hello buddy!","joe","2010-04-24 17:15:23"
2,"I like it!","nancy","2010-04-26 12:14:20"

用 Excel 或者 OpenOffice 來對(duì)這種格式進(jìn)行編輯是非常方便的,但是在 CSV DataSet 中無法指定 NULL 值。給出一個(gè)空白列的結(jié)果是往這個(gè)列中插入數(shù)據(jù)庫的默認(rèn)空值。

可以這樣創(chuàng)建 CSV DataSet:

<?php
class CsvGuestbookTest extends PHPUnit_Extensions_Database_TestCase
{
    protected function getDataSet()
    {
        $dataSet = new PHPUnit_Extensions_Database_DataSet_CsvDataSet();
        $dataSet->addTable('guestbook', dirname(__FILE__)."/_files/guestbook.csv");
        return $dataSet;
    }
}
?>

Array DataSe (數(shù)組數(shù)據(jù)集)

在 PHPUnit 的數(shù)據(jù)庫擴(kuò)展中,(尚)沒有基于數(shù)組的 DataSet,不過很容易自行實(shí)現(xiàn)之。留言本的例子大致是這樣:

<?php
class ArrayGuestbookTest extends PHPUnit_Extensions_Database_TestCase
{
    protected function getDataSet()
    {
        return new MyApp_DbUnit_ArrayDataSet(array(
            'guestbook' => array(
                array('id' => 1, 'content' => 'Hello buddy!', 'user' => 'joe', 'created' => '2010-04-24 17:15:23'),
                array('id' => 2, 'content' => 'I like it!',   'user' => null,  'created' => '2010-04-26 12:14:20'),
            ),
        ));
    }
}
?>

PHP 版本的 DataSet 相比于所有其他基于文件的 DataSet 相比有很明顯的優(yōu)點(diǎn):

  • PHP 數(shù)組顯然可以處理 NULL 值。

  • 不需要為斷言提供任何額外文件,可以直接在 TestCase 中指定。

對(duì)于這種 DataSet 而言,和平直 XML、CSV、YAML DataSet 一樣,表的列名信息由第一個(gè)指定的行的鍵名定義。在上面這個(gè)例子里,就是 “id”、“content”、“user” 和 “created”。

這個(gè)數(shù)組 DataSet 類的實(shí)現(xiàn)是非常簡單直接的:

<?php
class MyApp_DbUnit_ArrayDataSet extends PHPUnit_Extensions_Database_DataSet_AbstractDataSet
{
    /**
     * @var array
     */
    protected $tables = array();

    /**
     * @param array $data
     */
    public function __construct(array $data)
    {
        foreach ($data AS $tableName => $rows) {
            $columns = array();
            if (isset($rows[0])) {
                $columns = array_keys($rows[0]);
            }

            $metaData = new PHPUnit_Extensions_Database_DataSet_DefaultTableMetaData($tableName, $columns);
            $table = new PHPUnit_Extensions_Database_DataSet_DefaultTable($metaData);

            foreach ($rows AS $row) {
                $table->addRow($row);
            }
            $this->tables[$tableName] = $table;
        }
    }

    protected function createIterator($reverse = FALSE)
    {
        return new PHPUnit_Extensions_Database_DataSet_DefaultTableIterator($this->tables, $reverse);
    }

    public function getTable($tableName)
    {
        if (!isset($this->tables[$tableName])) {
            throw new InvalidArgumentException("$tableName is not a table in the current database.");
        }

        return $this->tables[$tableName];
    }
}
?>

Query (SQL) DataSet (查詢(SQL)數(shù)據(jù)集)

對(duì)于數(shù)據(jù)庫斷言,不僅需要有基于文件的 DataSet,同時(shí)也需要有一種內(nèi)含數(shù)據(jù)庫實(shí)際內(nèi)容的基于查詢/SQL 的 DataSet。Query DataSet 在此閃亮登場:

<?php
$ds = new PHPUnit_Extensions_Database_DataSet_QueryDataSet($this->getConnection());
$ds->addTable('guestbook');
?>

單純以名稱來添加表是一種隱式地用以下查詢來定義 DataTable 的方法:

<?php
$ds = new PHPUnit_Extensions_Database_DataSet_QueryDataSet($this->getConnection());
$ds->addTable('guestbook', 'SELECT * FROM guestbook');
?>

可以在這種用法中為你的表任意指定查詢,例如限定行、列,或者加上 ORDER BY 子句:

<?php
$ds = new PHPUnit_Extensions_Database_DataSet_QueryDataSet($this->getConnection());
$ds->addTable('guestbook', 'SELECT id, content FROM guestbook ORDER BY created DESC');
?>

在關(guān)于數(shù)據(jù)庫斷言的那一節(jié)中有更多關(guān)于如何使用 Query DataSet 的細(xì)節(jié)。

Database (DB) Dataset (數(shù)據(jù)庫數(shù)據(jù)集)

通過訪問測試所使用的數(shù)據(jù)庫連接,可以自動(dòng)創(chuàng)建包含數(shù)據(jù)庫所有表以及其內(nèi)容的 DataSet。所使用的數(shù)據(jù)庫由數(shù)據(jù)庫連接工廠方法的第二個(gè)參數(shù)指定。

可以像 testGuestbook() 中那樣創(chuàng)建整個(gè)數(shù)據(jù)庫所對(duì)應(yīng)的 DataSet,或者像 testFilteredGuestbook() 方法中那樣用一個(gè)白名單來將 DataSet 限制在若干表名的集合上。

<?php
class MySqlGuestbookTest extends PHPUnit_Extensions_Database_TestCase
{
    /**
     * @return PHPUnit_Extensions_Database_DB_IDatabaseConnection
     */
    public function getConnection()
    {
        $database = 'my_database';
        $user = 'my_user';
        $password = 'my_password';
        $pdo = new PDO('mysql:...', $user, $password);
        return $this->createDefaultDBConnection($pdo, $database);
    }

    public function testGuestbook()
    {
        $dataSet = $this->getConnection()->createDataSet();
        // ...
    }

    public function testFilteredGuestbook()
    {
        $tableNames = array('guestbook');
        $dataSet = $this->getConnection()->createDataSet($tableNames);
        // ...
    }
}
?>

Replacement DataSet (替換數(shù)據(jù)集)

前面談到了 Flat XML 和 CSV DataSet 所存在的 NULL 問題,不過有一種稍微有點(diǎn)復(fù)雜的解決方法可以讓這兩種數(shù)據(jù)集都能正常處理 NULL。

Replacement DataSet 是已有數(shù)據(jù)集的修飾器(decorator),能夠?qū)?shù)據(jù)集中任意列的值替換為其他替代值。為了讓留言本的例子能夠處理 NULL 值,首先指定類似這樣的文件:


<?xml version="1.0" ?>
<dataset>
    <guestbook id="1" content="Hello buddy!" user="joe" created="2010-04-24 17:15:23" />
    <guestbook id="2" content="I like it!" user="##NULL##" created="2010-04-26 12:14:20" />
</dataset>

然后將 Flat XML DataSet 包裝在 Replacement DataSet 中:

<?php
class ReplacementTest extends PHPUnit_Extensions_Database_TestCase
{
    public function getDataSet()
    {
        $ds = $this->createFlatXmlDataSet('myFlatXmlFixture.xml');
        $rds = new PHPUnit_Extensions_Database_DataSet_ReplacementDataSet($ds);
        $rds->addFullReplacement('##NULL##', null);
        return $rds;
    }
}
?>

DataSet Filter (數(shù)據(jù)集篩選器)

如果有一個(gè)非常大的基境文件,可以用數(shù)據(jù)集篩選器來為需要包含在子數(shù)據(jù)集中的表和列指定白/黑名單。與 DB DataSet 聯(lián)用來對(duì)數(shù)據(jù)集中的列進(jìn)行篩選尤其方便。

<?php
class DataSetFilterTest extends PHPUnit_Extensions_Database_TestCase
{
    public function testIncludeFilteredGuestbook()
    {
        $tableNames = array('guestbook');
        $dataSet = $this->getConnection()->createDataSet();

        $filterDataSet = new PHPUnit_Extensions_Database_DataSet_DataSetFilter($dataSet);
        $filterDataSet->addIncludeTables(array('guestbook'));
        $filterDataSet->setIncludeColumnsForTable('guestbook', array('id', 'content'));
        // ..
    }

    public function testExcludeFilteredGuestbook()
    {
        $tableNames = array('guestbook');
        $dataSet = $this->getConnection()->createDataSet();

        $filterDataSet = new PHPUnit_Extensions_Database_DataSet_DataSetFilter($dataSet);
        $filterDataSet->addExcludeTables(array('foo', 'bar', 'baz')); // only keep the guestbook table!
        $filterDataSet->setExcludeColumnsForTable('guestbook', array('user', 'created'));
        // ..
    }
}
?>

注意:不能對(duì)同一個(gè)表同時(shí)應(yīng)用排除與包含兩種列篩選器,只能分別應(yīng)用于不同的表。另外,表的白名單和黑名單也只能選擇其一,不能二者同時(shí)使用。

Composite DataSet (組合數(shù)據(jù)集)

Composite DataSet 能將多個(gè)已存在的數(shù)據(jù)集聚合成單個(gè)數(shù)據(jù)集,因此非常有用。如果多個(gè)數(shù)據(jù)集中存在同樣的表,其中的數(shù)據(jù)行將按照指定的順序進(jìn)行追加。例如,假設(shè)有兩個(gè)數(shù)據(jù)集, fixture1.xml


<?xml version="1.0" ?>
<dataset>
    <guestbook id="1" content="Hello buddy!" user="joe" created="2010-04-24 17:15:23" />
</dataset>

fixture2.xml


<?xml version="1.0" ?>
<dataset>
    <guestbook id="2" content="I like it!" user="##NULL##" created="2010-04-26 12:14:20" />
</dataset>

通過 Composite DataSet 可以把這兩個(gè)基境文件聚合在一起:

<?php
class CompositeTest extends PHPUnit_Extensions_Database_TestCase
{
    public function getDataSet()
    {
        $ds1 = $this->createFlatXmlDataSet('fixture1.xml');
        $ds2 = $this->createFlatXmlDataSet('fixture2.xml');

        $compositeDs = new PHPUnit_Extensions_Database_DataSet_CompositeDataSet();
        $compositeDs->addDataSet($ds1);
        $compositeDs->addDataSet($ds2);

        return $compositeDs;
    }
}
?>

當(dāng)心外鍵

在建立基境的過程中, PHPUnit 的數(shù)據(jù)庫擴(kuò)展模塊按照基境中所指定的順序?qū)?shù)據(jù)行插入到數(shù)據(jù)庫內(nèi)。假如數(shù)據(jù)庫中使用了外鍵,這就意味著必須指定好表的順序,以避免外鍵約束失敗。

實(shí)現(xiàn)自有的 DataSet/DataTable

為了理解 DataSet 和 DataTable 的內(nèi)部實(shí)現(xiàn),讓我們來看看 DataSet 的接口。如果沒打算自行實(shí)現(xiàn) DataSet 或者 DataTable,可以直接跳過這一部分。

<?php
interface PHPUnit_Extensions_Database_DataSet_IDataSet extends IteratorAggregate
{
    public function getTableNames();
    public function getTableMetaData($tableName);
    public function getTable($tableName);
    public function assertEquals(PHPUnit_Extensions_Database_DataSet_IDataSet $other);

    public function getReverseIterator();
}
?>

這些 public 接口在數(shù)據(jù)庫 TestCase 中 assertDataSetsEqual() 斷言內(nèi)使用,用以檢測數(shù)據(jù)集是否相等。IDataSet 中繼承自 IteratorAggregate 接口的 getIterator() 方法用于對(duì)數(shù)據(jù)集中的所有表進(jìn)行迭代。逆序迭代器讓 PHPUnit 能夠按照與創(chuàng)建時(shí)相反的順序?qū)λ斜韴?zhí)行 TRUNCATE 操作,以此來保證滿足外鍵約束。

根據(jù)具體實(shí)現(xiàn)的不同,要采取不同的方法來將表實(shí)例添加到數(shù)據(jù)集中。例如,在所有基于文件的數(shù)據(jù)集中,表都是在構(gòu)造過程中直接從源文件生成并加入數(shù)據(jù)集中,比如 YamlDataSetXmlDataSetFlatXmlDataSet均是如此。

數(shù)據(jù)表則由以下接口表示:

<?php
interface PHPUnit_Extensions_Database_DataSet_ITable
{
    public function getTableMetaData();
    public function getRowCount();
    public function getValue($row, $column);
    public function getRow($row);
    public function assertEquals(PHPUnit_Extensions_Database_DataSet_ITable $other);
}
?>

除了 getTableMetaData() 方法之外,這個(gè)接口是一目了然的。數(shù)據(jù)庫擴(kuò)展模塊中的各種斷言(將于下一章中介紹)用到了所有這些方法,因此它們?nèi)慷际潜匦璧摹?code>getTableMetaData() 方法需要返回一個(gè)實(shí)現(xiàn)了 PHPUnit_Extensions_Database_DataSet_ITableMetaData 接口的描述表結(jié)構(gòu)的對(duì)象。這個(gè)對(duì)象包含如下信息:

  • 表的名稱

  • 表的列名數(shù)組,按照列在結(jié)果集中出現(xiàn)的順序排列。

  • 構(gòu)成主鍵的列的數(shù)組。

這個(gè)接口還包含有檢驗(yàn)兩個(gè)表的元數(shù)據(jù)實(shí)例是否彼此相等的斷言,供數(shù)據(jù)集相等斷言使用。

以上內(nèi)容是否對(duì)您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號(hào)
微信公眾號(hào)

編程獅公眾號(hào)