作者:Maria Colgan
介绍
你是否经历过因为执行计划变得更差而导致的性能退化?如果是的话,那么我们有一个称为SQL PLAN MANAGEMENT(SPM)的精致解决方案给到你。接下来的4篇博客将涵盖SPM的详情。让我们从回顾执行计划改变的主要原因开始。
执行计划改变的发生可以归因于多种系统变化。比如,你也许(手动或自动)更新了部分对象的统计信息,或者修改了少量优化器相关的参数。而更显著的改变是数据库升级(比如从11g升级到12c)。所有这些变化都会潜在导致很多SQL语句生成新的执行计划。大多数新的执行计划都是明显的改进,因为它们是针对新的系统环境而定制的,但有些可能更糟,从而导致性能下降。正是后者导致许多dba彻夜难眠。
处理这些退化问题,DBA有几个选项。然而,大多数dba想要的很简单:只有当计划能够带来性能提升时,才应该改变计划。换句话说,优化器不应该挑选坏的执行计划。
在这个系列文章中的第一篇,会描述SQL PLAN MANAGEMENT的概念以及如何创建SQL PLAN BASELINES(SQL执行计划基线)。第二篇会描述Sql Plan Baseline是如何以及何时使用。第三篇将讨论演进,即在SQL计划基线中添加新的和改进的执行计划的过程。最后,第四部分将描述用户界面以及与其他Oracle对象(如存储的大纲)的交互。
SQL PLAN MANAGEMENT
SQL Plan Management (SPM)允许数据库用户为一组SQL语句保持稳定而最佳的性能。SPM融合了计划适应性和计划稳定性的优点,同时又避免了它们的缺点。它有两个主要目标:
- 当面对数据库系统变化时,防止性能退化。
- 通过友好地适应数据库系统变化,提供性能改善。
托管SQL语句是启用了SPM的语句。SPM可以配置为自动工作,也可以全部或部分手动控制(稍后介绍)。SPM通过启用对托管SQL语句的执行计划更改的检测,来帮助防止性能退化。为此,SPM在磁盘上维护一个执行计划历史记录,其中包含了为每个托管SQL语句生成的不同执行计划。Oracle优化器的增强版称为SPM感知优化器,它访问、使用和管理存储在名为SQL Management Base(SMB)的存储库中的信息。
执行计划历史使SPM感知优化器能够确定,使用基于成本的方法生成的最佳成本计划是否是全新的计划。一个全新的执行计划代表了一个有可能导致性能退化的执行计划。基于这个原因,SPM感知优化器不会选择一个全新的最佳成本执行计划。相反,它会从一系列已接受的执行计划中进行选择。一个已接受的执行计划是一个已经被证实不会导致性能下降或被指定具有良好性能的执行计划。一组可接受的执行计划称为SQL执行计划基线,它是执行计划历史的子集。
全新执行计划将作为未接受计划添加到执行计划历史记录中。稍后,SPM实用程序将验证其性能,并将其作为不可接受的执行计划(如果它会导致性能退化),或者将其更改为可接受的执行计划(如果它将提供性能改进)。执行计划性能验证过程确保了计划的稳定性和执行计划的适应性。
下图展示了SMB中保存的三条语句的执行计划历史。每一个执行计划历史由已接受的执行计划(SQL Plan Baseline)和未接受的执行计划组成。
你可以通过若干种方法创建Sql Plan Baseline: 通过SQL Tuing Set(STS,SQL调优集);通过缓存的游标;从一个库中导出然后导入到另一个库;自动为每条语句创建。让我们逐一看一看。此博客中的示例使用了Oracle数据库示例SCHEMA,因此您可以自己尝试它们。
通过SQL Tuing Set(STS)创建Sql plan Baselines
如果正在升级,则您可能已经有一个包含部分或全部SQL语句的STS。该STS可能包含令人满意的执行计划。我们把这个STS称为 MY_STS。可以从该STS创建SQL计划基线,如下所示:
SQL> variable pls number;
SQL> exec :pls := dbms_spm.load_plans_from_sqlset(sqlset_name => 'MY_STS', -
> basic_filter => 'sql_text like ''select%p.prod_name%''');
这将会为所有符合指定过滤条件的语句创建SQL Plan Baselines.
通过缓存的游标创建SQL plan baselines
您可以为当前缓存中的任意游标自动创建SQL plan baselines,如下所示:
SQL> exec :pls := dbms_spm.load_plans_from_cursor_cache( -
> attribute_name => 'SQL_TEXT', -
> attribute_value => 'select%p.prod_name%');
这将会为所有SQL文本匹配指定字符串的SQL创建SQL plan baseline。该函数有多个重载,允许你通过其它游标属性进行过滤。
通过中间表创建SQL plan baselines
如果你已经有SQL plan baseline了(比如在11g的测试系统中),你可以导出它们到另一个系统中(比如一个生产系统)
首先,在测试系统中,创建一个中间表并将你要导出的SQL plan baseline打包:
SQL> exec dbms_spm.create_stgtab_baseline(table_name => 'MY_STGTAB', -
> table_owner => 'SH');
PL/SQL procedure successfully completed.
SQL> exec :pls := dbms_spm.pack_stgtab_baseline( -
> table_name => 'MY_STGTAB', -
> table_owner => 'SH', -
> sql_text => 'select%p.prod_name%');
这将会对所有匹配指定过滤条件的语句的SQL plan baseline进行打包。中间表MY_STGTAB是一个常规表,你可以使用数据泵导出的功能将其导到生产系统中。
在生产系统上,就可以解包中间表来创建SQL plan baseline:
SQL> exec :pls := dbms_spm.unpack_stgtab_baseline( -
> table_name => 'MY_STGTAB', -
> table_owner => 'SH', -
> sql_text => 'select%p.prod_name%');
这将解包中间表并创建SQL plan baseline。注意,解包中间表时使用的过滤条件是可选项,是可以不同于打包时的条件的。这意味着,你可以打包若干个SQL plan baseline到中间表,然后仅解包其子集到目标系统中。
自动创建SQL plan baseline
您可以通过设置参数optimizer_capture_sql_plan_baselines为TRUE(默认为FALSE),为所有重复执行过的语句自动创建SQL plan baseline。这些语句首次被捕获到的执行计划会自动做为可接受的执行计划并成为SQL plan baseline的一部分。因此,只有在你确认这些默认的执行计划性能良好时,才应打开这个参数。
从以前的数据库版本升级后,可以使用自动计划捕获模式。将optimizer_features_enable设置为早期版本并执行工作负载。每个重复执行的语句都将捕获其计划,从而创建SQL plan baseline。在确定工作负载中的所有语句都有机会执行之后,可以将optimizer_features_enable重置为其默认值。
请注意,此自动计划捕获仅适用于重复执行的语句,即至少执行两次的语句。只执行一次的语句将不会从SQL plan baseline中获益,因为接受的计划只在后续的硬解析中使用。
以下示例显示同一语句执行两次时自动捕获计划:
SQL> alter session set optimizer_capture_sql_plan_baselines = true;
Session altered.
SQL> var pid number
SQL> exec :pid := 100;
PL/SQL procedure successfully completed.
SQL> select p.prod_name, s.amount_sold, t.calendar_year
2 from sales s, products p, times t
3 where s.prod_id = p.prod_id
4 and s.time_id = t.time_id
5 and p.prod_id < :pid;
PROD_NAME AMOUNT_SOLD CALENDAR_YEAR
--------- ----------- -------------
...
9 rows selected.
SQL> select p.prod_name, s.amount_sold, t.calendar_year
2 from sales s, products p, times t
3 where s.prod_id = p.prod_id
4 and s.time_id = t.time_id
5 and p.prod_id < :pid;
PROD_NAME AMOUNT_SOLD CALENDAR_YEAR
--------- ----------- -------------
...
9 rows selected.
SQL> alter session set optimizer_capture_sql_plan_baselines = false;
Session altered
如果一个语句存在存储大纲,且参数use_stored_outlines为TRUE,那么自动计划捕获并不工作。在这种情况下,使用DBMS_SQLTUNE包中的函数capture_cursor_cache_sqlset()将计划增量捕获到STS中。在将工作负载的执行计划收集到STS中之后,使用前面描述的方法手动创建SQL计划基线。然后,禁用存储大纲或将use_stored_outlines设置为FALSE。从现在起,SPM将管理您的工作负载,存储大纲将不会用于这些语句。
在本文中,我们了解了如何创建SQL plan baseline。下一步,我们将描述SPM感知优化器是如何使用SQL plan baseline的(第二篇 SQL Plan Management (4-2) SPM感知优化器)。
原文链接:https://blogs.oracle.com/optimizer/sql-plan-management-part-1-of-4-creating-sql-plan-baselines
原文内容:
SQL Plan Management (Part 1 of 4) Creating SQL plan baselines
Maria Colgan
Master Product Manager
Introduction
Do you ever experience performance regressions because an execution plan has changed for the worse? If you have, then we have an elegant solution for you called SQL Plan Management (SPM). The next four posts on our blog will cover SPM in detail. Let’s begin by reviewing the primary causes for plan changes.
Execution plan changes occur due to various system changes. For example, you might have (manually or automatically) updated statistics for some objects, or changed a few optimizer-related parameters. A more dramatic change is a database upgrade (say from 11g to 12c). All of these changes have the potential to cause new execution plans to be generated for many of your SQL statements. Most new plans are obviously improvements because they are tailored to the new system environment, but some might be worse leading to performance regressions. It is the latter that cause sleepless nights for many DBAs.
DBAs have several options for addressing these regressions. However, what most DBAs want is simple: plans should only change when they will result in performance gains. In other words, the optimizer should not pick bad plans, period.
This first post in our series, describes the concepts of SQL Plan Management and how to create SQL plan baselines. The second part will describe how and when these SQL plan baselines are used. The third part will discuss evolution, the process of adding new and improved plans to SQL plan baselines. Finally, the fourth part will describe user interfaces and interactions with other Oracle objects (like stored outlines).
SQL Plan Management
SQL Plan Management (SPM) allows database users to maintain stable yet optimal performance for a set of SQL statements. SPM incorporates the positive attributes of plan adaptability and plan stability, while simultaneously avoiding their shortcomings. It has two main objectives:
- prevent performance regressions in the face of database system changes
- offer performance improvements by gracefully adapting to database system changes
A managed SQL statement is one for which SPM has been enabled. SPM can be configured to work automatically or it can be manually controlled either wholly or partially (described later). SPM helps prevent performance regressions by enabling the detection of plan changes for managed SQL statements. For this purpose, SPM maintains, on disk, a plan history consisting of different execution plans generated for each managed SQL statement. An enhanced version of the Oracle optimizer, called SPM aware optimizer, accesses, uses, and manages this information which is stored in a repository called the SQL Management Base (SMB).
The plan history enables the SPM aware optimizer to determine whether the best-cost plan it has produced using the cost-based method is a brand new plan or not. A brand new plan represents a plan change that has potential to cause performance regression. For this reason, the SPM aware optimizer does not choose a brand new best-cost plan. Instead, it chooses from a set of accepted plans. An accepted plan is one that has been either verified to not cause performance regression or designated to have good performance. A set of accepted plans is called a SQL plan baseline, which represents a subset of the plan history.
A brand new plan is added to the plan history as a non-accepted plan. Later, an SPM utility verifies its performance, and keeps it as a non-accepted plan if it will cause a performance regression, or changes it to an accepted plan if it will provide a performance improvement. The plan performance verification process ensures both plan stability and plan adaptability.
The figure below shows the SMB containing the plan history for three SQL statements. Each plan history contains some accepted plans (the SQL plan baseline) and some non-accepted plans.
You can create a SQL plan baseline in several ways: using a SQL Tuning Set (STS); from the cursor cache; exporting from one database and importing into another; and automatically for every statement. Let’s look at each in turn. The examples in this blog entry use the Oracle Database Sample Schemas so you can try them yourself.
Creating SQL plan baselines from STS
If you are upgrading you might already have an STS containing some or all of your SQL statements. This STS might contain plans that perform satisfactorily. Let’s call this STS MY_STS. You can create a SQL plan baseline from this STS as follows:
SQL> variable pls number;
SQL> exec :pls := dbms_spm.load_plans_from_sqlset(sqlset_name => 'MY_STS', -
> basic_filter => 'sql_text like ''select%p.prod_name%''');
This will create SQL plan baselines for all statements that match the specified filter.
Creating SQL plan baselines from cursor cache
You can automatically create SQL plan baselines for any cursor that is currently in the cache as follows:
SQL> exec :pls := dbms_spm.load_plans_from_cursor_cache( -
> attribute_name => 'SQL_TEXT', -
> attribute_value => 'select%p.prod_name%');
This will create SQL plan baselines for all statements whose text matches the specified string. Several overloaded variations of this function allow you to filter on other cursor attributes.
Creating SQL plan baselines using a staging table
If you already have SQL plan baselines (say on an 11g test system), you can export them to another system (a production system for instance).
First, on the test system, create a staging table and pack the SQL plan baselines you want to export:
SQL> exec dbms_spm.create_stgtab_baseline(table_name => 'MY_STGTAB', -
> table_owner => 'SH');
PL/SQL procedure successfully completed.
SQL> exec :pls := dbms_spm.pack_stgtab_baseline( -
> table_name => 'MY_STGTAB', -
> table_owner => 'SH', -
> sql_text => 'select%p.prod_name%');
This will pack all SQL plan baselines for statements that match the specified filter. The staging table, MY_STGTAB, is a regular table that you should export to the production system using Datapump Export.
On the production system, you can now unpack the staging table to create the SQL plan baselines:
SQL> exec :pls := dbms_spm.unpack_stgtab_baseline( -
> table_name => 'MY_STGTAB', -
> table_owner => 'SH', -
> sql_text => 'select%p.prod_name%');
This will unpack the staging table and create SQL plan baselines. Note that the filter for unpacking the staging table is optional and may be different than the one used during packing. This means that you can pack several SQL plan baselines into a staging table and selectively unpack only a subset of them on the target system.
Creating SQL plan baselines automatically
You can create SQL plan baselines for all repeatable statements automatically by setting the parameter optimizer_capture_sql_plan_baselines to TRUE (default is FALSE). The first plan captured for any statement is automatically accepted and becomes part of the SQL plan baseline, so enable this parameter only when you are sure that the default plans are performing well.
You can use the automatic plan capture mode when you have upgraded from a previous database version. Set optimizer_features_enable to the earlier version and execute your workload. Every repeatable statement will have its plan captured thus creating SQL plan baselines. You can reset optimizer_features_enable to its default value after you are sure that all statements in your workload have had a chance to execute.
Note that this automatic plan capture occurs only for repeatable statements, that is, statements that are executed at least twice. Statements that are only executed once will not benefit from SQL plan baselines since accepted plans are only used in subsequent hard parses.
The following example shows a plan being captured automatically when the same statement is executed twice:
SQL> alter session set optimizer_capture_sql_plan_baselines = true;
Session altered.
SQL> var pid number
SQL> exec :pid := 100;
PL/SQL procedure successfully completed.
SQL> select p.prod_name, s.amount_sold, t.calendar_year
2 from sales s, products p, times t
3 where s.prod_id = p.prod_id
4 and s.time_id = t.time_id
5 and p.prod_id < :pid;
PROD_NAME AMOUNT_SOLD CALENDAR_YEAR
--------- ----------- -------------
...
9 rows selected.
SQL> select p.prod_name, s.amount_sold, t.calendar_year
2 from sales s, products p, times t
3 where s.prod_id = p.prod_id
4 and s.time_id = t.time_id
5 and p.prod_id < :pid;
PROD_NAME AMOUNT_SOLD CALENDAR_YEAR
--------- ----------- -------------
...
9 rows selected.
SQL> alter session set optimizer_capture_sql_plan_baselines = false;
Session altered
Automatic plan capture will not occur for a statement if a stored outline exists for it and is enabled and the parameter use_stored_outlines is TRUE. In this case, turn on incremental capture of plans into an STS using the function capture_cursor_cache_sqlset() in the DBMS_SQLTUNE package. After you have collected the plans for your workload into the STS, manually create SQL plan baselines using the method described earlier. Then, disable the stored outlines or set use_stored_outlines to FALSE. From now on, SPM will manage your workload and stored outlines will not be used for those statements.
In this article, we have seen how to create SQL plan baselines. In the next, we will describe the SPM aware optimizer and how it uses SQL plan baselines.