化零为整WCF(14) - 事务(Transaction)

[索引页]
[源码下载] 


化零为整WCF(14) - 事务(Transaction)


作者:webabcd


介绍
WCF(Windows Communication Foundation) - 事务(Transaction):
    ·对契约方法使用TransactionFlowAttribute声明(设置TransactionFlowOption参数),以指定服务操作的事务流策略
    ·对服务方法是用OperationBehaviorAttribute声明(设置TransactionScopeRequired参数),以指定方法是否在事务范围(TransactionScope)内执行
    ·配置host和client的binding节点的transactionFlow属性,以指定绑定是否支持流事务


示例
1、服务
Hello.cs
InBlock.gifusing System; 
InBlock.gifusing System.Collections.Generic; 
InBlock.gifusing System.Linq; 
InBlock.gifusing System.Text; 
InBlock.gif 
InBlock.gifusing System.ServiceModel; 
InBlock.gif 
InBlock.gifnamespace WCF.ServiceLib.Transaction 
InBlock.gif
InBlock.gif        /// <summary> 
InBlock.gif        /// IHello接口 
InBlock.gif        /// </summary> 
InBlock.gif        [ServiceContract] 
InBlock.gif        public interface IHello 
InBlock.gif        { 
InBlock.gif                /// <summary> 
InBlock.gif                /// 打招呼方法 
InBlock.gif                /// </summary> 
InBlock.gif                /// <param name="name">人名</param> 
InBlock.gif                /// <remarks> 
InBlock.gif                /// TransactionFlow - 指定服务操作是否愿意接受来自客户端的传入事务 
InBlock.gif                /// NotAllowed - 禁止事务。默认值 
InBlock.gif                /// Allowed - 允许事务 
InBlock.gif                /// Mandatory - 强制事务 
InBlock.gif                /// </remarks> 

InBlock.gif                /// <returns></returns> 
InBlock.gif                [OperationContract] 
InBlock.gif                [TransactionFlow(TransactionFlowOption.Mandatory)] 
InBlock.gif                void WriteHello(string name); 
InBlock.gif        } 
InBlock.gif 
InBlock.gif        /// <summary> 
InBlock.gif        /// Hello类 
InBlock.gif        /// </summary> 
InBlock.gif        public class Hello : IHello 
InBlock.gif        { 
InBlock.gif                /// <summary> 
InBlock.gif                /// 打招呼方法 
InBlock.gif                /// </summary> 
InBlock.gif                /// <param name="name">人名</param> 
InBlock.gif                /// <remarks> 
InBlock.gif                /// OperationBehavior - 指定服务方法的本地执行行为 
InBlock.gif                /// 1、TransactionScopeRequired - 如果方法需要事务范围才能执行,则为 true;否则为 false。默认值为false 
InBlock.gif                /// 将 TransactionScopeRequired 设置为 true,可以要求操作在事务范围内执行。如果流事务可用,则操作会在该事务内执行。如果流事务不可用,则会创建一个新事务并使用它来执行操作 
InBlock.gif                /// 2、TransactionAutoComplete - 默认值为 true 
InBlock.gif                /// true - 当方法完成执行时,将把该事务标志为完成(自动提交事务) 
InBlock.gif                /// false - 需要调用OperationContext.Current.SetTransactionComplete()方法来手工配置该事务的正确完成;否则,该事务将被标志为失败(手动提交事务) 
InBlock.gif                /// </remarks> 

InBlock.gif                /// <returns></returns> 
InBlock.gif                [OperationBehavior(TransactionScopeRequired = true, TransactionAutoComplete = true)] 
InBlock.gif                public void WriteHello(string name) 
InBlock.gif                { 
InBlock.gif                        DBDataContext ctx = new DBDataContext(); 
InBlock.gif 
InBlock.gif                        ctx.Items.InsertOnSubmit( 
InBlock.gif                                new Item 
InBlock.gif                                { 
InBlock.gif                                        Title = string.Format("Hello: {0}, TransactionId: {1}", name, System.Transactions.Transaction.Current.TransactionInformation.LocalIdentifier), 
InBlock.gif                                        CreatedTime = DateTime.Now 
InBlock.gif                                }); 
InBlock.gif 
InBlock.gif                        ctx.SubmitChanges(); 
InBlock.gif                } 
InBlock.gif        } 
InBlock.gif}
Hi.cs
InBlock.gifusing System; 
InBlock.gifusing System.Collections.Generic; 
InBlock.gifusing System.Linq; 
InBlock.gifusing System.Text; 
InBlock.gif 
InBlock.gifusing System.ServiceModel; 
InBlock.gif 
InBlock.gifnamespace WCF.ServiceLib.Transaction 
InBlock.gif
InBlock.gif        /// <summary> 
InBlock.gif        /// IHi接口 
InBlock.gif        /// </summary> 
InBlock.gif        [ServiceContract] 
InBlock.gif        public interface IHi 
InBlock.gif        { 
InBlock.gif                /// <summary> 
InBlock.gif                /// 打招呼方法 
InBlock.gif                /// </summary> 
InBlock.gif                /// <param name="name">人名</param> 
InBlock.gif                /// <returns></returns> 
InBlock.gif                [OperationContract] 
InBlock.gif                [TransactionFlow(TransactionFlowOption.Mandatory)] 
InBlock.gif                void WriteHi(string name); 
InBlock.gif        } 
InBlock.gif 
InBlock.gif        /// <summary> 
InBlock.gif        /// Hi类 
InBlock.gif        /// </summary> 
InBlock.gif        public class Hi : IHi 
InBlock.gif        { 
InBlock.gif                /// <summary> 
InBlock.gif                /// 打招呼方法 
InBlock.gif                /// </summary> 
InBlock.gif                /// <param name="name">人名</param> 
InBlock.gif                /// <returns></returns> 
InBlock.gif                [OperationBehavior(TransactionScopeRequired = true, TransactionAutoComplete = true)] 
InBlock.gif                public void WriteHi(string name) 
InBlock.gif                { 
InBlock.gif                        if (DateTime.Now.Second % 2 == 0) 
InBlock.gif                                throw new System.Exception("为测试事务而抛出的异常"); 
InBlock.gif 
InBlock.gif                        DBDataContext ctx = new DBDataContext(); 
InBlock.gif 
InBlock.gif                        ctx.Items.InsertOnSubmit( 
InBlock.gif                                new Item 
InBlock.gif                                { 
InBlock.gif                                        Title = string.Format("Hi: {0}, TransactionId: {1}", name, System.Transactions.Transaction.Current.TransactionInformation.LocalIdentifier), 
InBlock.gif                                        CreatedTime = DateTime.Now 
InBlock.gif                                }); 
InBlock.gif 
InBlock.gif                        ctx.SubmitChanges(); 
InBlock.gif                } 
InBlock.gif        } 
InBlock.gif}
Result.cs
InBlock.gifusing System; 
InBlock.gifusing System.Collections.Generic; 
InBlock.gifusing System.Linq; 
InBlock.gifusing System.Text; 
InBlock.gif 
InBlock.gifusing System.ServiceModel; 
InBlock.gif 
InBlock.gifnamespace WCF.ServiceLib.Transaction 
InBlock.gif
InBlock.gif        /// <summary> 
InBlock.gif        /// 结果接口 
InBlock.gif        /// </summary> 
InBlock.gif        [ServiceContract] 
InBlock.gif        public interface IResult 
InBlock.gif        { 
InBlock.gif                [OperationContract] 
InBlock.gif                List<Item> GetResult(); 
InBlock.gif        } 
InBlock.gif 
InBlock.gif        /// <summary> 
InBlock.gif        /// 结果类 
InBlock.gif        /// </summary> 
InBlock.gif        public class Result : IResult 
InBlock.gif        { 
InBlock.gif                /// <summary> 
InBlock.gif                /// 返回数据库结果 
InBlock.gif                /// </summary> 
InBlock.gif                /// <returns></returns> 
InBlock.gif                public List<Item> GetResult() 
InBlock.gif                { 
InBlock.gif                        DBDataContext ctx = new DBDataContext(); 
InBlock.gif 
InBlock.gif                        var result = from l in ctx.Items 
InBlock.gif                                                 orderby l.CreatedTime descending 
InBlock.gif                                                 select l; 
InBlock.gif 
InBlock.gif                        return result.ToList(); 
InBlock.gif                } 
InBlock.gif        } 
InBlock.gif}

