insert/update操作失败原因及解决方法

原创 2007年09月19日 14:15:00
    很奇怪,asp+sql server人数在300人左右的时候,插入数据的操作有数据丢失的情况。

    查了半天,数据库相关表的编号列是primary key,有identity属性,每次自动加一。代码里面显示编号是通过select max(clNum)拿出来的。这样的话,在插入数据成功之后,还需要select一把。
     这里就有个漏洞,如果同时做插入动作的用户很多,对单个用户来说,插入数据之后,可能有很多其他用户也插入数据成功。这时再去取最大编号的话,就不能保证这个号码是这个用户插入的数据。
     到sql server的book online里查了查,sql server提供了取得插入记录后identity值的功能,
   
  • IDENT_CURRENT 返回为任何会话和任何作用域中的特定表最后生成的标识值。

  • @@IDENTITY 返回为当前会话的所有作用域中的任何表最后生成的标识值。

  • SCOPE_IDENTITY 返回为当前会话和当前作用域中的任何表最后生成的标识值。

    sql server7只提供了@@IDENTITY一种,而且没有区分会话和作用域,感觉是后两种的一个合集。
    引入以上概念,修改的代码,可以运行通过。只是有另外一个疑问,如果只是号码错,insert的动作应该成功,数据也应该能查到。但是,实际上用户输入的数据在数据库里没法找到。这个问题先挂着,继续观察。

================================================    

     隔了快一年,以上的问题算是有了个结果。经过sql profiler的监测,以及用户的反馈,经过分析发现,插入数据失败主要的原因还是由于用户输入的数据有问题造成的。这个有问题有不少种情况,例如,数据库里定义的是数字整型字段,而用户输入字母; 例如,定义的字段长度是varchar(100),用户输入超长的字符串; 前一种情况,只要在页面的逻辑上控制输入内容的类型就可以了。

     后一种情况比较有特点,因为用户有时会输入特殊字符,例如单引号,双引号,尖括号等等,这些字符输入后如果要正常显示在页面上,必须要用htmlencode转换一下才可以。正是这个htmlencode出了问题,一个双引号被转换之后,一个字符变成了6个字符(&quote;)。这样如果字符串正好100个字符长度,而里面又有2,3个双引号或是其他特殊字符的话,操作数据库肯定是会失败的。

     所以弄清楚了问题所在,解决起来也就容易了。首先是扩大数据库所定义字段的长度到200,但页面上还是限制长度100。但如果用户输入100个双引号,还不够用呢。不过这个是比较极端的情况,为了节省资源,暂时定义成200的长度。 其次,在连续的数据库操作之后(指update, insert,不包括select)加上检查,如果遇到错误,将提示用户有错误,但操作不回滚。

     这里要做个说明,原本想采用transaction模式来处理操作数据库所遇到的错误。但是,写好代码之后发现,使用transaction模式的性能有很大的问题。可能是操作数据库的步骤多,相关记录被锁住,直到操作结束。这样导致,其他用户无法同时查看相关记录。我用一台机器开两个IE窗口,模拟修改和查看操作时,修改记录就导致查看要等待很久。要是300个用户同时用,那要出大问题了。另外,象这样操作失败的情况不应该作为处理逻辑的一部分,错误只要出现就应改被及时纠正,而不是用一个单独的逻辑去专门处理。所以,只是检查是否有错误,而不回滚操作。

     另外,做个备忘。要在asp页面里实现事务模式有两种,一种是靠页面的事务,一种是数据库自身的事务。页面事务,遇到失败时,把所有操作,不仅仅是数据库事务都回滚。有时会和数据库的事务冲突,而不能正常使用。 数据库事务,只是针对数据库本身的操作,不牵涉页面上其他的逻辑。

     贴两个例子:

     sql server transaction:

Dim   connas   New   ADODB.Connection  
  ...  
   
  Sub   Date_syori()    
  On   Error   GoTo   Err_syori       
          conn.BeginTrans  
          ...  
   
          conn.execute("sql1")  
          ...  
           
          conn.execute("sql2")       
          conn.CommitTrans       
          Set   conn   =   Nothing       
          Exit   Sub       
  Err_syori:       
          conn   .RollbackTrans           
          Set   conn   =   Nothing      
          Exit   Sub       
  End   Sub  

     asp transaction:

<%@ TRANSACTION=Required%>
<%
Option Explicit
On Error Resume Next
Dim oConn, oRS

