SQL statements for generators

<< Generator basics | Firebird Generator Guide | Creating unique row IDs >>

SQL statements for generators

Statement overview

The name of a generator must be a usual DB meta-identifier: 31 chars maximum, no special characters except the underscore "_" (unless you use quoted identifiers). The SQL commands and statements that apply to generators are listed below. Their use will be discussed in some detail in the section Use of generator statements.

DDL (Data Definition Language) statements:

 CREATE GENERATOR <name>;
 SET GENERATOR <name> TO <value>;
 DROP GENERATOR <name>;

DML (Data Manipulation Language) statements in client SQL:

 SELECT GEN_ID( <GeneratorName>, <increment> ) FROM RDB$DATABASE;

DML statements in PSQL (Procedural SQL), available in stored procedures and triggers):

 <intvar> = GEN_ID( <GeneratorName>, <increment> );

Firebird 2 recommended syntax

Although the traditional syntax is still fully supported in Firebird 2, these are the recommended DDL equivalents:

 CREATE SEQUENCE <name>;
 ALTER SEQUENCE <name> RESTART WITH <value>;
 DROP SEQUENCE <name>;

And for the DML statements:

 SELECT NEXT VALUE FOR <SequenceName> FROM RDB$DATABASE;

 <intvar> = NEXT VALUE FOR <SequenceName>;

Currently the new syntax does not support an increment other than 1. This limitation will be lifted in a future version. In the meantime, use GEN_ID if you need to apply another increment value.

back to top of page

Use of generator statements

The availability of statements and functions depends on whether you use them in:

  • Client SQL: The language you use when you, as a client, talk to a Firebird server.
  • PSQL: The server-side programming language used in Firebird stored procedures and triggers.

Creating a generator (INSERT)

Client SQL

 CREATE GENERATOR <GeneratorName>;

Preferred for Firebird 2 and up:

 CREATE SEQUENCE <SequenceName>;

PSQL

Not possible. Since you cannot change database metadata inside stored procedures or triggers, you cannot create generators there either.

Note: In Firebird 1.5 and up, you can circumvent this limitation with the EXECUTE STATEMENT feature.

back to top of page

Getting the current value (SELECT)

Client SQL

 SELECT GEN_ID( <GeneratorName>, 0 ) FROM RDB$DATABASE;

This syntax is still the only option in Firebird 2.

Note: In Firebird's command-line tool isql there are two additional commands for retrieving current generator values:

 SHOW GENERATOR <GeneratorName>;
 SHOW GENERATORS;
The first reports the current value of the named generator. The second does the same for all non-system generators in the database.
The preferred Firebird 2 equivalents are, as you could guess:
 SHOW SEQUENCE <SequenceName>;
 SHOW SEQUENCES;
Please notice again that these SHOW... commands are only available in the Firebird isql tool. Unlike GEN_ID, you can't use them from within other clients (unless these clients are isql frontends).

PSQL

 <intvar> = GEN_ID( <GeneratorName>, 0 );

Firebird 2: same syntax.

back to top of page

Generating the next value (UPDATE + SELECT)

Just like getting the current value, this is done with GEN_ID, but now you use an increment value of 1. Firebird will:

  1. get the current generator value;
  2. increment it by 1;
  3. return the incremented value.

Client SQL'

 SELECT GEN_ID( <GeneratorName>, 1 ) FROM RDB$DATABASE;

The new syntax, which is preferred for Firebird 2, is entirely different:

 SELECT NEXT VALUE FOR <SequenceName> FROM RDB$DATABASE;

PSQL

 <intvar> = GEN_ID( <GeneratorName>, 1 );

Preferred for Firebird 2 and up:

 <intvar> = NEXT VALUE FOR <SequenceName>;

back to top of page

Setting a generator directly to a certain value (UPDATE)

Client SQL

 SET GENERATOR <GeneratorName> TO <NewValue>;

This is useful to preset generators to a value other than 0 (which is the default value after you created it) in e.g. a script to create the database. Just like CREATE GENERATOR, this is a DDL (not DML) statement.

Preferred syntax for Firebird 2 and up:

 ALTER SEQUENCE <SequenceName> RESTART WITH <NewValue>;

PSQL

 GEN_ID( <GeneratorName>, <NewValue> - GEN_ID( <GeneratorName>, 0 ) );

Warning: This is more of a dirty little trick to do what you normally cannot and should not do in stored procedures and triggers: setting generators. They are for getting, not setting values.

back to top of page

Dropping a generator (DELETE)

Client SQL

 DROP GENERATOR <GeneratorName>;

Preferred for Firebird 2 and up:

 DROP SEQUENCE <SequenceName>;

PSQL

Not possible, unless... (Same explanation as with Create: you can't – or rather, shouldn't – change metadata in PSQL.)

Dropping a generator does not free the space it occupied for use by a new generator. In practice this rarely hurts, because most databases don't have the tens of thousands of generators that Firebird allows, so there's bound to be room for more anyway. But if your database does risk to hit the 32767 ceiling, you can free up dead generator space by performing a backup-restore cycle. This will neatly pack the RDB$GENERATORS table, re-assigning a contiguous series of IDs. Depending on the situation, the restored database may also need less pages for the generator values.

Dropping generators in old InterBase® and Firebird versions

InterBase® 6 and earlier, as well as early pre-1.0 Firebird versions, do not have a DROP GENERATOR command. The only way to drop a generator in these versions is:

 DELETE FROM RDB$GENERATORS WHERE RDB$GENERATOR_NAME = '<GeneratorName>';

...followed by a backup and restore.

In these versions, with the maximum number of generators typically a couple of hundred, it is much more likely that the need will arise to reuse space from deleted generators.

See also:
Generator

back to top of page
<< Generator basics | Firebird Generator Guide | Creating unique row IDs >>