8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-22 22:43:03 +01:00
firebird-mirror/doc/README.Optimizer.txt

132 lines
4.2 KiB
Plaintext
Raw Normal View History

2005-02-17 01:19:20 +01:00
Optimizer enhancements in FB2.0
The Optimizer has two different optimizer "modules" depending on the ODS of the
database. The reason for this is that in ODS 11 segment-selectivity is introduced.
Improvements for every ODS:
A) Deliver allowable comparisons from HAVING clause to WHERE clause so that indexes
could be used. Only simple comparisons are checked if they are deliverable. At
least 1 side of the comparison should be a field contained in the GROUP BY clause.
Note that WHERE clauses on aggregate views are also put in the HAVING clause.
SELECT Count(*) FROM RelationCategories GROUP BY CategoryID HAVING CategoryID = 5
B) Try to deliver conjuntions made on a UNION inside every single stream.
SELECT Count(*) FROM
(SELECT RelationID FROM Relations
UNION ALL
SELECT CategoryID FROM Categories) dt (ID)
WHERE
dt.ID = 1
C) Cardinality calculation has been improved for a better estimated value.
D) MERGE/SORT operation may now be generated for inner joins using equality
2005-02-21 07:01:01 +01:00
comparison on expressions.
2005-02-17 01:19:20 +01:00
SELECT Count(*) FROM
Relations r
JOIN RelationCategories rc ON (rc.RelationID + 0 = r.RelationID + 0)
WHERE
rc.CategoryID = 3
2005-02-21 07:01:01 +01:00
E) When a UNIQUE index used on a relation can be fully matched (equality), use this
2005-02-17 01:19:20 +01:00
index and forget any possible indexes.
F) Indexed order read with an OUTER JOIN.
SELECT * FROM
Relations r
LEFT JOIN RelationCategories rc ON (rc.RelationID = r.RelationID and rc.CategoryID = 3)
ORDER BY
r.LASTNAME
Note: Patch provided by <Radek Palmowski>
Improvements for ODS 11:
G) Ignore NULL values in index scan for operators other than IS NULL and Equivalent.
In previous versions with a NULL-able index NULL-keys could be put in the bitmap
for record retrieval.
Note: This works currently only at single segment indexes.
2005-02-21 07:01:01 +01:00
H) Calculate with more accuracy selectivity for compound indexes which are partially
2005-02-17 01:19:20 +01:00
matched with conjunctions. Due this we can now use cost-based approach also for
partially matched indexes.
Example:
SELECT * FROM RelationCategories WHERE CategoryID = 1
I) IS NULL can be used inside compound indexes on every segment (assuming previous
segments are matched too).
Example:
SELECT * FROM Relations WHERE LastName = 'Goebel' and MiddleName IS NULL
J) STARTING WITH can be used inside compound indexes on every segment, but only the
last matching segment.
Example:
SELECT * FROM Relations WHERE FirstName = 'Joe' and LastName STARTING WITH 'D'
K) Use AND conjunctions also inside OR conjunctions to help with matching on a
2005-02-21 07:01:01 +01:00
compound index. The same calculation for every left and right node of an OR
2005-02-17 01:19:20 +01:00
conjunction is used as for the AND conjunctions.
Example:
SELECT * FROM RelationCategories WHERE CategoryID = 3 and RelationID IN (123, 720, 708)
Above select will use 3 times the index created for the primary key.
L) JOIN order calculation doesn't prefer primary indexes, but uses cost based
calculation for every relation (selectivity, cardinality and number of indexes).
Selectivity for a node is calculated over all selectivities used in that node.
Indexes matched with IS NULL and STARTING WITH are now also considered in this
calculation. Less JOIN order combinations are being calculated.
SELECT * FROM
Relations r
JOIN RelationCategories rc ON (rc.RelationID = r.RelationID and rc.CATEGORYID = 1)
WHERE
r.FIRSTNAME LIKE 'J%'
Example metadata:
CREATE TABLE Relations (
RelationID INT NOT NULL PRIMARY KEY,
FirstName VARCHAR(35),
MiddleName VARCHAR(35),
LastName VARCHAR(35)
);
CREATE INDEX IDX_FIRSTNAME ON Relations (FirstName);
CREATE INDEX IDX_LASTNAME ON Relations (LastName);
CREATE INDEX IDX_FIRST_LASTNAME ON Relations (FirstName, LastName);
CREATE INDEX IDX_LAST_MIDDLENAME ON Relations (MiddleName, LastName);
CREATE TABLE Categories (
CategoryID INT NOT NULL PRIMARY KEY,
Description VARCHAR(30)
);
CREATE TABLE RelationCategories (
RelationID INT NOT NULL,
CategoryID INT NOT NULL,
CONSTRAINT FK_RC_RELATIONS FOREIGN KEY (RelationID)
REFERENCES Relations (RelationID),
CONSTRAINT FK_RC_CATEGORIES FOREIGN KEY (CategoryID)
REFERENCES Categories (CategoryID),
PRIMARY KEY (CategoryID, RelationID)
);