如何使此查询有效?

时间:2011-05-24 15:24:56

标签: sql sql-server-2008 subquery

昨天我发布了这个question,由于某些原因我不能让它工作。我得到了许多好的答案,但他们不符合我需要的范围。基本上我需要运行一个循环遍历所有客户并获取其Acct_Balance并将其设置为0的查询,但在我将其设置为0之前,我需要向AR_Transactions表添加具有该负数量的条目。所以对于实数来说,如果客户在客户表的Acct_Balance字段中有-200,那么我需要添加一个条目或200以使该值为0.有点像一个行项目0到它。如果客户在客户表的Acct_Balance字段中有200,那么我需要添加一个条目或者-200也将其归零。我试了几件事。这是我尝试过的一个例子。

BEGIN TRANSACTION 
INSERT INTO [cresql].[dbo].[AR_Transactions] 
(Trans_ID, DateTime , Dirty, Store_ID, Trans_Type,  Cashier_ID,  CustNum,     Trans_Amount, Prev_Cust_Balance ) 
SELECT  (SELECT MAX(Trans_ID ) + 1 FROM [cresql].[dbo].[AR_Transactions]), DATEADD(MINUTE, -30, Getdate()), 1, 1001, 'C', 100199,     CustNum,     -Acct_Balance,   Acct_Balance 
FROM  [cresql].[dbo].[Customer] 
WHERE Acct_Balance <> 0  
UPDATE [cresql].[dbo].[Customer] SET Acct_Balance = 0 
WHERE Acct_Balance <> 0  
COMMIT TRANSACTION

但是我收到了这个错误

Msg 2627, Level 14, State 1, Line 3
Violation of PRIMARY KEY constraint 'pkAR_Transactions'. Cannot insert duplicate key in object 'dbo.AR_Transactions'.
The statement has been terminated.

我尝试运行insert语句并且一切正常但是这个查询失败了....而且pkAR_Transactions主键显然不是非增量的,这就是为什么我要做这个hack来获取最后一个条目字段

这是两个表的数据库结构..

AR_Transactions表

column name type allow null 
Trans_ID bigint Unchecked
DateTime datetime Unchecked
Cashier_ID nvarchar(50) Checked
CustNum nvarchar(12) Unchecked
Trans_Type nvarchar(2) Unchecked
Prev_Cust_Balance money Checked
Prev_Inv_Balance money Checked
Trans_Amount money Unchecked
Payment_Method nvarchar(4) Checked
Payment_Info nvarchar(20) Checked
Description nvarchar(38) Checked
Invoice_Number bigint Unchecked
Store_ID nvarchar(10) Unchecked
Dirty bit Unchecked
Station_ID nvarchar(5) Checked
Payment_Type smallint Checked

客户表

column name type allow null 
CustNum nvarchar(12) Unchecked
First_Name nvarchar(15) Checked
Last_Name nvarchar(15) Unchecked
Company nvarchar(30) Checked
Address_1 nvarchar(30) Checked
Address_2 nvarchar(30) Checked
City nvarchar(20) Checked
State nvarchar(12) Checked
Zip_Code nvarchar(10) Checked
Phone_1 nvarchar(15) Checked
Phone_2 nvarchar(15) Checked
CC_Type nvarchar(5) Checked
CC_Num nvarchar(50) Checked
CC_Exp nvarchar(8) Checked
Discount_Level nvarchar(1) Unchecked
Discount_Percent real Unchecked
Acct_Open_Date datetime Checked
Acct_Close_Date datetime Checked
Acct_Balance money Checked
Acct_Max_Balance money Checked
Bonus_Plan_Member bit Unchecked
Bonus_Points int Checked
Tax_Exempt bit Unchecked
Member_Exp datetime Checked
Dirty bit Unchecked
Phone_3 nvarchar(15) Checked
Phone_4 nvarchar(15) Checked
EMail nvarchar(50) Checked
County nvarchar(30) Checked
Def_SP nvarchar(10) Checked
CreateDate datetime Checked
Referral nvarchar(20) Checked
Birthday datetime Checked
Last_Birthday_Bonus datetime Checked
Last_Visit datetime Checked
Require_PONum bit Unchecked
Max_Charge_NumDays int Checked
Max_Charge_Amount money Checked
License_Num nvarchar(20) Checked
ID_Last_Checked datetime Checked
Next_Start_Date datetime Checked
Checking_AcctNum nvarchar(20) Checked
PrintNotes bit Unchecked
Loyalty_Plan_ID bigint Checked
Tax_Rate_ID int Checked
Bill_To_Name nvarchar(30) Checked
Contact_1 nvarchar(30) Checked
Contact_2 nvarchar(30) Checked
Terms nvarchar(15) Checked
Resale_Num nvarchar(15) Checked
Last_Coupon datetime Checked
Account_Type smallint Checked
ChargeAtCost bit Checked
Disabled bit Checked
ImagePath nvarchar(255) Checked
License_ExpDate datetime Checked
TaxID nvarchar(20) Checked

2 个答案:

答案 0 :(得分:2)

运行此

SELECT  
    (SELECT MAX(Trans_ID ) + 1 
     FROM [cresql].[dbo].[AR_Transactions]), 
     DATEADD(MINUTE, -30, Getdate()), 
     1, 
     1001, 
     'C', 
     100199,
     CustNum,
     -Acct_Balance,
     Acct_Balance  
FROM  [cresql].[dbo].[Customer]  
WHERE Acct_Balance <> 0

我怀疑你会看到所有记录都有相同的trans_id。此外,这种黑客攻击总会导致竞争条件出现问题。

所以真正的问题是为什么不自动增量,如何在不引起竞争条件的情况下获得价值呢?该表的其他条目如何获得该值?这是您无法更改的第三方软件吗?如果有任何可能的方法来解决这个问题,我会先做。

由于您说您无法更改数据库,因此您仍需要了解它如何保留新的转换,以便其他任何人都不会尝试使用相同的数据。这很关键。通常,他们通过将id添加到单独的id表中并输出该插入的结果以在其余查询中使用来保留id - 如果不使用相同的方法,则会破坏内容。在您的情况下,您将需要获取与要插入的记录数相同的ID数。获得ID后,即可进行插入和更新。

答案 1 :(得分:1)

如果以下任何问题的答案为“是”,请不要继续。

  • 当您尝试进行更新时,是否正在使用AR表?
  • 您是否正在使用生产系统来开发脚本?

如果以下任何问题的答案为否,请不要继续。

  • 我正在开发系统
  • 如果出现任何问题,我可以从生产中获得经过验证的备份。

我认为问题是您目前用于计算AR id值的方法。代码

(SELECT MAX(Trans_ID ) + 1 FROM [cresql].[dbo].[AR_Transactions])

在脚本运行时返回单个值。为每个具有非零余额的客户插入单个值。您可以选择为customer表创建一个游标,以检索balance_id值为零的customer_id值。在代码中使用客户值将行添加到AR表,然后将客户表中的余额归零。转到下一位客户。

我不喜欢使用游标,所以请不要投票赞成这个建议。 OP不知道程序如何生成Id,所以这可能是他唯一的机会。