2、宿主
Hello.svc
<%@ ServiceHost Language="C#" Debug="true" Service="WCF.ServiceLib.Transaction.Hello" %>
Hi.svc
<%@ ServiceHost Language="C#" Debug="true" Service="WCF.ServiceLib.Transaction.Hi" %>
Result.svc
<%@ ServiceHost Language="C#" Debug="true" Service="WCF.ServiceLib.Transaction.Result" %>
Web.config
<?xml version="1.0"?> 
<configuration> 
        <system.serviceModel> 
                <behaviors> 
                        <serviceBehaviors> 
                                <behavior name="TransactionBehavior"> 
                                        <!--httpGetEnabled - 指示是否发布服务元数据以便使用 HTTP/GET 请求进行检索,如果发布 WSDL,则为 true,否则为 false,默认值为 false--> 
                                        <serviceMetadata httpGetEnabled="true" /> 
                                        <serviceDebug includeExceptionDetailInFaults="true"/> 
                                </behavior> 
                        </serviceBehaviors> 
                </behaviors> 
                <services> 
                        <!--name - 提供服务的类名--> 
                        <!--behaviorConfiguration - 指定相关的行为配置--> 
                        <service name="WCF.ServiceLib.Transaction.Hello" behaviorConfiguration="TransactionBehavior"> 
                                <!--address - 服务地址--> 
                                <!--binding - 通信方式--> 
                                <!--contract - 服务契约--> 
                                <!--bindingConfiguration - 指定相关的绑定配置--> 
                                <endpoint address="" binding="wsHttpBinding" contract="WCF.ServiceLib.Transaction.IHello" bindingConfiguration="TransactionConfiguration" /> 
                        </service> 
                        <service name="WCF.ServiceLib.Transaction.Hi" behaviorConfiguration="TransactionBehavior"> 
                                <endpoint address="" binding="wsHttpBinding" contract="WCF.ServiceLib.Transaction.IHi" bindingConfiguration="TransactionConfiguration" /> 
                        </service> 
                        <service name="WCF.ServiceLib.Transaction.Result" behaviorConfiguration="TransactionBehavior"> 
                                <endpoint address="" binding="basicHttpBinding" contract="WCF.ServiceLib.Transaction.IResult" /> 
                        </service> 
                </services> 
                <bindings> 
                        <wsHttpBinding> 
                                <!--transactionFlow - 指定该绑定是否应支持流事务--> 
                                <binding name="TransactionConfiguration" transactionFlow="true" /> 
                        </wsHttpBinding> 
                </bindings> 
        </system.serviceModel> 
