zoukankan      html  css  js  c++  java
  • VS2008的Linq更新数据就那么费劲

    VS2008的Linq更新数据就那么费劲

      用.Net两年了,也积累了一些知识和经验,觉得应该做出点自己的东西,而并不只是给别人打工。
    3D*j)@,?4T
    #`1l%Q)]4j,S,P&ktech.techweb.com.cn  所以决定利用最新发布的VS2008(Orcas) Beta2也加入到WEB 2.0的大潮中来,一来是学以所用,二来在实践中掌握最新的技术。
    .]9g$r$p+\5Y0x'G7u'U0_!\2N.i*|"[5N
      现在流行在开发阶段给项目起个Code Name,我也来凑凑热闹,就叫Pluto,以纪念不久前被剥夺九大行星资格的我们天蝎座的守护星——冥王星'f"f0c1[ ]
    TechWeb-技术社区:K-k.i4M$r6`9h
      平时有自己的工作,只能利用不多的业余时间开发,所以预计(争取)在VS2008正式发布之际,Pluto也能开发完成。
    +U5O5F/`+B3@9i0N7W程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛!C3R0{*F \0@9b+E;[
      在这里,我会记录下开发Pluto中的一些事情。
    1V4e+Y/^!c._4S#}:~程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛3Z-O%T(~)z"D,H3A
      WEB 2.0的网站少不了数据库、数据访问,也是一切操作之本,而VS 2008中最大的亮点之一Linq也恰巧是做这个的,所以我的开发从Linq、从数据库开始。网上关于Linq的教学铺天盖地,我不准备重复,我只写下我遇到的问题。
    :o;k5^5v)x
    ;Z E#q)y6[4u ~  Linq,更新数据怎么就那么费劲?/`1S+H)i.V!f7\;Y

    /z:_7H1i6A1T-t1o+M  Linq的全称是Language Integrated Query ,也就是说Linq是以一个查询语言的方式出现在我们面前的。在查询方面Linq做了不少的优化,我们不用在费尽心思去拼装SQL语句、组装实体等,所有操作在Linq里都是强类型的,我们用C#代码轻松地写出漂亮的SQL语句。%^3t o,s#O
    程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛4H4C1Z+S'X%~6L3E0U;y9l
      那么做为一个查询语言,Linq在数据更新方面又是怎么表现的呢?通常来说Linq的更新会以以下的方式出现(绝大部分教程中都是这么写的)
    )S'B2^%T&G1_;O(}.n&jTechWeb-技术社区*o-s(Z&s$T
    1var ctx = new MyDataContext();"p0f;A)h%U
    2var user = ctx.Users.Where(u => u.UserId == userId).Single();.[;|0l o/C-g3p:R'w
    3user.UserName = "New User Name";
    %S/S/P9|)~.Z;t8x4ctx.SubmitChanges();
    "J'y(m1Y$a(g!C p
    +L5L#N/z"L)N+A(H2S.J$R  这些是C#代码,但是背后做了什么呢?Linq会为我们生成类似一下的SQL语句tech.techweb.com.cn2m3?9L!` G%`%B(?0S
    TechWeb-技术社区3['L)X!Q4w5b%l
    1--第一步,查询
    )q6P&j&m k$t
    #m R8D$s3w,X2SELECT UserId, UserName, FirstName, LastName, CreatTime From User WHERE UserId = @userId*`&w3A%H9u6H;t!O
    3/J/[!w8z-s2Q4j5{
    4--第二部,更新:d3V$X.j5C-M
    5UPDATE User SET UserName = @newUserName9q3h9F-o:T
    6WHERE UserId = @oldUserId, userName = @oldUserName, FirstName = @oldFirstName, LastName = @oldLastName
    'U5R&d4Q(w.x*t7M;c6G)D4E/x!?程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛
    2y0p7{"K3y-_  发现了什么?首先Linq会取出所有的字段,在user.UserName = "New User Name"的时候,记录下UserName字段被更新过了,UPDATE时会只更新UserName,但是把之前所有字段的值放在WHERE语句里来做为条件。,c3d8J-j0V/G-l,~&Y

    1h$f$X+y2^$u9Q"[.F8Z  Are you kidding?! 这样的效率实在是太差了吧?!tech.techweb.com.cn K(v4r/~+A"[,e!T2J
    (j9c7_1Y$P#e9c
      抛开效率问题,接下来我们看另外一种更新,有个某个字段记录页面被访问的次数,平时我们会用
    -R3Z.N!B%j6L-g-F2^"@&L b"l0v:G+I+|7K
    1UPDATE POST SET Views = Views + 1 WHERE PostId = @PostId
    .K i)T0c-J;H {%`5g%U程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛
    $T#A:g7Z)s  但是如果我们写下如下C#代码 [,@(g6r.l,W!z.k-P

    1b$q*~2b(r6?%@1var ctx = MyDataContext();TechWeb-技术社区0E!|!n4q6f"^
    2var post = ctx.Posts.Where(p => p.PostId = @postId).Single();
    %W2B"I-~9a3@)~/Otech.techweb.com.cn3post.Views++程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛8}#D7}0{7M3k;t"o
    4ctx.SubmitChanges();
    /I"A2J)d;t1W
    6v$D!e'\6r'q  Linq会怎么做呢?和上面一样!取出所有字段,把View加一,用所有字段做为条件(包括Views),更新回去。8R:v0?0s+e0b

    !{#B1c0g&x1]程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛  设想一下,这样一个被频繁使用的计数器,两次操作出现SELECT与UPDATE交叉情况的可能性很大,那么后者还能更新成功么?tech.techweb.com.cn,C"G!t"p+K,[)h
    程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛/|*L;b3B.q/V)O#h-J
      微软就是这样解释的,如果在你更新过程中,有其他人更新了这一行,那么这一行也就不是你所需要的那一行了,为了防止这样的冲突,所以把所有字段都放在WHERE语句中,这是by design的。程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛8k3d;S:y#W1x3z

    4E5f#Q$K*c1e2v+^7e4f8l程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛  你可以通过其他方法进行更新数据,然而在目前版本,这个方法也表现的不怎么样。-C(H6k-E1L/o&k6H)Y

    /w6S;q7p"m4m/P8Q!Z  System.Data.Linq.Table<T>有一个Attach方法,带有三个重载,用来直接更新数据的,我们来一个一个的来看看。程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛4`:P"N(W,t#}3q

    (c(d-S$[7@:b+A9w!~0V Ntech.techweb.com.cnAttach(T entity)
    &{1y;U:p r:B0I)C"K
    3N2p-@%F$z0o/u1var ctx = new MyDataContext();
    0c6N0i'}/K6u4FTechWeb-技术社区2var newUser = new User();
    -s5d/j#G*}6Q3newUser.UserId = new Guid("xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx");//假设作为参数传进来的TechWeb-技术社区*M)o7W+q'Y
    4newUser.UserName = "New User Name";-p;a8{0|$m;k;k
    5ctx.Users.Attach(newUser);!f a(s&j%f)z7}2d,v5G8s
    6ctx.SubmitChanges(); 程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛7D7E,G+|6w,q:n/I8y l
    tech.techweb.com.cn4b%[,C.i*\%\
      运行完全没有任何效果,SQL Profiler无任何记录。
    't.]$}-i9](i0l r(y!r%P:E3x.B6X
    Attach(T entity, T original)程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛-p/n/e/}&R

    )A/H6l1t&A3E6}#V程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛1var ctx = new MyDataContext();程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛!d%O;Z5]2e2z5x
    2var newUser = new User();程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛5m-f6Q(Y+R0g%N7l3R
    3newUser.UserId = new Guid("xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx");//假设作为参数传进来的3u,G*D.|%l/{5G
    4newUser.UserName = "New User Name";tech.techweb.com.cn*U"_$`3b:z
    5var user = ctx.User.Where(u => u.UserId = newUser.UserId).Single();
    4a9^'_-t*r6e7m"C;^%n5K-qtech.techweb.com.cn6ctx.Users.Attach(newUser, user);'O*h2o$F7w%y1S-s
    7ctx.SubmitChanges();
    /l2v.~2D&{ {/?3z(~3O
    2B(r'`0T&f \$G  运行时提示: Cannot add an entity with a key that is already in use.7k"K I.V%},g)n5g9~

    #k3B.v ]+IAttch(T entity, bool asmodified)程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛9W.c)H%O"M(u;s

    3h0s'{%x6d+]+\;R(b6{0]程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛1var ctx = new MyDataContext();-p2_'q,a9[8b/D
    2var newUser = new User();-H5N.a6s9~1W
    3newUser.UserId = new Guid("xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx");//假设作为参数传进来的
    (?2Z6k$E'@-x$n.[(n4X9L4newUser.UserName = "New User Name";
    9W%w5F7~-S9X7i*k$N8R程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛5ctx.Users.Attach(newUser, true);
    &@4W7[)G+Y4R:}8N#I0?6ctx.SubmitChanges();
    7J,Z9L!R!^0e,O.Itech.techweb.com.cn
    7\#p;P;W4Q4w'h程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛  运行时提示:An entity can only be attached as modified without original state if it declares a version member or does not have an update check policy."]7@)o,Z u

    %L3G#M/s+L5Z(D.q"a程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛  怎么办?提示中说"declares a version member ",通常来说是指SQL SERVER中TimeStamp类型的字段,在你所需要更新的表中加上一个字段,并标记为TimeStamp就可以了。但是这样做,对于我们来说仍然是个浪费,并且WHERE语句中仍然会出现TimeStamp的限制。
    (E#w'L0\;d6R7E,G%})J,O.J-T9o-^
      你还可以通过在字段上设置UpdateCheck.Never属性来避免更新检查,但是如果数据表更新、新增存储过程,需要重新生成dbml的话,你需要手动重新设置一遍。TechWeb-技术社区5M%J;b!y.y)`5~

    7G7^6p$V.Q$f3P9nTechWeb-技术社区  Linq甚至没有一个类似Web引用中Update Web Reference的操作来让你方便的在数据表更新后更新dbml,并且在这个版本都不会提供,你所能做的只有删除原来的表,刷新Server Exploer,重新拖拽到dbml的设计视图中,或者,写个脚本,让SQLMETAL来帮你完成这些。6P!Y"d%B-L&r

    1W h!U2o&?(p/?6~0]程序开发,操作系统,服务器,源码下载,Linux,Unix,BSD,PHP,Apach,asp,下载,源码,黑客,安全,技术社区,技术论坛  结论:
    +s,K1}+t3x'e y:q"v#Mtech.techweb.com.cnTechWeb-技术社区4i7z/?)[:Q'g%d+f8[
      Linq虽然做为一个查询语言出现,但是在数据更新方面也是做了不少工作的,尤其是一些CHECK的工作,但对于写惯SQL的我们来说,还是很不习惯,甚至觉得,这些工作你不替我做才好呢。
    *T(v)[&p;d%S"Dtech.techweb.com.cn k3N'v7M*f5b3T$w,R/k,v
      在没有更好解决办法的前提下,在更新操作上,老老实实的写SQL语句或者存储过程应该是个不坏的选择。

  • 相关阅读:
    简单工厂模式
    1.go的Hello
    Scrapy003-项目流程
    Django1.9开发博客(6)- 模板继承
    Django开发博客- 页面美化
    Django开发博客- 三部曲
    Django开发博客- 部署
    Django开发博客- 模型
    Django开发博客 入门篇
    内装式模块
  • 原文地址:https://www.cnblogs.com/dudu837/p/1397699.html
Copyright © 2011-2022 走看看