Set oConn = Server.CreateObject("ADODB.Connection")
oConn.Open "Provider=SQLOleDB;server=servername;Initial Catalog=pubs;uid=sa;pwd="
if err.Number <> 0 Then
    Response.Write "<BR>Error Occurred Opening Connection...<BR>"
    Response.Write "<BR>Error Description:" & err.Description & "...<BR>"
ObjectContext.SetAbort
    Response.End
else
    Response.Write "Connection Opened Successfully...<BR>"
    ObjectContext.SetComplete
End If


oConn.Execute "Select * from Authors"
if err.Number <> 0 Then
    Response.Write "<BR>Error Occurred Executing Query...<BR>"
    Response.Write "<BR>Error Description:" & err.Description & "...<BR>"
    oConn.Close
    Set oConn = Nothing
ObjectContext.SetAbort
Response.End
else
    Response.Write "<BR>Query Completed Successfully...<BR>"
    ObjectContext.SetComplete
End If

oConn.Close
Response.Write "<BR>Connection Closed Successfully...<BR>"
set oConn = Nothing
Response.Write "<BR>Test Completed Successfully...<BR>"

Sub OnTransactionCommit()
Response.Write "<p><b>The Transaction just committed</b>."
Response.Write "This message came from the "
Response.Write "OnTransactionCommit() event handler."
End Sub

Sub OnTransactionAbort()
Response.Write "<p><b>The Transaction just aborted</b>."
Response.Write "This message came from the "
Response.Write "OnTransactionAbort() event handler."
End Sub

%>

浅谈SQL Update返回影响行数的意义

在大学的时候学习了JSP,其中使用JDBC进行数据库操作,有一个语句是Statement.ExecuteUpdate,这个语句执行一个SQL的更新操作(如delete,update,insert),返...
  • muximuxi_kgsecond
  • muximuxi_kgsecond
  • 2012年12月09日 11:12
  • 6793

sql更新出错,事物回滚

private void Save11()         {             SqlConnection connection = OpenSql();          ...
  • misssix
  • misssix
  • 2016年09月09日 17:12
  • 921

Meteor update等操作失败原因及解决方法

Meteor 提供了两个 MongoDB 数据库:一个客户端缓存数据库和服务器上的一个 MongoDB 数据库。当一个用户更改一些数据时(例如通过单击 Save),在浏览器中运行的 JavaScrip...
  • huangxiongbiao
  • huangxiongbiao
  • 2015年04月29日 17:02
  • 1697

sqlserver高并发情况下 select 和update操作造成死锁的解决方法

最近在项目上线使用过程中使用SqlServer的时候发现在高并发情况下,频繁更新和频繁查询引发死锁。通常我们知道如果两个事务同时对一个表进行插入或修改数据,会发生在请求对表的X锁时,已经被对方持有了。...
  • u013547384
  • u013547384
  • 2016年12月16日 09:14
  • 3388

配置windows update失败还原更改_解决方案

开启后提示如下: 配置windows update失败还原更改 最近公司同事经常遇到这个问题,因为很多人对电脑的使用程度还不高,不太理解这种问题的起因。 原因如题: ...
  • fhqsse220
  • fhqsse220
  • 2016年02月03日 15:07
  • 68592

win7旗舰版系统360浏览器安装扩展插件失败原因和三种解决方法

  • 2017年04月18日 10:51
  • 114KB
  • 下载

Hibernate_should be mapped with insert="false" update="false"的解决方法

今天在配置hibernate 单向多对一映射时出现了启动异常,第一句末尾出现了should be mapped with insert="false" update="false"提示 原因是存在账...
  • dd_forlife
  • dd_forlife
  • 2015年11月10日 09:43
  • 161

kali apt-get update失败解决方法

apt-get update ->”Hash Sum mismatch“ 原来是因为url被墙了,要翻墙解决
  • wang471003247
  • wang471003247
  • 2015年05月10日 21:17
  • 4069

微信公众号基本配置时token获取失败原因及解决方法

微信公众号基本配置时token获取失败原因及解决方法目前SAE(该服务器拿来专门做微信开发)中获得的所有请求应为来自微信服务器发出的请求,比如有人关注了你的公众号并向你发送了一条信息,微信服务器会把信...
  • Part_In
  • Part_In
  • 2017年05月09日 16:41
  • 2498

mysql安装不上怎么办 mysql安装失败原因和解决方法

困难1:MySQL 5.1 安装过程中报apply security setting错误 1.卸载MySQL。 2.删除目录 C:\Documents and Settings\All Users...
  • DickyQie
  • DickyQie
  • 2017年12月09日 08:59
  • 335
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:insert/update操作失败原因及解决方法
举报原因:
原因补充:

(最多只允许输入30个字)