Skip to content

類型

MySQL 型別

SQLDelight 欄位定義與一般的 H2 欄位定義相同,但支援一個 額外的欄位約束,該約束指定了在生成的介面中該欄位的 Kotlin 型別。

sql
CREATE TABLE some_types (
  some_tiny_int TINYINT,                           -- 檢索為 Byte
  some_small_int SMALLINT,                         -- 檢索為 Short
  some_integer INTEGER,                            -- 檢索為 Int
  some_int INT,                                    -- 檢索為 Int
  some_big_int BIGINT,                             -- 檢索為 Long
  some_decimal DECIMAL(6,5),                       -- 檢索為 Int
  some_dec DEC(6,5),                               -- 檢索為 Int
  some_numeric NUMERIC(6,5),                       -- 檢索為 Int
  some_float FLOAT(6),                             -- 檢索為 Double
  some_real REAL,                                  -- 檢索為 Double
  some_double DOUBLE,                              -- 檢索為 Double
  some_double_precision DOUBLE PRECISION,          -- 檢索為 Double
  some_boolean BOOLEAN,                            -- 檢索為 Boolean
  some_date DATE,                                  -- 檢索為 String
  some_time TIME,                                  -- 檢索為 String
  some_timestamp2 TIMESTAMP(6),                    -- 檢索為 String
  some_char CHAR,                                  -- 檢索為 String
  some_character CHARACTER(6),                     -- 檢索為 String
  some_char_varying CHAR VARYING(6),               -- 檢索為 String
  some_longvarchar LONGVARCHAR,                    -- 檢索為 String
  some_character_varying CHARACTER VARYING(6),     -- 檢索為 String
  some_varchar VARCHAR(16),                        -- 檢索為 String
  some_clo CHARACTER LARGE OBJECT(16),             -- 檢索為 String
  some_clob clob(16 M CHARACTERS),                 -- 檢索為 String
  some_binary BINARY,                              -- 檢索為 ByteArray
  some_binary2 BINARY(6),                          -- 檢索為 ByteArray
  some_longvarbinary LONGVARBINARY,                -- 檢索為 ByteArray
  some_longvarbinary2 LONGVARBINARY(6),            -- 檢索為 ByteArray
  some_binary_varying BINARY VARYING(6),           -- 檢索為 ByteArray
  some_varbinary VARBINARY(8),                     -- 檢索為 ByteArray
  some_uuid UUID,                                  -- 檢索為 ByteArray
  some_blob BLOB,                                  -- 檢索為 ByteArray
  some_blo BINARY LARGE OBJECT(6),                 -- 檢索為 ByteArray
  some_bit BIT,                                    -- 檢索為 ByteArray
  some_bit2 BIT(6),                                -- 檢索為 ByteArray
  some_bit_varying BIT VARYING(6),                 -- 檢索為 ByteArray
  some_interval INTERVAL YEAR TO MONTH,            -- 檢索為 ByteArray
  some_interval2 INTERVAL YEAR(3),                 -- 檢索為 ByteArray
  some_interval3 INTERVAL DAY(4) TO HOUR,          -- 檢索為 ByteArray
  some_interval4 INTERVAL MINUTE(4) TO SECOND(6),  -- 檢索為 ByteArray
  some_interval5 INTERVAL SECOND(4,6)              -- 檢索為 ByteArray
);

自訂欄位型別

如果您想將欄位擷取為自訂型別,可以指定一個 Kotlin 型別:

sql
import kotlin.String;
import kotlin.collections.List;

CREATE TABLE hockeyPlayer (
  cup_wins TEXT AS List<String> NOT NULL
);

然而,建立 Database 將需要您提供一個 ColumnAdapter,它知道如何將資料庫型別映射到您的自訂型別:

kotlin
val listOfStringsAdapter = object : ColumnAdapter<List<String>, String> {
  override fun decode(databaseValue: String) =
    if (databaseValue.isEmpty()) {
      listOf()
    } else {
      databaseValue.split(",")
    }
  override fun encode(value: List<String>) = value.joinToString(separator = ",")
}

val queryWrapper: Database = Database(
  driver = driver,
  hockeyPlayerAdapter = hockeyPlayer.Adapter(
    cup_winsAdapter = listOfStringsAdapter
  )
)

列舉 (Enums)

為方便起見,SQLDelight 執行時 (runtime) 包含一個 ColumnAdapter,用於將列舉 (enum) 儲存為字串資料。

sql
import com.example.hockey.HockeyPlayer;

CREATE TABLE hockeyPlayer (
  position TEXT AS HockeyPlayer.Position
)
kotlin
val queryWrapper: Database = Database(
  driver = driver,
  hockeyPlayerAdapter = HockeyPlayer.Adapter(
    positionAdapter = EnumColumnAdapter()
  )
)

值型別

如果需要,SQLDelight 可以為欄位生成一個值型別,該型別會包裝底層 (underlying) 資料庫型別:

sql
CREATE TABLE hockeyPlayer (
  id INT AS VALUE
);

## 樂觀鎖定

如果將欄位指定為 `LOCK`,它會為其產生一個值型別,並要求 `UPDATE` 陳述式正確使用該鎖定來執行更新。

```sql
CREATE TABLE hockeyPlayer(
  id INT AS VALUE,
  version_number INT AS LOCK,
  name VARCHAR(8)
);

-- 這將會失敗(IDE 外掛會建議改寫成下面的形式)
updateName:
UPDATE hockeyPlayer
SET name = ?;

-- 這將會通過編譯
updateNamePassing:
UPDATE hockeyPlayer
SET name = ?
    version_number = :version_number + 1
WHERE version_number = :version_number;

遷移中的自訂型別

如果遷移是綱要 (schema) 的事實來源,您也可以在修改表格時指定公開的 Kotlin 型別:

sql
import kotlin.String;
import kotlin.collection.List;

ALTER TABLE my_table
  ADD COLUMN new_column VARCHAR(8) AS List<String>;