Я анализирую планы выполнения Oracle и обнаружил удивительный факт. Проверьте этот запрос. Подсказка - просто показать, что у меня есть индекс, и я ожидаю, что Oracle будет использовать его для сканирования диапазона:
// execute_at is of type DATE.
PreparedStatement stmt = connection.prepareStatement(
"SELECT /*+ index(my_table my_index) */ * " +
"FROM my_table " +
"WHERE execute_at > ? AND execute_at < ?");
Эти две привязки приводят к совершенно разному поведению (чтобы исключить проблемы просмотра переменных при привязке, я фактически ввел два жестких анализа):
// 1. with timestamps
stmt.setTimestamp(1, start);
stmt.setTimestamp(2, end);
// 2. with dates
stmt.setDate(1, start);
stmt.setDate(2, end);
1) С отметками времени я получаю INDEX FULL SCAN
и, таким образом, предикат фильтра
--------------------------------------------------------------
| Id | Operation | Name |
--------------------------------------------------------------
| 0 | SELECT STATEMENT | |
|* 1 | FILTER | |
| 2 | TABLE ACCESS BY INDEX ROWID| my_table |
|* 3 | INDEX FULL SCAN | my_index |
--------------------------------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------
1 - filter(:1<:2)"
3 - filter((INTERNAL_FUNCTION(""EXECUTE_AT"")>:1 AND
INTERNAL_FUNCTION(""EXECUTE_AT"")<:2))
2) С датами я получаю намного лучше INDEX RANGE SCAN
и предикат доступа
--------------------------------------------------------------
| Id | Operation | Name |
--------------------------------------------------------------
| 0 | SELECT STATEMENT | |
|* 1 | FILTER | |
| 2 | TABLE ACCESS BY INDEX ROWID| my_table |
|* 3 | INDEX RANGE SCAN | my_index |
--------------------------------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------
1 - filter(:1<:2)"
3 - access(""EXECUTE_AT"">:1 AND ""EXECUTE_AT""<:2)
Теперь мой пример - просто пример. Реальный запрос гораздо сложнее, где важно иметь RANGE SCANS
или UNIQUE SCANS
(в зависимости от предиката), а не FULL SCANS
.
Есть ли что-то, что я здесь неправильно понимаю? Может ли кто-нибудь указать мне лучшее решение / практику? Потому что в мире Java я думаю, что java.sql.Timestamp
гораздо более подходит, но большинство наших столбцов относятся к типу Oracle DATE
. Мы используем Java 6 и Oracle 11g