走进Linq-How do I(4)拾遗补零篇第一节
2008-08-13 15:23 横刀天笑 阅读(3121) 评论(7) 编辑 收藏 举报
强类型DataContext
在Linq to SQL的第一篇的时候就说道DataContext是一个入口点,我们使用Linq to SQL做的一些操作几乎都是施加在这个类上的。在使用VS的设计器生成的代码里,我们会看到一个从DataContext继承的局部类,大家都习惯的将这个类称之为强类型的DataContext,她对DataContext做了进一步的封装。
今天我们先就对DataContext一些没有介绍过的地方详细讨论一下。
首先我们先手写一个强类型的DataContext:
在这个类里出现了四个前面没有看到的特性:Database,Function,Parameter,ResultType至于Database就不用说了,就是做数据库映射的。下面对其它三个做一些讨论:
Function、Parameter和ResultType
Linq to SQL不仅仅能做字段与属性之间的映射,还可以将存储过程或用户自定义方法与.net里的方法之间做映射,功能是不是很强大?这个映射就是通过Function和Parameter共同完成的。
Function有两个属性IsComposable和Name,Name就是用来指定数据库中存储过程或者用户自定义方法的名字,当IsComposable为true的时候,则表明该方法对应着一个用户自定义方法,否则对应一个存储过程,默认是false。Function特性只能放在方法上面。
Parameter就是用来映射存储过程或方法接受的参数。
还是用例子来说明:
假设有这样的一个存储过程,通过blogid找出其所有的随笔
(
@blogid int
)
AS
SELECT postid,blogid,title,body,createdate FROM posts WHERE blogid = @blogid
RETURN
我要在.net里写一个方法来对应这个存储过程
public ISingleResult<Post> GetPostsByBlogId(
[Parameter(Name="blogid",DbType="int")]
int blogid)
{
IExecuteResult result = this.ExecuteMethodCall(this, ((MethodInfo)(MethodInfo.GetCurrentMethod())), blogid);
return (ISingleResult<Post>)result.ReturnValue;
}
现在你可以以调用.net方法的形式直接调用这个方法来调用存储过程
这样就将存储过程封装成.NET的方法了,调用看看:
使用存储过程的时候,我们往往使用一个条件参数,根据条件参数的不同返回不同的结果集:
(
@kind int,
)
AS
if @kind = 1
SELECT * FROM blogs
ELSE
SELECT * FROM posts
RETURN
在.NET里使用这样的方法映射:
[ResultType(typeof(Blog))]
[ResultType(typeof(Post))]
public IMultipleResults GetBlogsOrPosts(
[Parameter(Name = "kind", DbType = "int")]
int kind)
{
IExecuteResult result = this.ExecuteMethodCall(this, ((MethodInfo)MethodInfo.GetCurrentMethod()), kind);
return (IMultipleResults)result.ReturnValue;
}
返回一个ImultipleResults对象,该对象有一个GetResult方法:
foreach (var b in results.GetResult<Blog>())
{
Console.WriteLine(b.Name;
}
AS
SELECT * FROM blogs
SELECT * FROM posts
RETURN
使用方法还是和上面的一样。
在DataContext还有几个我们比较感兴趣的方法:
ExecuteCommand,ExecuteQuery<TResult>,Translate
这几个方法都是为了和传统的ADO.NET集成的。
ExecuteCommand可以做insert,update,delete操作,第一个参数接受一个格式化的sql语句,第二个参数是参数数组:
使用起来很简单:
dbCtx.ExecuteCommand("insert into posts(blogid,title,body) values({0},{1},{2})", "2", "走进Linq-How do I(4)", "废话一篇");
ExecuteQuery<TResult>方法呢?顾名思义,就是执行查询的,也有两个参数,第一个接受查询的SQL语句,第二个接受参数,返回一个IEnumerable<T>对象:
IEnumerable<Post> posts = dbCtx.ExecuteQuery<Post>("select * from posts where postid = {0}", "1");
foreach (var post in posts)
Console.WriteLine(post.Title);
Translate方法是将以前的DataReader转换为现在的Linq to SQL方式,这样有什么好处呢:
1. 可以利用现在的映射,不再需要这样的语句post.Title = dr[1].ToString();
2. 可以使用查询表达式的语法
对于插入,更新,删除这些经常使用的操作,为了提高性能,或者做一些自定义的操作,你还可以覆盖微软默认为你提供的Insert,Update等方法:
1. 写一个存储过程
CREATE PROCEDURE dbo.CreatePost
(
@blogid int,
@title nvarchar(50),
@body nvarchar(500)
)
AS
insert into posts(blogid,title,body) values(@blogid,@title,@body)
RETURN
当然,这个存储过程非常简单,不值得这么去做,你就假设存在一个复杂的存储过程
2. 把Posts表拖到设计器上,上面的存储过程拖到方法设计器上
3. 右键点击设计器上的表,在右键菜单里会有一项:配置行为(Config Behavior):
这样就可以用你自定义的存储过程做插入操作了。
Linq to SQL中的事务
事务这个东西在并发的时候特别重要,Linq to SQL本身就支持事务,不过是乐观锁。我们也可以显式的启动Linq to SQL的事务。
dbCtx.Transaction = dbCtx.Connection.BeginTransaction();
try
{
dbCtx.ExecuteCommand("insert into posts(blogid,title,body) values({0},{1},{2})", "2", "走进Linq-How do I(4)", "废话一篇");
dbCtx.ExecuteCommand("insert into posts(blogid,title,body) values({0},{1},{2})", "3", "走进Linq-How do I(5)", "废话两篇");
dbCtx.Transaction.Commit();
}
catch
{
dbCtx.Transaction.Rollback();
}
DbConnection的BeginTransaction方法还可以接受一个枚举参数IsolationLevel,来指定事务隔离的级别。
对于那些多个DataContext之间的事务,我们可以使用.NET 2.0引入的TransactionScope
下一篇谈谈RowVersion的问题