讨论 并发执行SQL时的更新丢失,如何解决这种问题
最近遇到一个问题,在操作SQL更新账户余额时,常常发现更新后的余额与实际的不相符,马上想到了是不是存在并发更新,一查果然如此。之前也曾遇到过这样的问题,当时没太在意,现在是解决的时候了。用Mysql模拟了一下这个过程:创建两张表,paralltable和parallalter,每次根据paratable表更新parallalter表,有如下几条逻辑
① parallProcess.jsp页面接收parall.jsp传过来的值,首先向paralltable插入一条记录,同时用paralltable的id,num,向parallalter表插入一条新记录;
② 之后parallProcess.jsp准备更新刚插入的paralltable记录,随之用更新后的paralltable的id,num,向parallalter表插入一条新记录,但在更新之前,先人为的设置一个延时,在延时期内执行并发操作③ ;
③ 从parall2.jsp页面向parallProcess2.jsp页面传值,在parallProcess.jsp延时期内,更新①中向paralltable插入的记录,同时用更新后的paralltable的id,num,向parallalter表插入一条新记录;
④ 执行完③的操作后,立刻执行②的操作。
如此的话,就会发现本来③是在②完全执行后,对paralltable进行更新,结果由于并发操作,③的更新结果丢失了,这种问题在SQL操作中经常发生。不知大家有在遇到这种问题时,是如何解决的。
附:模拟程序见 http://download.csdn.net/source/2389003
相关文档:
固定服务器角色
sysadmin 可以在 SQL Server 中执行任何活动。
serveradmin 可以设置服务器范围的配置选项,关闭服务器。
setupadmin 可以管理链接服务器和启动过程。
securityadmin 可以管理登录和 CREATE DATABASE 权限,还可以读取错误日志和更改密码。
processadmin 可以管理在 SQL Server 中运行的进程。
......
use RetalDB--表示在数据库RetalDB中进行的操作
go
if exists (select * from sysobjects where name='tb_user')
drop table tb_user
go
--创建客户表tb_user
create table tb_user
(
user_id int primary key,--指定为主键时,此列默认为非空,指定过多个限制条件时不用“'”隔开
us ......
今天写一个商品的修改功能时遇到的问题
商品中重量 weight 的数据库(SQL Server2005)类型定义为 float
在mappings 中转换为c#类型的一句为
<result property="Goods_Weight" column="Goods_Weight" type="float" dbType="float"/>
按理说这个 float 是一样的,转换完全不会出现问题,
实际程序运行时,系统报错
......
在Oracle中,如果在sql中出现'&'符号,会被自动转义;
而被要求输入在&符号后跟随的字符串的值,例如:
update tablename set columnName='http://www.g.cn/cv2.jsp?spid=222&cid=333';
执行这个操作时,Oracle会提示
Enter value for cid:
原因是在Oracle中 & 符号是作为转义字符使用的。
解决方法 ......
纵表转横表的"SQL"示例:
纵表结构:
fname ftype fvalue
小乔 zaocan 10
小乔 zhongcan 20
小乔 wancan 5
转换后的表结构:
fname zaocan_value zhongcan_value wancan_value
小乔 10 20 5
纵表转横表SQL示例:
select Fname, sum(case ......