Skip to main content
Modifies the structure of an existing table without requiring you to recreate the table and copy its data.

Syntax

ALTER TABLE [schema_name.]table_name RENAME TO new_table_name;
ALTER TABLE [schema_name.]table_name RENAME COLUMN old_column_name TO new_column_name;
ALTER TABLE [schema_name.]table_name ADD COLUMN column_def;
ALTER TABLE [schema_name.]table_name DROP COLUMN column_name;

Parameters

ParameterDescription
schema_nameThe name of an attached database containing the table. Defaults to the main database if omitted.
table_nameThe name of the table to alter.

RENAME TO

Renames an existing table.
ALTER TABLE old_name RENAME TO new_name;
Renaming a table automatically updates all references to it in triggers, indexes, foreign key constraints, CHECK constraints, and views.
The new table name must not collide with an existing table, view, or index name in the same database.

Example

CREATE TABLE customers (
    id INTEGER PRIMARY KEY,
    name TEXT NOT NULL
);

ALTER TABLE customers RENAME TO clients;

-- The table is now accessible as 'clients'
SELECT * FROM clients;

RENAME COLUMN

Renames an existing column in a table.
ALTER TABLE table_name RENAME COLUMN old_column_name TO new_column_name;
Renaming a column automatically updates all references to it in indexes, triggers, CHECK constraints, and foreign key constraint definitions.

Restrictions

A column rename fails if:
  • The column does not exist in the table.
  • The new column name conflicts with an existing column in the same table.
  • A trigger on the table uses a qualified reference (e.g., table_name.column_name) to the column in its body.
  • A trigger’s WHEN clause references the column.

Example

CREATE TABLE products (
    id INTEGER PRIMARY KEY,
    product_name TEXT NOT NULL,
    price REAL
);

CREATE INDEX idx_name ON products(product_name);

ALTER TABLE products RENAME COLUMN product_name TO name;

-- The index is automatically updated
SELECT name, price FROM products;

ADD COLUMN

Adds a new column to an existing table. The new column is always appended as the last column.
ALTER TABLE table_name ADD COLUMN column_name type [constraints];

Restrictions

RequirementReason
No PRIMARY KEY constraintThe primary key of a table cannot be changed after creation.
No UNIQUE constraintAdding a UNIQUE column would require scanning all existing rows.
NOT NULL requires a non-NULL defaultExisting rows must have a valid value for the new column.
Default must be a constant expressionOnly literal values, signed literals, and parenthesized constant expressions are allowed. Functions like random() are not permitted.
CURRENT_TIME, CURRENT_DATE, CURRENT_TIMESTAMP defaults require an empty tableThese non-deterministic values cannot be retroactively applied to existing rows.
CHECK constraints require an empty tableThe constraint cannot be validated against existing rows.
No duplicate column nameThe column name must not already exist in the table.
No GENERATED ALWAYS ASGenerated columns cannot be added via ALTER TABLE.
STRICT tables require an explicit typeThe column type must be a valid STRICT type.

Supported column constraints on added columns

  • NOT NULL (with a non-NULL default)
  • DEFAULT expression
  • CHECK (expression)
  • REFERENCES (foreign key)
  • COLLATE

Examples

Add a simple column

ALTER TABLE users ADD COLUMN phone TEXT;

Add a column with a default value

ALTER TABLE orders ADD COLUMN priority INTEGER DEFAULT 0;

Add a NOT NULL column with a default

ALTER TABLE products ADD COLUMN in_stock INTEGER NOT NULL DEFAULT 1;

Add a column with a foreign key

ALTER TABLE orders ADD COLUMN warehouse_id INTEGER REFERENCES warehouses(id);

Add a column on a STRICT table

CREATE TABLE metrics (
    id INTEGER PRIMARY KEY,
    value REAL
) STRICT;

-- Must specify a valid STRICT type
ALTER TABLE metrics ADD COLUMN label TEXT DEFAULT 'unknown';

DROP COLUMN

Removes an existing column from a table.
ALTER TABLE table_name DROP COLUMN column_name;

Restrictions

A column cannot be dropped if any of the following conditions are true:
ConditionError
The column is a PRIMARY KEY or part of oneCannot drop PRIMARY KEY column.
The column has a UNIQUE constraintCannot drop UNIQUE column.
The column is referenced by an indexCannot drop indexed column.
The column is used in an expression indexCannot drop column used in expression index.
The column is named in a partial index WHERE clauseCannot drop column used in partial index.
The column is referenced in a CHECK constraintCannot drop column used in CHECK constraint.
The column is referenced by a foreign key in another tableCannot drop column used in foreign key.
The table has only one columnCannot drop the last remaining column.
The column appears in a trigger or viewCannot drop column referenced by trigger or view.

Examples

Drop a column

ALTER TABLE users DROP COLUMN phone;

Verify columns after dropping

CREATE TABLE example (
    id INTEGER PRIMARY KEY,
    name TEXT NOT NULL,
    old_field TEXT,
    active INTEGER DEFAULT 1
);

ALTER TABLE example DROP COLUMN old_field;

PRAGMA table_info(example);
-- cid | name   | type    | notnull | dflt_value | pk
-- 0   | id     | INTEGER | 0       | NULL       | 1
-- 1   | name   | TEXT    | 1       | NULL       | 0
-- 2   | active | INTEGER | 0       | 1          | 0

Examples

Complete schema evolution

-- Create initial table
CREATE TABLE articles (
    id INTEGER PRIMARY KEY,
    title TEXT NOT NULL,
    body TEXT
);

-- Add an author column with a foreign key
ALTER TABLE articles ADD COLUMN author_id INTEGER REFERENCES users(id);

-- Add a publication status
ALTER TABLE articles ADD COLUMN status TEXT NOT NULL DEFAULT 'draft';

-- Rename a column for clarity
ALTER TABLE articles RENAME COLUMN body TO content;

-- Add a nullable timestamp column
ALTER TABLE articles ADD COLUMN updated_at TEXT;

-- Rename the table
ALTER TABLE articles RENAME TO posts;

-- Final schema
PRAGMA table_info(posts);
-- cid | name       | type    | notnull | dflt_value | pk
-- 0   | id         | INTEGER | 0       | NULL       | 1
-- 1   | title      | TEXT    | 1       | NULL       | 0
-- 2   | content    | TEXT    | 0       | NULL       | 0
-- 3   | author_id  | INTEGER | 0       | NULL       | 0
-- 4   | status     | TEXT    | 1       | 'draft'    | 0
-- 5   | updated_at | TEXT    | 0       | NULL       | 0

See also

Build docs developers (and LLMs) love