1、报错如下:Invalid Operation the connection is closed,无效操作,连接被关闭。这个错误是并不是每次都报,只有在复杂操作、大事务的情况下才偶然报出来。

stackOverflow上有很多关于这个问题的讨论,例如这个:《System.Data.OracleClient random Invalid Operation the connection is closed》,但较零散,全扫了一遍之后,我仍然有如下疑问:

1)怎么看TransactionScope里的Timeout到底是多少,比如通过TransactionOptions.Timeout = new TimeSpan(0,0,3)设置之后,怎么知道TransactionScope里的超时就是3s?

2)默认MaxTimeout=10m,怎么才能设置任意值?

3)嵌套事务的外层超时是否包括所有内层事务的时间?

4)超时机制是怎么起作用的?

通过Reflector之类的工具查看源码,终于有了初步的理解,不对的地方,请大家指正!

2、怎么看TransactionScope里的Timeout?

用调试器逐级展开TransactionScope实例如下:

scope->committableTransaction->base->internalTransaction->absoluteTimeout //注:这是展开过程,不是代码

这个absoluteTimeout就是实际结果,但往往与你设置的值不匹配,比如:

TimeSpan absoluteTimeout
3s 5
10s 19
1m 117
10m 1171
30m 3515
1h 7031

看起来是经过某个算法转换后的值,这个算法写在TransactionTable.TimeoutTicks()里:

internal long TimeoutTicks(TimeSpan timeout){
if(timeout != TimeSpan.Zero) {
return (timeout.Ticks / 10000L >> ) + this.ticks;
}
return 9223372036854775807L;
}

这就清楚了,再加上上面那张对照表,就可以通过absoluteTimeout清楚知道设置是否生效。

3、默认MaxTimeout=10m,怎么才能设置任意值?

1)先说一下这个10m怎么来的?

TransactionScope.ctr()->TransactionManager.ValidateTimeout()->MaximumTimeout->MachineSettingsSection.MaximumTimeout

这个属性先读取c:\windows\Microsoft.NET\Framework\v4.0.30319\Config\machine.config里的值://注意对应的framework的版本!

<system.transactions>
<machineSettings maxTimeout="00:10:00" />
</system.transactions>

当然,我查看下来,所有版本的machine.config里默认都没有这个节点。那么退一步,由Attribute设置默认值:

[ConfigurationProperty("maxTimeout", DefaultValue = "00:10:00")]
public TimeSpan MaxTimeout

2)再说一下怎么改成任意值?

2.1)修改machine.config,优点:方便,缺点:会影响机器上的所有事务。方法是在全局的machine.config里添加对应节点,注意在自己的app/web.config里添加<machineSettings>节点会报错。

2.2)用反射(神器)!优点:只作用于自己的事务,缺点:用反射、稍麻烦、稍慢。

static void ChangeMaximumTimeout() {
Type t = typeof(TransactionManager);
FieldInfo f = t.GetField("_cachedMaxTimeout", BindingFlags.NonPublic | BindingFlags.Static);
f.SetValue(null, true);
f = t.GetField("_maximumTimeout", BindingFlags.NonPublic | BindingFlags.Static);
f.SetValue(null, new TimeSpan(, , ));
}

这么写是因为最终大家取的都是static类里的static属性:

public static class TransactionManager {
public static TimeSpan MaximumTimeout {
get {
if(!_cachedMaxTimeout) {
_maximumTimeout = MachineSettings.MaxTimeout;
}
return _maximumTimeout; }}}

贴出来,一目了然!

4、嵌套事务的外层超时是否包括所有内层事务的时间?

我理解外层超时包含所有内层事务的执行时间,测试代码如下:

option.Timeout = new TimeSpan(,,);
using(var scope = new TransactionScope(TransactionScopeOption.Required, option)) {
Thread.Sleep(); option1.Timeout = new TimeSpan(,,);
using(var scope1 = new TransactionScope(TransactionScopeOption.Required, option)) {
Thread.Sleep(); // 800以上基本就报超时错误了
}}

5、超时机制是怎么起作用的?

1)对于非根(committableTransaction == null)的事务,会构造一个scopeTimer = new Timer(timeSpan, TimeoutCallback)来触发超时回滚;

2)对于committableTransaction != null的根事务,超时机制尚未找到,有待补充完整。

