- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章利用AOP实现SqlSugar自动事务由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
本文实例为大家分享了如何利用AOP实现SqlSugar自动事务,供大家参考,具体内容如下 。
先看一下效果,带接口层的三层架构
BL层:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
public
class
StudentBL : IStudentService
{
private
ILogger mLogger;
private
readonly
IStudentDA mStudentDa;
private
readonly
IValueService mValueService;
public
StudentService(IStudentDA studentDa,IValueService valueService)
{
mLogger = LogManager.GetCurrentClassLogger();
mStudentDa = studentDa;
mValueService = valueService;
}
[TransactionCallHandler]
public
IList<Student> GetStudentList(Hashtable paramsHash)
{
var list = mStudentDa.GetStudents(paramsHash);
var value = mValueService.FindAll();
return
list;
}
}
|
假设GetStudentList方法里的mStudentDa.GetStudents和mValueService.FindAll不是查询操作,而是更新操作,当一个失败另一个需要回滚,就需要在同一个事务里,当一个出现异常就要回滚事务.
特性TransactionCallHandler就表明当前方法需要开启事务,并且当出现异常的时候回滚事务,方法执行完后提交事务.
DA层:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
public
class
StudentDA : IStudentDA
{
private
SqlSugarClient db;
public
StudentDA()
{
db = SugarManager.GetInstance().SqlSugarClient;
}
public
IList<Student> GetStudents(Hashtable paramsHash)
{
return
db.Queryable<Student>().AS(
"T_Student"
).With(SqlWith.NoLock).ToList();
}
}
|
对SqlSugar做一下包装 。
。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
|
public
class
SugarManager
{
private
static
ConcurrentDictionary<
string
,SqlClient> _cache =
new
ConcurrentDictionary<
string
, SqlClient>();
private
static
ThreadLocal<
string
> _threadLocal;
private
static
readonly
string
_connStr =
@"Data Source=localhost;port=3306;Initial Catalog=thy;user id=root;password=xxxxxx;Charset=utf8"
;
static
SugarManager()
{
_threadLocal =
new
ThreadLocal<
string
>();
}
private
static
SqlSugarClient CreatInstance()
{
SqlSugarClient client =
new
SqlSugarClient(
new
ConnectionConfig()
{
ConnectionString = _connStr,
//必填
DbType = DbType.MySql,
//必填
IsAutoCloseConnection =
true
,
//默认false
InitKeyType = InitKeyType.SystemTable
});
var key=Guid.NewGuid().ToString().Replace(
"-"
,
""
);
if
(!_cache.ContainsKey(key))
{
_cache.TryAdd(key,
new
SqlClient(client));
_threadLocal.Value = key;
return
client;
}
throw
new
Exception(
"创建SqlSugarClient失败"
);
}
public
static
SqlClient GetInstance()
{
var id= _threadLocal.Value;
if
(
string
.IsNullOrEmpty(id)||!_cache.ContainsKey(id))
return
new
SqlClient(CreatInstance());
return
_cache[id];
}
public
static
void
Release()
{
try
{
var id = GetId();
if
(!_cache.ContainsKey(id))
return
;
Remove(id);
}
catch
(Exception e)
{
throw
e;
}
}
private
static
bool
Remove(
string
id)
{
if
(!_cache.ContainsKey(id))
return
false
;
SqlClient client;
int
index = 0;
bool
result =
false
;
while
(!(result = _cache.TryRemove(id,
out
client)))
{
index++;
Thread.Sleep(20);
if
(index > 3)
break
;
}
return
result;
}
private
static
string
GetId()
{
var id = _threadLocal.Value;
if
(
string
.IsNullOrEmpty(id))
{
throw
new
Exception(
"内部错误: SqlSugarClient已丢失."
);
}
return
id;
}
public
static
void
BeginTran()
{
var instance=GetInstance();
//开启事务
if
(!instance.IsBeginTran)
{
instance.SqlSugarClient.Ado.BeginTran();
instance.IsBeginTran =
true
;
}
}
public
static
void
CommitTran()
{
var id = GetId();
if
(!_cache.ContainsKey(id))
throw
new
Exception(
"内部错误: SqlSugarClient已丢失."
);
if
(_cache[id].TranCount == 0)
{
_cache[id].SqlSugarClient.Ado.CommitTran();
_cache[id].IsBeginTran =
false
;
}
}
public
static
void
RollbackTran()
{
var id = GetId();
if
(!_cache.ContainsKey(id))
throw
new
Exception(
"内部错误: SqlSugarClient已丢失."
);
_cache[id].SqlSugarClient.Ado.RollbackTran();
_cache[id].IsBeginTran =
false
;
_cache[id].TranCount = 0;
}
public
static
void
TranCountAddOne()
{
var id = GetId();
if
(!_cache.ContainsKey(id))
throw
new
Exception(
"内部错误: SqlSugarClient已丢失."
);
_cache[id].TranCount++;
}
public
static
void
TranCountMunisOne()
{
var id = GetId();
if
(!_cache.ContainsKey(id))
throw
new
Exception(
"内部错误: SqlSugarClient已丢失."
);
_cache[id].TranCount--;
}
}
|
_cache保存SqlSugar实例,_threadLocal确保同一线程下取出的是同一个SqlSugar实例.
不知道SqlSugar判断当前实例是否已经开启事务,所以又将SqlSugar包了一层.
1
2
3
4
5
6
7
8
9
10
11
|
public
class
SqlClient
{
public
SqlSugarClient SqlSugarClient;
public
bool
IsBeginTran =
false
;
public
int
TranCount = 0;
public
SqlClient(SqlSugarClient sqlSugarClient)
{
this
.SqlSugarClient = sqlSugarClient;
}
}
|
IsBeginTran标识当前SqlSugar实例是否已经开启事务,TranCount是一个避免事务嵌套的计数器.
一开始的例子 。
1
2
3
4
5
6
7
|
[TransactionCallHandler]
public
IList<Student> GetStudentList(Hashtable paramsHash)
{
var list = mStudentDa.GetStudents(paramsHash);
var value = mValueService.FindAll();
return
list;
}
|
TransactionCallHandler表明该方法要开启事务,但是如果mValueService.FindAll也标识了TransactionCallHandler,又要开启一次事务?所以用TranCount做一个计数.
使用Castle.DynamicProxy 。
要实现标识了TransactionCallHandler的方法实现自动事务,使用Castle.DynamicProxy实现BL类的代理 。
Castle.DynamicProxy一般操作 。
。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
|
public
class
MyClass : IMyClass
{
public
void
MyMethod()
{
Console.WriteLine(
"My Mehod"
);
}
}
public
class
TestIntercept : IInterceptor
{
public
void
Intercept(IInvocation invocation)
{
Console.WriteLine(
"before"
);
invocation.Proceed();
Console.WriteLine(
"after"
);
}
}
var proxyGenerate =
new
ProxyGenerator();
TestIntercept t=
new
TestIntercept();
var pg = proxyGenerate.CreateClassProxy<MyClass>(t);
pg.MyMethod();
//输出是
//before
//My Mehod
//after
|
before就是要开启事务的地方,after就是提交事务的地方 。
最后实现 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
|
public
class
TransactionInterceptor : IInterceptor
{
private
readonly
ILogger logger;
public
TransactionInterceptor()
{
logger = LogManager.GetCurrentClassLogger();
}
public
void
Intercept(IInvocation invocation)
{
MethodInfo methodInfo = invocation.MethodInvocationTarget;
if
(methodInfo ==
null
)
{
methodInfo = invocation.Method;
}
TransactionCallHandlerAttribute transaction =
methodInfo.GetCustomAttributes<TransactionCallHandlerAttribute>(
true
).FirstOrDefault();
if
(transaction !=
null
)
{
SugarManager.BeginTran();
try
{
SugarManager.TranCountAddOne();
invocation.Proceed();
SugarManager.TranCountMunisOne();
SugarManager.CommitTran();
}
catch
(Exception e)
{
SugarManager.RollbackTran();
logger.Error(e);
throw
e;
}
}
else
{
invocation.Proceed();
}
}
}
[AttributeUsage(AttributeTargets.Method, Inherited =
true
)]
public
class
TransactionCallHandlerAttribute : Attribute
{
public
TransactionCallHandlerAttribute()
{
}
}
|
Autofac与Castle.DynamicProxy结合使用 。
创建代理的时候一个BL类就要一次操作 。
1
|
proxyGenerate.CreateClassProxy<MyClass>(t);
|
而且项目里BL类的实例化是交给IOC容器控制的,我用的是Autofac。当然Autofac和Castle.DynamicProxy是可以结合使用的 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
using
System.Reflection;
using
Autofac;
using
Autofac.Extras.DynamicProxy;
using
Module = Autofac.Module;
public
class
BusinessModule : Module
{
protected
override
void
Load(ContainerBuilder builder)
{
var business = Assembly.Load(
"FTY.Business"
);
builder.RegisterAssemblyTypes(business)
.AsImplementedInterfaces().InterceptedBy(
typeof
(TransactionInterceptor)).EnableInterfaceInterceptors();
builder.RegisterType<TransactionInterceptor>();
}
}
|
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我.
原文链接:http://www.cnblogs.com/jaycewu/archive/2017/10/25/7733114.html 。
最后此篇关于利用AOP实现SqlSugar自动事务的文章就讲到这里了,如果你想了解更多关于利用AOP实现SqlSugar自动事务的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我在同一类中的方法之间进行方法调用并应用事务建议时遇到问题。 Spring Framework .NET 文档声明它支持基于组合和继承的代理,并且您可以强制 Spring 创建要实例化的基于继承的代理
我理解这些原则,但我很难看到实际应用程序在少数几个之外。请赐教;) 最佳答案 问任何支持人员:日志记录不是 AOP 的一个好的应用程序。他们不在乎应用程序内部调用了什么方法。他们关心应用程序正在执行的
我知道以前有人问过这个问题,但这是一年半前的事了,尽管我认为现在可能是重新提问的时候了。我也认识到它可能被视为主观的,但我想有一些客观的原因支持/反对 AOP。 我会对 感兴趣谁在使用 AOP 在软件
我想这个问题以前有人问过,但我无法立即找到相关的 SO 问题或其他地方的相关文章。 令我震惊的是,AOP 中的某些术语相当奇怪。看来我不是唯一一个-这个article ,例如,指出“不幸的是,AOP
面向切面编程可能的和严重的缺点是什么? 例如:新手的神秘调试(可读性影响) 最佳答案 工具链支持不佳 - 调试器、分析器等可能不了解 AOP,因此可能会在代码上工作,就好像所有方面都已被过程代码替换
这两种AOP框架的优缺点是什么?我使用 Unity 作为我的 aop 框架,但我猜想编译时 aop 框架(例如 postsharp)可能比运行时 aop 框架具有更好的性能?看起来运行时 aop 框架
我现在正在学习 spring aop,我不知道将上下文参数传递给建议。 请注意,我指的是 context 参数,而不是 normal 参数。 传递普通参数很简单,例如: a join point: p
来自类路径资源 [ApplicationContextAOP.xml] 的 XML 文档中的第 13 行无效;嵌套异常是 org.xml.sax.SAXParseException: cvc-comp
我使用 spring boot 2 和 spring security。 使用 aop,我搜索以获取调用该方法的用户。 @Aspect @Component public class LogAspec
我最近一直在一个非常简单的应用程序上尝试 Spring 的 AOP 功能,并且我坚持在适当的时间运行该方法,这意味着该部分中定义的方法应该在 中定义的方法之后运行 在我的代码中,这两个方法都在主方法中
我试图在网上找到如何通过 Ninject 使用 AOP 的例子。有人可以确认 AOP 在 Ninject 2 中是否可用而不使用外部库(即 CaSTLe Windsor?)。 如果可以的话,您能否发布
Aop配置已经在我的项目中完成了。为此添加了以下配置。问题是当下面的代码没有注释时,不会调用 formService 中的方法。因此我得到空指针异常。知道问题出在哪里吗?我附上了下面的代码.. AOP
我是 AOP 的新手。我遇到了这样的问题。 package org.suman.Aspect; import org.aspectj.lang.annotation.Aspect; import or
在我们的企业应用程序中,我们希望将日志记录、度量等横切关注点作为方面。我们已经准备好了 aspectj 建议(来自我们现有的 java 应用程序),但我没有找到将 aspectj 与 Grails 集
我正在向外部系统编写 Web 服务。 我的服务包装类有许多方法可以调用Web服务的所有soap接口(interface)。该调用可能会引发异常,然后该异常会自动触发重新连接到 Web 服务。 为了处理
已结束。此问题不符合 Stack Overflow guidelines .它目前不接受答案。 我们不允许提出有关书籍、工具、软件库等方面的建议的问题。您可以编辑问题,以便用事实和引用来回答它。 关闭
我是 spring 框架的新手,正在尝试一些示例来理解 AOP,这是我到目前为止所做的,但它不起作用。 问题是我一添加 对于 spring.xml,我的构建失败说无法创建具有空指针异常的 bean。但
下面是我要创建的方面。我想将两个切入点表达式合并为一个。我已经看到这可以使用带注释的切入点来完成,但是 xml 中的相同语法失败了。谁能帮帮我? 提前致谢 最佳答案
我对 Spring 事务管理感到困惑。在我的应用程序中,我在服务类中使用 @Transactional 实现了事务管理。我配置的 spring.xml 如下:
我知道围绕 Controller 方法编写 AOP 建议的标准方法,并且如果在 Controller 方法中声明,您可以访问 HttpServletRequest arg。 但我的情况是我有一个翻译服
我是一名优秀的程序员,十分优秀!