</configuration>
3、客户端
Sample.aspx
<%@ Page Language="C#" MasterPageFile="~/Site.master" AutoEventWireup="true" CodeFile="Sample.aspx.cs" 
        Inherits="Transaction_Sample" Title="事务(Transaction)" %> 

<asp:Content ID="Content1" ContentPlaceHolderID="head" runat="Server"> 
</asp:Content> 
<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolder1" runat="Server"> 
        <p> 
                <asp:Label ID="lblErr" runat="server" ForeColor="Red" /> 
        </p> 
        <p> 
                <asp:Button ID="btnSubmit" runat="server" Text="事务测试" OnClick="btnSubmit_Click" /> 
                <br /> 
                <br /> 
                <asp:GridView ID="GridView1" runat="server"> 
                </asp:GridView> 
        </p> 
        <p> 
                2PC(Two Phase Commitment Protocol)两阶段提交协议(WCF的事务的实现基于此协议) 
                <br /> 
                实现分布式事务的关键就是两阶段提交协议。在此协议中,一个或多个资源管理器的活动均由一个称为事务协调器的单独软件组件来控制。此协议中的五个步骤如下: 
                <br /> 
                1、应用程序调用事务协调器中的提交方法。 
                <br /> 
                2、事务协调器将联络事务中涉及的每个资源管理器,并通知它们准备提交事务(这是第一阶段的开始)。 
                <br /> 
                3、为 了以肯定的方式响应准备阶段,资源管理器必须将自己置于以下状态:确保能在被要求提交事务时提交事务,或在被要求回滚事务时回滚事务。大多数资源管理器会将包含其计划更改的日记文件(或等效文件)写入持久存储区中。如果资源管理器无法准备事务,它会以一个否定响应来回应事务协调器。 
                <br /> 
                4、事务协调器收集来自资源管理器的所有响应。 
                <br /> 
                5、在 第二阶段,事务协调器将事务的结果通知给每个资源管理器。如果任一资源管理器做出否定响应,则事务协调器会将一个回滚命令发送给事务中涉及的所有资源管理 器。如果资源管理器都做出肯定响应,则事务协调器会指示所有的资源管理器提交事务。一旦通知资源管理器提交,此后的事务就不能失败了。通过以肯定的方式响应第一阶段,每个资源管理器均已确保,如果以后通知它提交事务,则事务不会失败。 
        </p> 
