Учитывая
Я думал о OFFSET / LIMIT, но это означало бы, что я знаю количество последних строк, чего я не знаю.
Вы могли быподдерживать счет, имея минимальную таблицу с одной строкой на таблицу для подсчета двумя столбцами, именем таблицы (не нужно, если бы была только одна таблица) и счетом.Два ТРИГГЕРА могут автоматически поддерживать счет.
Альтернативой, но аналогично не такой интенсивной (чтобы выяснить представление о счетчике), если вы используете AUTOINCREMENT, было бы получить оценку (максимально возможную без каких-либо удалений) путемполучить доступ к столбцу seq для таблицы sqlite_sequence table.
Возможно, рассмотрим следующий пример обоих: -
DROP TRIGGER IF EXISTS newmain1row;
DROP TRIGGER IF EXISTS deletedmain1row;
DROP TABLE IF EXISTS rowcount;
DROP TABLE IF EXISTS main1;
DROP TABLE IF EXISTS main2;
CREATE TABLE IF NOT EXISTS main1 (id INTEGER PRIMARY KEY, mycolumn TEXT);
CREATE TABLE IF NOT EXISTS main2 (id INTEGER PRIMARY KEY AUTOINCREMENT, mycolumn TEXT); -- is it slower/faster than
CREATE TABLE IF NOT EXISTS rowcount (tablename TEXT PRIMARY KEY, rowcount INTEGER) WITHOUT ROWID;
INSERT INTO rowcount VALUES('main1',0); -- initialise rowcount table
-- Trigger for when a new row is added to the table
CREATE TRIGGER newmain1row
AFTER INSERT ON main1 BEGIN
UPDATE rowcount SET rowcount = rowcount + 1 WHERE tablename = 'main1';
END
;
-- Trigger for when a row is deleted from the table
CREATE TRIGGER deletedmain1row
AFTER DELETE ON main1 BEGIN
UPDATE rowcount SET rowcount = rowcount - 1 WHERE tablename = 'main1';
END
;
-- populate the two example main tables twice so in reverse order 2nd time
-- with a million rows so 2 million per table
WITH RECURSIVE cte1(counter) AS
(
SELECT 1
UNION ALL SELECT counter+1 FROM cte1 WHERE counter < 1000000
)
INSERT INTO main1 (mycolumn) SELECT counter FROM cte1
;
WITH RECURSIVE cte1(counter) AS
(
SELECT 1
UNION ALL SELECT counter+1 FROM cte1 WHERE counter < 1000000
)
INSERT INTO main2 (mycolumn) SELECT counter FROM cte1
;
WITH RECURSIVE cte1(counter) AS
(
SELECT 1
UNION ALL SELECT counter+1 FROM cte1 WHERE counter < 1000000
)
INSERT INTO main2 (mycolumn) SELECT counter FROM cte1
;
WITH RECURSIVE cte1(counter) AS
(
SELECT 1
UNION ALL SELECT counter+1 FROM cte1 WHERE counter < 1000000
)
INSERT INTO main1 (mycolumn) SELECT counter FROM cte1
;
-- DELETE 40,000 rows
DELETE FROM main1 WHERE CAST(mycolumn AS INTEGER) % 50 = 0;
DELETE FROM main2 WHERE CAST(mycolumn AS INTEGER) % 50 = 0;
-- get the number of rows
SELECT rowcount FROM rowcount WHERE tablename = 'main1';
SELECT seq FROM sqlite_sequence WHERE name = 'main2';
This: -
- УДАЛЯЕТ ТРИГГЕРЫ и ТАБЛИЦЫ, позволяя перезапускать его.
- СОЗДАЕТ 2 основных ТАБЛИЦЫ (main1 и main2) для каждого метода, т.е. оба в основном одинаковы, за исключением последнего с AUTOINCREMENT
- СОЗДАЕТ таблицу счетчиков для первого подхода и добавляет строку со счетчиком 0 для таблицы main1
- СОЗДАЕТ 2 ТРИГГЕРА для случая, когда в первой (main1) таблице вставлена строка (увеличивает счет) инапример, когда строка удаляется (уменьшает количество).
- вставляет 2 000 000 строк в каждую таблицу 4 партиями по 1 000 000 строк, обращаяrder, возможно, для лучшего понимания времени.
- удаляет 40000 строк из каждой таблицы.
- извлекает счетчики из каждой (обратите внимание, что счетчик, возвращаемый во втором методе, равен 2 000 000 вместо 1 960 000 какэто последний выделенный идентификатор).
Журнал: -
DROP TRIGGER IF EXISTS newmain1row
> OK
> Time: 0.187s
DROP TRIGGER IF EXISTS deletedmain1row
> OK
> Time: 0.084s
DROP TABLE IF EXISTS rowcount
> OK
> Time: 0.153s
DROP TABLE IF EXISTS main1
> OK
> Time: 2.534s
DROP TABLE IF EXISTS main2
> OK
> Time: 2.547s
CREATE TABLE IF NOT EXISTS main1 (id INTEGER PRIMARY KEY, mycolumn TEXT)
> OK
> Time: 0.158s
CREATE TABLE IF NOT EXISTS main2 (id INTEGER PRIMARY KEY AUTOINCREMENT, mycolumn TEXT)
> OK
> Time: 0.167s
-- is it slower/faster than
CREATE TABLE IF NOT EXISTS rowcount (tablename TEXT PRIMARY KEY, rowcount INTEGER) WITHOUT ROWID
> OK
> Time: 0.167s
INSERT INTO rowcount VALUES('main1',0)
> Affected rows: 1
> Time: 0.165s
-- initialise rowcount table
-- Trigger for when a new row is added to the table
CREATE TRIGGER newmain1row
AFTER INSERT ON main1 BEGIN
UPDATE rowcount SET rowcount = rowcount + 1 WHERE tablename = 'main1';
END
> Affected rows: 1
> Time: 0.086s
-- Trigger for when a row is deleted from the table
CREATE TRIGGER deletedmain1row
AFTER DELETE ON main1 BEGIN
UPDATE rowcount SET rowcount = rowcount - 1 WHERE tablename = 'main1';
END
> Affected rows: 1
> Time: 0.096s
-- populate the two example main tables twice so in reverse order 2nd time
-- with a million rows so 2 million per table
WITH RECURSIVE cte1(counter) AS
(
SELECT 1
UNION ALL SELECT counter+1 FROM cte1 WHERE counter < 1000000
)
INSERT INTO main1 (mycolumn) SELECT counter FROM cte1
> Affected rows: 1000000
> Time: 1.199s
WITH RECURSIVE cte1(counter) AS
(
SELECT 1
UNION ALL SELECT counter+1 FROM cte1 WHERE counter < 1000000
)
INSERT INTO main2 (mycolumn) SELECT counter FROM cte1
> Affected rows: 1000000
> Time: 0.811s
WITH RECURSIVE cte1(counter) AS
(
SELECT 1
UNION ALL SELECT counter+1 FROM cte1 WHERE counter < 1000000
)
INSERT INTO main2 (mycolumn) SELECT counter FROM cte1
> Affected rows: 1000000
> Time: 1.058s
WITH RECURSIVE cte1(counter) AS
(
SELECT 1
UNION ALL SELECT counter+1 FROM cte1 WHERE counter < 1000000
)
INSERT INTO main1 (mycolumn) SELECT counter FROM cte1
> Affected rows: 1000000
> Time: 1.177s
DELETE FROM main1 WHERE CAST(mycolumn AS INTEGER) % 50 = 0
> Affected rows: 40000
> Time: 2.82s
DELETE FROM main2 WHERE CAST(mycolumn AS INTEGER) % 50 = 0
> Affected rows: 40000
> Time: 2.649s
-- get the number of rows
SELECT rowcount FROM rowcount WHERE tablename = 'main1'
> OK
> Time: 0s
SELECT seq FROM sqlite_sequence WHERE name = 'main2'
> OK
> Time: 0s
Общее время, которое заняло чуть более 16 секунд.
2 результата1,960,000 и 2,000,000 соответственно.