复制代码
代码如下:
if OBJECT_ID('tb') is not null drop table tbif OBJECT_ID('TEMP') is not null drop table TEMPif OBJECT_ID('FUN_NOWPRICE') is not null drop FUNCTION FUN_NOWPRICEif OBJECT_ID('FUN_NOWQTY') is not null drop FUNCTION FUN_NOWQTYgocreate table tb(id INT,Date1 datetime,ctype varchar(10),qnt float,pri float)--qnt 数量--pri 单价INSERT tbselect 0,'2009-1-1', '进货', 10, 100 union allselect 1,'2009-1-1', '进货', 50, 120 union allselect 2,'2009-1-2', '出货', 30, 150 union allselect 3,'2009-1-3', '进货', 40, 130 union allselect 4,'2009-1-3', '出货', 25, 160GO-- 我要算成本价,按移动加权平均/*1进货以后的成本价c1=(10*100+50*120)/(10+50)2出货以后的成本价c2=((10+50)*c1-30*c1)/((10+50)-30)=C2--也就是说出货的时候价格不变3进货以后的成本价c3=(((10+50)-30)*c2+40*130)/((10+50)-30+40)--也就是说进货的时候单价更新为(当前库存的总价值+库总价值)/入库后总数量以此类推...*/--想了半天,觉得只能用循环、递归、游标实现,因为出库时的价格是根据之前的记录算出来的。--也许有经典的算法,谁知道的麻烦教教我或者发个链接。--这个FUNCTION就是变相实现递归的CREATE FUNCTION FUN_NOWPRICE(@ID INT)RETURNS NUMERIC(19,6)ASBEGINRETURN (SELECT ISNULL(NOWPRICE,0) From(SELECT MAX(NOWPRICE) 'NOWPRICE' FROM TEMP T1 WHERE ID<@ID ANDNOT EXISTS(SELECT 1 FROM TEMP WHERE ID>T1.ID AND ID<@ID))T)ENDGO--这个FUNCTION是为了计算方便CREATE FUNCTION FUN_NOWQTY(@ID INT)RETURNS NUMERIC(19,6)ASBEGINRETURN (SELECT ISNULL(SUM(CASE CTYPE WHEN '进货' THEN QNT ELSE 0-QNT END),0) FROM TEMP WHERE ID<@ID)ENDGO--建一个临时表,包含原表参与运算的全部字段create table TEMP(id INT,Date1 datetime,ctype varchar(10),qnt float,pri float,NOWPRICE ASCASE ctypeWHEN '出货' THEN DBO.FUN_NOWPRICE(ID)ELSE (DBO.FUN_NOWPRICE(ID)*DBO.FUN_NOWQTY(ID)+QNT*PRI)/(DBO.FUN_NOWQTY(ID)+QNT)END)INSERT INTO TEMPSELECT * FROM TBORDER BY DATE1 ASC,ID ASCSELECT * FROM TEMP/*0 2009-01-01 00:00:00.000 进货 10 100 1001 2009-01-01 00:00:00.000 进货 50 120 116.6666666666672 2009-01-02 00:00:00.000 出货 30 150 116.6666673 2009-01-03 00:00:00.000 进货 40 130 124.2857144285714 2009-01-03 00:00:00.000 出货 25 160 124.285714*/
这个写法的不完善处在于它是根据ID和日期对记录进行排序的,对于同一天的出入库情况没有处理。实际运用中可以根据CREATEDATE等时间标志性字段来进行排序。--------------------------------------------------------------------------------第一次写技术性博客,希望这是一个好的开始,欢迎大家对我的算法进行指正^_^














发表评论