</asp:Content>
Sample.aspx.cs
InBlock.gifusing System; 
InBlock.gifusing System.Collections; 
InBlock.gifusing System.Configuration; 
InBlock.gifusing System.Data; 
InBlock.gifusing System.Linq; 
InBlock.gifusing System.Web; 
InBlock.gifusing System.Web.Security; 
InBlock.gifusing System.Web.UI; 
InBlock.gifusing System.Web.UI.HtmlControls; 
InBlock.gifusing System.Web.UI.WebControls; 
InBlock.gifusing System.Web.UI.WebControls.WebParts; 
InBlock.gifusing System.Xml.Linq; 
InBlock.gif 
InBlock.gifusing System.Threading; 
InBlock.gif 
InBlock.gifpublic partial class Transaction_Sample : System.Web.UI.Page 
InBlock.gif
InBlock.gif        protected void Page_Load(object sender, EventArgs e) 
InBlock.gif        { 
InBlock.gif 
InBlock.gif        } 
InBlock.gif 
InBlock.gif        protected void btnSubmit_Click(object sender, EventArgs e) 
InBlock.gif        { 
InBlock.gif                var proxyHello = new TransactionSvc.Hello.HelloClient(); 
InBlock.gif                var proxyHi = new TransactionSvc.Hi.HiClient(); 
InBlock.gif                var proxyResult = new TransactionSvc.Result.ResultClient(); 
InBlock.gif 
InBlock.gif                System.Transactions.TransactionOptions to = new System.Transactions.TransactionOptions(); 
InBlock.gif                // 设置事务的超时时间 
InBlock.gif                to.Timeout = new TimeSpan(0, 0, 30); 
InBlock.gif                // 设置事务的隔离级别 
InBlock.gif                to.IsolationLevel = System.Transactions.IsolationLevel.Serializable; 
InBlock.gif 
InBlock.gif                using (var ts = new System.Transactions.TransactionScope()) 
InBlock.gif                { 
InBlock.gif                        try 
InBlock.gif                        { 
InBlock.gif                                proxyHello.WriteHello("webabcd"); 
InBlock.gif                                proxyHello.Close(); 
InBlock.gif 
InBlock.gif                                proxyHi.WriteHi("webabcd"); 
InBlock.gif                                proxyHi.Close(); 
InBlock.gif 
InBlock.gif                                ts.Complete(); 
InBlock.gif 
InBlock.gif                                lblErr.Text = "OK"
InBlock.gif                        } 
InBlock.gif                        catch (Exception ex) 
InBlock.gif                        { 
InBlock.gif                                lblErr.Text = ex.ToString(); 
InBlock.gif                        } 
InBlock.gif                } 
InBlock.gif 
InBlock.gif                GridView1.DataSource = proxyResult.GetResult(); 
InBlock.gif                GridView1.DataBind(); 
InBlock.gif                proxyHello.Close(); 
InBlock.gif        } 
InBlock.gif}
Web.config
<?xml version="1.0"?> 
<configuration> 
        <system.serviceModel> 
                <client> 
                        <!--address - 服务地址--> 
                        <!--binding - 通信方式--> 
                        <!--contract - 服务契约--> 
                        <endpoint address="http://localhost:3502/ServiceHost/Transaction/Hello.svc" binding="wsHttpBinding" contract="TransactionSvc.Hello.IHello" bindingConfiguration="TransactionBindingConfiguration" /> 
                        <endpoint address="http://localhost:3502/ServiceHost/Transaction/Hi.svc" binding="wsHttpBinding" contract="TransactionSvc.Hi.IHi" bindingConfiguration="TransactionBindingConfiguration" /> 
                        <endpoint address="http://localhost:3502/ServiceHost/Transaction/Result.svc" binding="basicHttpBinding" contract="TransactionSvc.Result.IResult" /> 
                </client> 
                <bindings> 
                        <wsHttpBinding> 
                                <!--transactionFlow - 指定该绑定是否应支持流事务--> 
                                <binding name="TransactionBindingConfiguration" transactionFlow="true" /> 
                        </wsHttpBinding> 
                </bindings> 
        </system.serviceModel> 
