web-dev-qa-db-fra.com

SQLite ajoute la clé primaire

J'ai créé une table dans Sqlite en utilisant la syntaxe CREATE TABLE AS pour créer une table basée sur une instruction SELECT. Maintenant, cette table n'a pas de clé primaire mais j'aimerais en ajouter une.

L'exécution de ALTER TABLE table_name ADD PRIMARY KEY(col1, col2,...) donne une erreur de syntaxe "near PRIMARY"

Existe-t-il un moyen d’ajouter une clé primaire lors de la création de la table ou ultérieurement dans Sqlite?

EDIT: Par "pendant la création", j'entends pendant la création avec CREATE TABLE AS.

91
Jack Edmonds

Vous ne pouvez pas modifier les tables SQLite de manière significative après leur création. La solution suggérée acceptée consiste à créer une nouvelle table avec les exigences correctes et à y copier vos données, puis à supprimer l'ancienne table.

voici la documentation officielle à ce sujet: http://sqlite.org/faq.html#q11

117
Nathan Ridley

Tant que vous utilisez CREATE TABLE, si vous créez la clé primaire sur un single field, vous pouvez utiliser:

CREATE TABLE mytable (
field1 TEXT,
field2 INTEGER PRIMARY KEY,
field3 BLOB,
);

Avec CREATE TABLE, vous pouvez également toujours utiliser l'approche suivante pour créer une clé primaire sur un ou plusieurs champs:

CREATE TABLE mytable (
field1 TEXT,
field2 INTEGER,
field3 BLOB,
PRIMARY KEY (field2, field1)
);

Référence: http://www.sqlite.org/lang_createtable.html

Cette réponse n'aborde pas la modification de table.

30
A-B-B

J'ai essayé d'ajouter la clé primaire par la suite en modifiant directement la table sqlite_master ... Cette astuce semble fonctionner ... .. C'est une solution de hack, bien sûr. 

En bref: créez un index régulier (unique) sur la table, puis insérez le schéma en écriture et remplacez le nom de l'index par le formulaire réservé à sqlite pour identifier un index de clé primaire (par exemple, sqlite_autoindex_XXX_1, où XXX est le nom de la table) et définissez la chaîne SQL sur NULL. Enfin, modifiez la définition de la table elle-même . Cela ressemble à un bogue, mais pas grave (même si vous ne rouvrez pas la base de données, vous pouvez toujours l’utiliser). 

Supposons que la table ressemble à:

CREATE TABLE tab1(i INTEGER, j INTEGER, t TEXT);

Puis j'ai fait ce qui suit:

BEGIN;
CREATE INDEX pk_tab1 ON tab1(i,j);
pragma writable_schema=1;
UPDATE sqlite_master SET name='sqlite_autoindex_tab1_1',sql=null WHERE name='pk_tab1';
UPDATE sqlite_master SET sql='CREATE TABLE tab1(i integer,j integer,t text,primary key(i,j))' WHERE name='tab1';
COMMIT;

Quelques tests (dans sqlite Shell):

sqlite> explain query plan select * from tab1 order by i,j;
0|0|0|SCAN TABLE tab1 USING INDEX sqlite_autoindex_tab1_1
sqlite> drop index sqlite_autoindex_tab1_1;
Error: index associated with UNIQUE or PRIMARY KEY constraint cannot be dropped    
10
user3098421

Selon sqlite docs à propos de la création de table, l 'utilisation de la table create en tant que select produit une nouvelle table sans contrainte ni clé primaire.

Cependant, la documentation indique également que les clés primaires et les index uniques sont logiquement équivalents ( voir la section sur les contraintes ):

Dans la plupart des cas, les contraintes UNIQUE et PRIMARY KEY sont implémentées en créant un index unique dans la base de données. (Les exceptions sont INTEGER PRIMARY KEY et PRIMARY KEY sur les tables WITHOUT ROWID.) Par conséquent, les schémas suivants sont logiquement équivalents:

CREATE TABLE t1(a, b UNIQUE);

CREATE TABLE t1(a, b PRIMARY KEY);

CREATE TABLE t1(a, b);
CREATE UNIQUE INDEX t1b ON t1(b); 

Ainsi, même si vous ne pouvez pas modifier la définition de votre table via la syntaxe de modification SQL, vous pouvez obtenir le même effet de clé primaire grâce à l'utilisation d'un index unique.

