JDBC Oracle-Fetch объяснить план для запроса
Мне интересно, как я могу получить план объяснения с помощью Java. Причина, по которой мне это нужно, заключается в том, что у нас есть структура, где специальные пользователи могут создавать отчеты. Эти отчеты иногда строят огромные запросы, в которых мы хотим объяснить на лету и сохранить стоимость. Таким образом, мы можем проанализировать запросы высокой стоимости позже и оптимизировать.
пример кода, который дает мне незаконное исключение столбца:
ResultSet rs = null;
try {
oracle = ConnectionManager.getConnection(ConnectionManager.Test);
pstmt = oracle.prepareStatement("begin execute immediate
'explain plan for SELECT 1 from Dual'; end;");
rs = pstmt.executeQuery();
while (rs.next()) {
System.out.println(rs.getString(1));
}
2 ответов
используйте этот:
oracle = ConnectionManager.getConnection(ConnectionManager.Test); stmt = oracle.createStatement() stmt.execute("explain plan for SELECT 1 from Dual"); rs = stmt.executeQuery("select plan_table_output from table(dbms_xplan.display())"); while (rs.next()) { System.out.println(rs.getString(1)); }
существует также способ показать реальный план, используемый для запуска последнего запроса в этом сеансе через DBMS_XPLAN.DISPLAY_CURSOR
. Запрос интереса не должен быть добавлен с EXPLAIN PLAN FOR
.
try (Statement st = connection.createStatement()) {
try (ResultSet rs = st.executeQuery(
"select plan_table_output from table(dbms_xplan.display_cursor())")) {
while (rs.next()) {
System.out.println(rs.getString(1));
}
}
}
обратите внимание, что пользователю необходимо предоставить следующие разрешения для того, чтобы использовать DBMS_XPLAN.DISPLAY_CURSOR
:
GRANT SELECT ON v_$session TO USER;
GRANT SELECT ON v_$sql_plan TO USER;
GRANT SELECT ON v_$sql_plan_statistics_all TO USER;
GRANT SELECT ON v_$sql TO USER;
кредиты перейти к https://myoracledbablog.wordpress.com/2016/07/26/dbms_xplan-and-the-user-has-no-select-privilege-on-v-error/.
см. также https://blogs.oracle.com/optimizer/how-do-i-display-and-read-the-execution-plans-for-a-sql-statement.
но я испытал это призвание dbms_xplan.display_cursor()
сразу после выполнения запроса может по-прежнему возвращать несвязанные результаты a в случае, если многопоточное приложение использует общий пул соединений.
этот можно обойти, выполнив поиск самого последнего sql_id
на v$sql
system view и предоставление его в качестве параметра dbms_xplan.display_cursor
.
Итак, вот готовый использовать java-код для регистрации фактического плана выполнения недавно выполненного запроса по его sql (возможно, частичному).
public void explainActualPlan(String sql, boolean sqlIsPartial, Logger log) {
if (!log.isTraceEnabled()) return;
try (Connection connection = dataSource.getConnection()) {
String sqlId;
String sqlFilter = sqlIsPartial
? "sql_text like '%' || ? || '%'"
//+ " and parsing_schema_id = sys_context('USERENV', 'CURRENT_SCHEMAID')"
: (sql.length() <= 1000 ? "sql_text = ?" : "dbms_lob.compare(sql_fulltext, ?) = 0");
try (PreparedStatement st = connection.prepareStatement(
"select sql_id from v$sql where " + sqlFilter +
" order by last_active_time desc fetch next 1 row only")) {
st.setString(1, sql);
try (ResultSet rs = st.executeQuery()) {
if (rs.next()) {
sqlId = rs.getString(1);
} else {
log.warn("Can't find sql_id for sql '{}'. Has it really been just executed?", sql);
return;
}
}
}
String planFormat = "TYPICAL";
if (sql.contains("GATHER_PLAN_STATISTICS")) {
planFormat += " ALLSTATS LAST +cost +bytes OUTLINE";
}
try (PreparedStatement st = connection.prepareStatement(
"select plan_table_output from table(dbms_xplan.display_cursor(" +
"sql_id => ?, format => '" + planFormat + "'))")) {
st.setString(1, sqlId);
try (ResultSet rs = st.executeQuery()) {
StringBuilder sb = new StringBuilder("Last query plan:\n");
while (rs.next()) {
sb.append(rs.getString(1)).append('\n');
}
log.trace(sb.toString());
}
}
} catch (Exception e) {
log.warn("Failed to explain query plan for '{}'", sql, e);
log.warn("Check that permissions are granted to the current db user:\n"
+ "GRANT SELECT ON v_$session TO <USER>;\n"
+ "GRANT SELECT ON v_$sql_plan TO <USER>;\n"
+ "GRANT SELECT ON v_$sql_plan_statistics_all TO <USER>;\n"
+ "GRANT SELECT ON v_$sql TO <USER>;\n"
);
}
}
некоторые замечания:
- Oracle всегда преобразует подготовленные параметры оператора из
?
to:n
синтаксис перед сохранением текста запроса вv$sql
, так что поиск по sql с?
' s не найдет никаких матчей - и
v$sql.sql_text
(усечено до первых 1000 символов) иv$sql.sql_fulltext
(полный CLOB) хранить текст sql без разрывов строк, поэтому может потребоваться выполнить a join СV$SQLTEXT_WITH_NEWLINES
в случае, если вы используете их в текст запроса -
LIKE
соответствие используется в частичном режиме, поэтому может потребоваться побег '%' и '_' специальные символы - я проверил, что Oracle позволяет включать любые неизвестные строки в комментариях подсказок, таких как
/*+ labuda FIRST_ROWS(200) */
. Он по-прежнему будет применять известные подсказки, если приложение является допустимым идентификатором (буквенно-цифровым и начинается с буквы). Это может быть полезно для отслеживания запросов интересов путем добавления хэш-кода в пункт подсказки. -
v@sql
может быть дополнительно отфильтрованand parsing_schema_id = sys_context('USERENV', 'CURRENT_SCHEMAID')
но это исключило бы некоторые планы, если экземпляр DB используется несколькими похожими приложениями в разных схемах с точно соответствующим sql запросы - приведенный выше код предоставляет дополнительные сведения в плане вывода в случае, если sql был выполнен с
GATHER_PLAN_STATISTICS
подсказка
вот пример приведенного выше вывода кода для запроса из мой еще один ответ:
22:54:53.558 TRACE o.f.adminkit.AdminKitSelectorQuery - Last query plan:
SQL_ID c67mmq4wg49sx, child number 0
-------------------------------------
select * from (select * from (select /*+ FIRST_ROWS(200)
INDEX_RS_DESC("FR_MESSAGE_PART" ("TS")) GATHER_PLAN_STATISTICS */ "ID",
"MESSAGE_TYPE_ID", "TS", "REMOTE_ADDRESS", "TRX_ID",
"PROTOCOL_MESSAGE_ID", "MESSAGE_DATA_ID", "TEXT_OFFSET", "TEXT_SIZE",
"BODY_OFFSET", "BODY_SIZE", "INCOMING" from "FR_MESSAGE_PART" where
"TS" + 0 >= :1 and "TS" < :2 and "ID" >= 376894993815568384 and "ID" <
411234940974268416 order by "TS" DESC) where ROWNUM <= 200) offset 180
rows
Plan hash value: 2499404919
----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
| Id | Operation | Name | Starts | E-Rows |E-Bytes|E-Temp | Cost (%CPU)| E-Time | Pstart| Pstop | A-Rows | A-Time | Buffers | OMem | 1Mem | Used-Mem |
----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | | | | 640K(100)| | | | 20 |00:00:00.01 | 322 | | | |
|* 1 | VIEW | | 1 | 200 | 130K| | 640K (1)| 00:00:26 | | | 20 |00:00:00.01 | 322 | | | |
| 2 | WINDOW NOSORT | | 1 | 200 | 127K| | 640K (1)| 00:00:26 | | | 200 |00:00:00.01 | 322 | 142K| 142K| |
| 3 | VIEW | | 1 | 200 | 127K| | 640K (1)| 00:00:26 | | | 200 |00:00:00.01 | 322 | | | |
|* 4 | COUNT STOPKEY | | 1 | | | | | | | | 200 |00:00:00.01 | 322 | | | |
| 5 | VIEW | | 1 | 780K| 487M| | 640K (1)| 00:00:26 | | | 200 |00:00:00.01 | 322 | | | |
|* 6 | SORT ORDER BY STOPKEY | | 1 | 780K| 68M| 89M| 640K (1)| 00:00:26 | | | 200 |00:00:00.01 | 322 | 29696 | 29696 |26624 (0)|
| 7 | PARTITION RANGE ITERATOR | | 1 | 780K| 68M| | 624K (1)| 00:00:25 | 3 | 2 | 400 |00:00:00.01 | 322 | | | |
|* 8 | COUNT STOPKEY | | 2 | | | | | | | | 400 |00:00:00.01 | 322 | | | |
|* 9 | TABLE ACCESS BY LOCAL INDEX ROWID| FR_MESSAGE_PART | 2 | 780K| 68M| | 624K (1)| 00:00:25 | 3 | 2 | 400 |00:00:00.01 | 322 | | | |
|* 10 | INDEX RANGE SCAN DESCENDING | IX_FR_MESSAGE_PART_TS | 2 | 559K| | | 44368 (1)| 00:00:02 | 3 | 2 | 400 |00:00:00.01 | 8 | | | |
----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
Outline Data
-------------
/*+
BEGIN_OUTLINE_DATA
IGNORE_OPTIM_EMBEDDED_HINTS
OPTIMIZER_FEATURES_ENABLE('12.1.0.2')
DB_VERSION('12.1.0.2')
OPT_PARAM('optimizer_dynamic_sampling' 0)
OPT_PARAM('_optimizer_dsdir_usage_control' 0)
FIRST_ROWS(200)
OUTLINE_LEAF(@"SEL")
OUTLINE_LEAF(@"SEL")
OUTLINE_LEAF(@"SEL")
OUTLINE_LEAF(@"SEL")
NO_ACCESS(@"SEL" "from$_subquery$_004"@"SEL")
NO_ACCESS(@"SEL" "from$_subquery$_001"@"SEL")
NO_ACCESS(@"SEL" "from$_subquery$_002"@"SEL")
INDEX_RS_DESC(@"SEL" "FR_MESSAGE_PART"@"SEL" ("FR_MESSAGE_PART"."TS"))
END_OUTLINE_DATA
*/
Predicate Information (identified by operation id):
---------------------------------------------------
1 - filter("from$_subquery$_004"."rowlimit_$$_rownumber">180)
4 - filter(ROWNUM<=200)
6 - filter(ROWNUM<=200)
8 - filter(ROWNUM<=200)
9 - filter("ID">=376894993815568384)
10 - access("TS"<:2)
filter((INTERNAL_FUNCTION("TS")+0>=:1 AND "TS"<:2))