</configuration>
运行结果:
单击"btnSubmit"按钮后,可以发现,两个数据库插入操作,要么都执行,要么都不执行


OK
[源码下载]
 

本文转自webabcd 51CTO博客,原文链接:http://blog.51cto.com/webabcd/344160,如需转载请自行联系原作者

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/257050.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

有限元分析笔记01-平面应力和平面应变

https://www.zhihu.com/question/30439292 http://blog.sina.cn/dpool/blog/s/blog_c4c804690102vqqs.html plate stress plate strain

MQTT-SN协议乱翻之实现要点

前言 本篇是MQTT-SN 1.2协议最后一篇翻译了&#xff0c;主要涉及实现要点&#xff0c;很简短。 需要支持QoS 值为 -1 QoS虽默认设置有0,1,2三个值&#xff0c;但还有一种情况其值为-1。来自客户端的PUBLISH消息中若QoS为-1的情况下&#xff0c;此刻客户端不会关心和网关有没有建…

oracle-REDO日志文件分析(insert)

1:记录当前scnselect dbms_flashback.get_system_change_number from dual;GET_SYSTEM_CHANGE_NUMBER------------------------11595722:创建表CREATE TABLE team (team_code VARCHAR2(3),team_name VARCHAR2(30),country_code VARCHAR2(3) );INSERT INTO team VALUES (M…

删除修改bond

参考地址&#xff1a;http://www.111cn.net/sys/linux/79301.htm 四、删除bonding设备 如由于最初配置的bonding设备取名为bond0&#xff0c;而后改名为了bond1&#xff0c;造成了两个bonding设备的存在&#xff0c;现在需删除bond0 。先查看下网络设备&#xff1a; # ls /sys/…

学习笔记(18):Python网络编程并发编程-守护进程

立即学习:https://edu.csdn.net/course/play/24458/296429?utm_sourceblogtoedu 守护进程&#xff08;了解&#xff09; 1.概念&#xff1a;守护进程是主进程在创建子进程的时候&#xff0c;将子进程设置成守护自己的进程&#xff0c;等主进程结束后&#xff0c;不管子进程的…

静态页面之间的转发与json与ajax做到动态数据

我们见过很多使用jsp ,php,asp的动态网页技术的网站了,我们知道如果一个网站内容更新频率极低,而内容量不是十分庞大时,这样的网站(一次开发完成后不会需要较多的维护成本)的完全可以使用全部使用静态页面来做,此时其实反而可以得到更好的效果(更快的响应时间(省掉了服务器各种…

数组的最后一位的下一位为什么是0?

以下是我做的两个实验&#xff0c;加证实了数组的最后一位的后一位是0&#xff0c;只应该是系统自动添加的标志位 1、比如 int a[5] 则a[5]0,这个是什么原因我还没有搞懂 #include<iostream> using namespace std;int main() {int a[5];int *pa;for(int i0;i<5;i){a[i…

iOS开发网络篇—NSURLConnection基本使用

iOS开发网络篇—NSURLConnection基本使用 一、NSURLConnection的常用类 &#xff08;1&#xff09;NSURL&#xff1a;请求地址 &#xff08;2&#xff09;NSURLRequest&#xff1a;封装一个请求&#xff0c;保存发给服务器的全部数据&#xff0c;包括一个NSURL对象&#xff0c;…

如何查看mysql连接相关参数

1.查看当前所有连接的详细资料: mysqladmin -u root -ppassword processlist 这里password为数据库用户root的密码 2.只查看当前连接数(Threads就是连接数.): mysqladmin -u root -ppassword status 这里password为数据库用户root的密码 3.如何知道当前MySQL设置的并发连接数是…

学习笔记(19):Python网络编程并发编程-互斥锁

立即学习:https://edu.csdn.net/course/play/24458/296430?utm_sourceblogtoedu 1.互斥锁&#xff1a; 多进程间的内存是相互隔离的&#xff0c;因此其数据也是相互隔离的&#xff0c;但是所有的进程都共享一个文件操作系统或者说共享文件处理器和打印端。而共享带来的是竞争…

使用HTML5+CSS3制作圆角内发光按钮----示例

<!doctype html> <html> <head> <meta charset"utf-8" /> <title>制作漂亮的圆角按钮<title> <style type"text/css"> .loginBtnDiv { float:right; padding-right:50px; padding-top:10px; } .loginBtn, .Resg…

C++中的sort()函数的原形

1、sor(a,an,compare) {//前两个是参数是待排序的数组首地址和尾地址 //最后一个参数是compare表示的比较类型 //可调用functional函数的less&#xff08;&#xff09;和greater&#xff08;&#xff09;函数比较大小}

鼠标放上超链接显示背景效果

鼠标放上超链接显示背景效果&#xff1a; <html> <head> <style type"text/css"> a.one:link {color: #ff0000} a.one:visited {color: #0000ff} a.one:hover {color: #ffcc00}a.two:link {color: #ff0000} a.two:visited {color: #0000ff} a.two:…

学习笔记(20):Python网络编程并发编程-互斥锁与join的区别

立即学习:https://edu.csdn.net/course/play/24458/296432?utm_sourceblogtoedu 互斥锁与join的异同&#xff1a; 1.同&#xff1a;都是将多进程并发模式变成多进程串行&#xff0c;保证了数据的有序性 2.异&#xff1a; 互斥锁只是对于进程的局部代码实施串行执行变化&#x…

Kali渗透(二)之被动信息收集

被动信息收集 公开渠道可获得的信息 与目标系统不产生直接交互 尽量避免留下一切痕迹 下面是两个关于开源情报&#xff08;OSINT&#xff09;的书籍 美国军方 北大西洋公约组织 信息收集内容 一.IP地址段与域名信息 二.邮件地址 三.文档图片数据 四.公司地址 五.公司组织…

ccBPM典型的树形表单和多表头表单的流程示例

ccBPM典型的树形表单和多表头表单的流程 关键字&#xff1a;树形表单。ccBPM支持树形表单&#xff0c;也可以称之为树结构的多表单或者多表头表单。 应用场景&#xff1a;比如项目招标类流程&#xff0c;在填写项目申请的环节&#xff0c;需要填写公司简介、公司业绩、项目实施…

由strcat函数引发的对char *a和char a[]以及sizeof和strlen

/* http://www.cnblogs.com/kaituorensheng/archive/2012/10/23/2736069.html参考这篇文章* char *a "Hello";Hello存放在常量区&#xff0c;是无法修改的。 * 通过指针只可以访问字符串常量&#xff0c;而不可以去改变它 * * * char a[] "Hello";…

for(;;)函数中判断条件执行顺序

for(i100;i>0&#xff1b;--i) { cout<<"i"<<i<<endl; } 它的执行过程如下&#xff1a; 1) 先求解表达式1。 2) 求解表达式2&#xff0c;若其值为真&#xff08;非0&#xff09;&#xff0c;则执行for语句中指定的内嵌语句&…

学习笔记(21):Python网络编程并发编程-队列的使用

立即学习:https://edu.csdn.net/course/play/24458/296433?utm_sourceblogtoedu 队列和管道 1.存在的意义&#xff1a; 1&#xff09;在前面的笔记中&#xff0c;使用join和互斥锁来保证了硬盘数据共享的数据安全性&#xff0c;但是降低了代码的运行效率&#xff1b; 2&#x…

dbms_xplan之display_cursor函数的使用

文章来源&#xff1a;http://blog.csdn.net/leshami/article/details/6866925 DBMS_XPLAN包中display_cursor函数不同于display函数&#xff0c;display_cursor用于显示SQL语句的真实的执行计划&#xff0c;在大多数情况下&#xff0c;显示真实 的执行计划有助于更好的分析SQL语…