De plus, toutes les tables (à l'exception de celles créées sans la syntaxe rowid) ont une colonne entière interne appelée "rowid". Selon la documentation, vous pouvez utiliser cette colonne interne pour extraire/modifier les tables d’enregistrement.

6
Carlitos Way

Vous pouvez le faire comme ça:

CREATE TABLE mytable (
field1 text,
field2 text,
field3 integer,
PRIMARY KEY (field1, field2)
);
5
Nick Dandoulakis

Introduction

Ceci est basé sur le Java d'Android et c'est un bon exemple pour changer la base de données sans gêner vos fans/clients d'applications. Ceci est basé sur l'idée de la page SQLite FAQ http://sqlite.org/faq.html#q11

Le problème 

Je n'ai pas remarqué que je devais définir un numéro_ligne ou un_nom_enregistrement pour supprimer un seul article acheté dans un reçu. En même temps, le numéro de code à barres de l'article m'a incité à penser qu'il serait la clé pour supprimer cet article. J'enregistre les détails d'un reçu dans la table Ticket_Barcode. Le laisser sans record_id peut signifier la suppression de tous les enregistrements du même article d'un reçu si j'ai utilisé le code à barres de l'article comme clé.

Remarquer

S'il vous plaît, comprenez qu'il s'agit d'un copier-coller de mon code sur lequel je travaille au moment de l'écriture. Utilisez-le uniquement à titre d'exemple, le copier-coller de manière aléatoire ne vous aidera pas. Modifiez ceci d'abord selon vos besoins

Merci également de ne pas oublier de lire les commentaires dans le code.

Le code

Utilisez cette méthode dans votre classe pour vérifier d’abord si la colonne que vous voulez ajouter est manquante. Nous faisons cela simplement pour ne pas répéter le processus de modification de la table Ticket_barcode . Il suffit de le mentionner dans le cadre de votre classe. Dans l'étape suivante, vous verrez comment nous allons l'utiliser.

public boolean is_column_exists(SQLiteDatabase mDatabase , String table_name,
String     column_name) {
    //checks if table_name has column_name
    Cursor cursor = mDatabase.rawQuery("pragma table_info("+table_name+")",null);
    while (cursor.moveToNext()){
    if (cursor.getString(cursor.getColumnIndex("name")).equalsIgnoreCase(column_name)) return true;
    }
    return false;
}

Ensuite, le code suivant est utilisé pour créer la table receive_barcode si elle est déjà NOT exit pour la première fois par les utilisateurs de votre application. Et s'il vous plaît noter le "SI N'EXISTE" dans le code. Ça a de l'importance.

//mDatabase should be defined as a Class member (global variable) 
//for ease of access : 
//SQLiteDatabse mDatabase=SQLiteDatabase.openOrCreateDatabase(dbfile_path, null);
creation_query = " CREATE TABLE if not exists receipt_barcode ( ";
creation_query += "\n record_id        INTEGER PRIMARY KEY AUTOINCREMENT,";
creation_query += "\n rcpt_id INT( 11 )       NOT NULL,";
creation_query += "\n barcode VARCHAR( 255 )  NOT NULL ,";
creation_query += "\n barcode_price VARCHAR( 255 )  DEFAULT (0),";
creation_query += "\n PRIMARY KEY ( record_id ) );";
mDatabase.execSQL(creation_query);

//This is where the important part comes in regarding the question in this page:

//adding the missing primary key record_id in table receipt_barcode for older versions
        if (!is_column_exists(mDatabase, "receipt_barcode","record_id")){
            mDatabase.beginTransaction();
            try{
                Log.e("record_id", "creating");


                 creation_query="CREATE TEMPORARY TABLE t1_backup(";
                 creation_query+="record_id INTEGER        PRIMARY KEY AUTOINCREMENT,";
                 creation_query+="rcpt_id INT( 11 )       NOT NULL,";
                 creation_query+="barcode VARCHAR( 255 )  NOT NULL ,";
                 creation_query+="barcode_price VARCHAR( 255 )  NOT NULL DEFAULT (0) );";
                 mDatabase.execSQL(creation_query);

                 creation_query="INSERT INTO t1_backup(rcpt_id,barcode,barcode_price) SELECT rcpt_id,barcode,barcode_price  FROM receipt_barcode;";
                 mDatabase.execSQL(creation_query);

                 creation_query="DROP TABLE receipt_barcode;";
                 mDatabase.execSQL(creation_query);

                 creation_query="CREATE TABLE receipt_barcode (";
                 creation_query+="record_id INTEGER        PRIMARY KEY AUTOINCREMENT,";
                 creation_query+="rcpt_id INT( 11 )       NOT NULL,";
                 creation_query+="barcode VARCHAR( 255 )  NOT NULL ,";
                 creation_query+="barcode_price VARCHAR( 255 )  NOT NULL DEFAULT (0) );";
                 mDatabase.execSQL(creation_query);

                 creation_query="INSERT INTO receipt_barcode(record_id,rcpt_id,barcode,barcode_price) SELECT record_id,rcpt_id,barcode,barcode_price  FROM t1_backup;";
                 mDatabase.execSQL(creation_query);

                 creation_query="DROP TABLE t1_backup;";
                 mDatabase.execSQL(creation_query);


                 mdb.setTransactionSuccessful();
            } catch (Exception exception ){
                Log.e("table receipt_bracode", "Table receipt_barcode did not get a primary key (record_id");
                exception.printStackTrace();
            } finally {
                 mDatabase.endTransaction();
            }
3
superlinux

J'ai eu le même problème et la meilleure solution que j'ai trouvée est de créer d'abord la table définissant la clé primaire, puis d'utiliser insert into statement.

CREATE TABLE mytable (
field1 INTEGER PRIMARY KEY,
field2 TEXT
);

INSERT INTO mytable 
SELECT field1, field2 
FROM anothertable;
1
Spas

Je pense que l'ajout d'un index sur cette colonne peut donner à peu près le même effet.

0
Qiulang
sqlite>  create table t(id int, col2 varchar(32), col3 varchar(8));
sqlite>  insert into t values(1, 'he', 'ha');
sqlite>
sqlite>  create table t2(id int primary key, col2 varchar(32), col3 varchar(8));
sqlite>  insert into t2 select * from t;
sqlite> .schema
CREATE TABLE t(id int, col2 varchar(32), col3 varchar(8));
CREATE TABLE t2(id int primary key, col2 varchar(32), col3 varchar(8));
sqlite> drop table t;
sqlite> alter table t2 rename to t;
sqlite> .schema
CREATE TABLE IF NOT EXISTS "t"(id int primary key, col2 varchar(32), col3 varchar(8));
sqlite>
0
sanshaoye

J'ai utilisé la syntaxe CREATE TABLE AS pour fusionner plusieurs colonnes et j'ai rencontré le même problème. Voici un AppleScript que j'ai écrit pour accélérer le processus. 

set databasePath to "~/Documents/Databases/example.db"
set tableOne to "separate" -- Table from which you are pulling data
set tableTwo to "merged" -- Table you are creating
set {tempCol, tempColEntry, permColEntry} to {{}, {}, {}}
set permCol to {"id integer primary key"}

-- Columns are created from single items  AND from the last item of a list
-- {{"a", "b", "c"}, "d", "e"} Columns "a" and "b" will be merged into a new column "c".  tableTwo will have columns "c", "d", "e"

set nonCoal to {"City", "Contact", "Names", {"Address 1", "Address", "address one", "Address1", "Text4", "Address 1"}, {"E-Mail", "E-Mail Address", "Email", "Email Address", "EmailAddress", "Email"}, {"Zip", "Zip Code", "ZipCode", "Zip"}, {"Telephone", "BusinessPhone", "Phone", "Work Phone", "Telephone"}, {"St", "State", "State"}, {"Salutation", "Mr/Ms", "Mr/s", "Salutations", "Sautation", "Salutation"}}

-- Build the COALESCE statements
repeat with h from 1 to count of nonCoal
set aColumn to item h of nonCoal
if class of aColumn is not list then
    if (count of words of aColumn) > 1 then set aColumn to quote & aColumn & quote
    set end of tempCol to aColumn
    set end of permCol to aColumn
else
    set coalEntry to {}
    repeat with i from 1 to count of aColumn
        set coalCol to item i of aColumn as string
        if (count of words of coalCol) > 1 then set coalCol to quote & coalCol & quote
        if i = 1 then
            set end of coalEntry to "TRIM(COALESCE(" & coalCol & ", '') || \" \" || "
        else if i < ((count of aColumn) - 1) then
            set end of coalEntry to "COALESCE(" & coalCol & ", '') || \" \" || "
        else if i = ((count of aColumn) - 1) then
            set as_Col to item (i + 1) of aColumn as string
            if (count of words of as_Col) > 1 then set as_Col to quote & as_Col & quote
            set end of coalEntry to ("COALESCE(" & coalCol & ", '')) AS " & as_Col) & ""
            set end of permCol to as_Col
        end if
    end repeat
    set end of tempCol to (coalEntry as string)
end if
end repeat

-- Since there are ", '' within the COALESCE statement, you can't use "TID" and "as string" to convert tempCol and permCol for entry into sqlite3. I rebuild the lists in the next block.
repeat with j from 1 to count of tempCol
if j < (count of tempCol) then
    set end of tempColEntry to item j of tempCol & ", "
    set end of permColEntry to item j of permCol & ", "
else
    set end of tempColEntry to item j of tempCol
    set end of permColEntry to item j of permCol
end if
end repeat
set end of permColEntry to ", " & item (j + 1) of permCol
set permColEntry to (permColEntry as string)
set tempColEntry to (tempColEntry as string)

-- Create the new table with an "id integer primary key" column
set createTable to "create table " & tableTwo & " (" & permColEntry & "); "
do Shell script "sqlite3 " & databasePath & space & quoted form of createTable

-- Create a temporary table and then populate the permanent table
set createTemp to "create temp table placeholder as select " & tempColEntry & " from " & tableOne & ";  " & "insert into " & tableTwo & " select Null, * from placeholder;"
do Shell script "sqlite3 " & databasePath & space & quoted form of createTemp

--export the new table as a .csv file
do Shell script "sqlite3 -header -column -csv " & databasePath & " \"select * from " & tableTwo & " ; \"> ~/" & tableTwo & ".csv"
0
adayzdone

Utilisez un outil comme DB Browser for SQLite, il permet d’ajouter PK, AI par simple clic droit sur le tableau -> modifier.

0
codeapply