从报错“无效操作,连接被关闭”探究Transaction的Timeout超时机制的更多相关文章

  1. DB2读取CLOB字段-was报错:操作无效:已关闭 Lob。 ERRORCODE=-4470, SQLSTATE=null

    DB2读取CLOB字段-was报错:操作无效:已关闭 Lob. ERRORCODE=-4470, SQLSTATE=null 解决方法,在WAS中要用的数据源里面配置连个定制属性: progressi ...

  2. Vmware Pro 14报错:无法连接 MKS: 套接字连接尝试次数太多;正在放弃。

    软件环境: 虚拟机软件:VMware Pro 14 母机操作系统:win7 客户机操作系统:CentOS 7     问题详情: 报错:无法连接 MKS: 套接字连接尝试次数太多:正在放弃.     ...

  3. 【已解决】ckfinder_php_3.4.4 IIS 报错 无效请求

    ckfinder_php_3.4.4 IIS 报错 无效请求 (Invalid request) Apache 正常,但是在IIS环境下报错,解决方法 设置 C:\Windows\Temp 目录 给 ...

  4. 【spring data jpa】使用spring data jpa 的删除操作,需要加注解@Modifying @Transactional 否则报错如下: No EntityManager with actual transaction available for current thread - cannot reliably process 'remove' call

    使用spring data jpa 的删除操作,需要加注解@Modifying     @Transactional 否则报错如下: No EntityManager with actual tran ...

  5. C# 保存PictureBox中的图片到数据库,并从数据库读取图片显示到PictrueBox,解决报错 “无效参数”

    下面是两段关键代码: /// <summary> /// 将一张图片转换为字节 /// </summary> /// <param name="img" ...

  6. 编译时.test文件报错无法解决的方法,关闭test编译

    有几次遇到从网上下载到的iOS开源代码编译报错,报错位置为Test Target的源文件,我就挺奇怪我又没做测试为啥会编译Test Target的源文件,之前的暴力解决方法是把Test Target直 ...

  7. 启动mongodb报错,无法连接mongodb

    报错原因如下: MongoDB shell version v3.4.2 connecting to: mongodb://127.0.0.1:27017 --01T12:: W NETWORK [t ...

  8. Python Django 协程报错,进程池、线程池与异步调用、回调机制

    一.问题描述 在Django视图函数中,导入 gevent 模块 import gevent from gevent import monkey; monkey.patch_all() from ge ...

  9. magento后台登陆被锁定 索引报错的解决:General error: 1205 Lock wait timeout

    1. magento在索引的时候用shell,有时候会报错: General error: 1205 Lock wait timeout exceeded 这个时候,是因为行锁的原因,在表中您直接用s ...

随机推荐

  1. you don't have permission to view it 解决

    the file couldn't be opened because you don't have permission to view it   简单设置下面的选项即可,不要谢我啊! change ...

  2. 开启所有PHP错误!无论在任何时候

    开发的时候,会屏蔽所有错误,但是有时候排查太蛋疼, 下面代码粘贴到控制器头部,开所有错误! 不可阻挡   ini_set('display_errors','1'); restore_error_ha ...

  3. JS 判断当前使用浏览器名及版本

    由于我的后台系统的上传图片不支持IE浏览器的,所以我需要判断公司人员在使用后台系统的时候,是否使用的浏览器是IE的. // 获取当前浏览器名 及 版本号 function appInfo(){ var ...

  4. oninput,onpropertychange,onchange的用法和区别【转载】

    1.前言 由于工作需要,需实现一个类似于微博输入框的功能,在用户动态输入文字的时候,修改提示“您还可以输入XX字”.如下图所示: 因此,稍微研究了一下oninput,onpropertychange, ...

  5. 启动一个新的activity并携带数据,返回数据给上一个activity

    一.在启动一个新的activity的时候可以通过Intent携带数据,通过Intent.putExtra()方法通过键值对的形势装入数据.在新启动的activity中通过           getI ...

  6. 自定义view获取宽高

    View在构造函数初始化并未布局处理,此时宽高均为0,待所有控件初始化完毕后,由上级容器对内部各控件进行布局,此时控件才会具有位置与大小属性,可以通过以下方法获取:1.在ondraw()函数中获取,2 ...

  7. html中盒子模型立体结构图

    边框(border),位于盒子的第一层..元素内容(content).内边距(padding),两者同位于第二层..背景图(background-image),位于第三层..背景色(backgroun ...

  8. JavaScript中Cookie的用法

    Javascript中Cookie主要存储于客户端的计算机中,用于存放已访问的站点信息,Cookie最大约为4k.以下实例主要用于页面在刷新时保存数据,具体的用法如下所示: <html> ...

  9. Sublime Text 3使用技巧

    工欲善其事必先利其器,用了很多编辑器,pycharm.Eclipse.notepad++.ultra edit等等,要么是太卡,要么是太简陋 论性感,论轻便,论功能,我选Sublime Text 参考 ...

  10. dedecms 网站优化技巧

    1.把列表文件中的<title>***</title>改为栏目名称-seo标题-网站名称即<title>{dede:field.title/}-{dede:fiel ...