【发布时间】:2020-04-24 21:13:59
【问题描述】:
我有一个关于交易的非常简单的问题。 (在 sql server 2000 中,但我猜它适用于一般的数据库事务)。
tblPrimaryKey
PkId
-----
1
2
3
tblForeignKey
Id ForeignKey
---- -----
1 1
2 2
3 3
4 1
我有 2 个表,一个引用另一个(tblForeignKey.ForeignKey 引用 tblPrimaryKey.PkID)。现在我有一些逻辑可以通过删除和重新插入一个键来改变主键的表。
删除后,数据库当然会处于不一致的状态。我查看了我的旧脚本,我首先放弃了关系,然后重新创建了它。但我的问题是:我了解到事务是原子的,所以事务内部允许不一致的状态。
所以我想这样的事情应该可以工作:
BEGIN TRAN eg
DELETE tblPrimaryKey WHERE PkId = 3
INSERT INTO tblPrimaryKey SELECT 3
COMMIT TRAN eg
但这不起作用。有人可以为我提供一个应用此逻辑的工作事务示例吗?
更新:
一致性 这个特性意味着数据库在事务之前和之后应该是一致的。
在任何情况下都不能将部分事务提交给数据库,因为这会使数据库处于不一致的状态。
这是否暗示中事务不一致是可能的?
更新:
有些人问我为什么在这种情况下不使用更新。有点复杂,但我试一试:所需的 sql 是发布脚本的一部分,该脚本从视图构建表,然后更新这些表。由于视图包含发布模型,因此仅在此处对视图进行了更改。脚本的其余部分不能依赖列名来进行更新。
当然我可以查询这些列名,但当时看起来很麻烦,所以我选择不这样做,而是放弃约束并重建它们。现在我必须承认我对那个解决方案感到不舒服,所以现在我确实使用了更新。我写了一个存储过程来做到这一点,如果有人现在有其他解决方案,请告诉我。
CREATE PROC usp_SyncRecords
(
@tableName1 as nvarchar(255),
@tableName2 as nvarchar(255),
@joinClause as nvarchar(255),
@whereClause as nvarchar(1000)
)
-- this proc updates all fields in table 1 that have corresponding names
-- in table2 to the value of the field in table2.
AS
BEGIN
DECLARE @sqlClause nvarchar(4000)
DECLARE @curFieldName nvarchar(255)
DECLARE @sqlColumnCursorClause nvarchar(1000)
SET @sqlClause = 'UPDATE [' + @tableName1 + '] SET '
-- get FieldNames for second table
SET @sqlColumnCursorClause =
'DECLARE cur CURSOR FAST_FORWARD FOR SELECT name FROM syscolumns ' +
'WHERE id=' + CAST(object_id(@tableName2) as nvarchar(50))
EXEC sp_executeSql @sqlColumnCursorClause
OPEN cur
-- compose sqlClause using fieldnames
FETCH NEXT FROM CUR INTO @curFieldName
WHILE @@fetch_status <> -1
BEGIN
SET @sqlClause = @sqlClause + @curFieldName + '=' +
@tableName2 + '.' + @curFieldName + ','
FETCH NEXT FROM CUR INTO @curFieldName
END
CLOSE cur
DEALLOCATE cur
-- drop last comma
SET @sqlClause = LEFT(@sqlClause,LEN(@sqlClause) -1)
-- adding from/join/where clauses
SET @sqlClause = @sqlClause + ' FROM [' + @tableName1 + '] INNER JOIN [' + @tableName2 + '] '
+ 'ON ' + @joinClause + ' WHERE ' + @whereClause
EXEC sp_executeSQL @sqlClause
END
【问题讨论】:
-
你到底想做什么?
-
您是说不能插入到 tblPrimaryKey 表中,因为 PkId(或 Id)列是身份种子列?
-
@kane :不,这里与身份无关
标签: sql transactions