用计算列实现移动加权平均算法_数据库其它

复制代码 代码如下:

if OBJECT_ID('tb') is not null drop table tb
if OBJECT_ID('TEMP') is not null drop table TEMP
if OBJECT_ID('FUN_NOWPRICE') is not null drop FUNCTION FUN_NOWPRICE
if OBJECT_ID('FUN_NOWQTY') is not null drop FUNCTION FUN_NOWQTY
go

create table tb(
id INT
,Date1 datetime
,ctype varchar(10)
,qnt float
,pri float
)

--qnt 数量
--pri 单价
insert tb
select 0,'2009-1-1', '进货', 10, 100 union all
select 1,'2009-1-1', '进货', 50, 120 union all
select 2,'2009-1-2', '出货', 30, 150 union all
select 3,'2009-1-3', '进货', 40, 130 union all
select 4,'2009-1-3', '出货', 25, 160
GO
-- 我要算成本价,按移动加权平均

/*
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)
AS
BEGIN
RETURN (SELECT ISNULL(NOWPRICE,0) FROM
(SELECT MAX(NOWPRICE) 'NOWPRICE' FROM TEMP T1 WHERE ID<@ID AND
NOT EXISTS(SELECT 1 FROM TEMP WHERE ID>T1.ID AND ID<@ID))
T)
END
GO
--这个FUNCTION是为了计算方便
CREATE FUNCTION FUN_NOWQTY(@ID INT)
RETURNS NUMERIC(19,6)
AS
BEGIN
RETURN (SELECT ISNULL(SUM(CASE CTYPE WHEN '进货' THEN QNT ELSE 0-QNT END),0) FROM TEMP WHERE ID<@ID)
END
GO

--建一个临时表,包含原表参与运算的全部字段
create table TEMP(
id INT
,Date1 datetime
,ctype varchar(10)
,qnt float
,pri float
,NOWPRICE AS
CASE ctype
WHEN '出货' THEN DBO.FUN_NOWPRICE(ID)
ELSE (DBO.FUN_NOWPRICE(ID)*DBO.FUN_NOWQTY(ID)+QNT*PRI)/(DBO.FUN_NOWQTY(ID)+QNT)
END)

INSERT INTO TEMP
SELECT * FROM TB
ORDER BY DATE1 ASC,ID ASC

SELECT * FROM TEMP

/*
0 2009-01-01 00:00:00.000 进货 10 100 100
1 2009-01-01 00:00:00.000 进货 50 120 116.666666666667
2 2009-01-02 00:00:00.000 出货 30 150 116.666667
3 2009-01-03 00:00:00.000 进货 40 130 124.285714428571
4 2009-01-03 00:00:00.000 出货 25 160 124.285714
*/

这个写法的不完善处在于它是根据ID和日期对记录进行排序的,对于同一天的出入库情况没有处理。实际运用中可以根据CREATEDATE等时间标志性字段来进行排序。
--------------------------------------------------------------------------------

第一次写技术性博客,希望这是一个好的开始,欢迎大家对我的算法进行指正^_^

时间: 2024-10-18 05:03:05

用计算列实现移动加权平均算法_数据库其它的相关文章

计算机科学中32个常用的基础算法_其它综合

奥地利符号计算研究所(Research Institute for Symbolic Computation,简称RISC)的Christoph Koutschan博士在自己的页面上发布了一篇文章,提到他做的一个调查,参与者大多数是计算机科学家,他请这些科学家投票选出最重要的算法,以下是这次调查的结果,按照英文名称字母顺序排序: 1.A* 搜索算法--图形搜索算法,从给定起点到给定终点计算出路径.其中使用了一种启发式的估算,为每个节点估算通过该节点的最佳路径,并以之为各个地点排定次序.算法以得到

数据库快照,自定义函数与计算列

数据库快照,自定义函数与计算列 1.数据库快照 数据库快照就是保存某个数据库在快照那一瞬间的状态.快照和备份原理上有所不同,但是功能有一点相同那就是可以将数据还原为备份的那个时刻.快照的原理是新建一个数据库指针,在原数据库没有变化的情况下快照是不占用空间的,而数据库发生了变化,那么在变化前,被修改的数据页会先复制一份到快照文件中,然后再对原数据页进行修改.显然这样做的好处就是比备份数据库占用空间小.快照是只读的,你可以直接在SQL语句中把他当数据库用: use snap1;--使用快照 sele

如何在SQL Server计算列和平均值

严格来说,Having并不需要一个子表,但没有子表的Having并没有实际意义.如果你只需要一个表,那么你可以用Where子句达到一切目的.为进行实践,Having预先假定至少两个表和一个基于第二个表的合计函数. 下面是一个简单的例子:你想要订单总数超过25000美元的客户清单.你需要适当连接的三个表:Customer.SalesOrderHeader和SalesOrderDetail.然后,你求Detail的和,并将总数与25000美元进行比较.查看列表A. 查看列表A中的代码时,有一件事情并

用SQL Server HAVING子句计算列和平均值

在本文中,简单说明这个子句,并提供一些代码实例,这是说明HAVING子句用法的最佳方法. 严格来说,HAVING并不需要一个子表,但没有子表的HAVING并没有实际意义.如果你只需要一个表,那么你可以用WHERE子句达到一切目的.为进行实践,HAVING预先假定至少两个表和一个基于第二个表的合计函数. 下面是一个简单的例子:你想要订单总数超过25000美元的客户清单.你需要适当连接的三个表:Customer.SalesOrderHeader和SalesOrderDetail.然后,你求Detai

在SQL Server数据库中求计算列和平均值

 严格来说,Having并不需要一个子表,但没有子表的Having并没有实际意义.如果你只需要一个表,那么你可以用Where子句达到一切目的.为进行实践,Having预先假定至少两个表和一个基于第二个表的合计函数. 下面是一个简单的例子:你想要订单总数超过25000美元的客户清单.你需要适当连接的三个表:Customer.SalesOrderHeader和SalesOrderDetail.然后,你求Detail的和,并将总数与25000美元进行比较.查看列表A. 查看列表A中的代码时,有一件事情

怎么在SQLServer计算列和平均值

严格来说,Having并不需要一个子表,但没有子表的Having并没有实际意义.如果你只需要一个表,那么你可以用Where子句达到一切目的.为进行实践,Having预先假定至少两个表和一个基于第二个表的合计函数. 下面是一个简单的例子:你想要订单总数超过25000美元的客户清单.你需要适当连接的三个表:Customer.SalesOrderHeader和SalesOrderDetail.然后,你求Detail的和,并将总数与25000美元进行比较.查看列表A. 查看列表A中的代码时,有一件事情并

SQL Server利用HashKey计算列解决宽字段查询的性能问题

SQL Server利用HashKey计算列解决宽字段查询的性能问题 主人翁        本文主人翁:MSSQL菜鸟和MSSQL老鸟. 问题提出        某年某月某日,某MSSQL菜鸟满脸愁容的跑到老鸟跟前,心灰意懒的对老鸟说"我最近遇到一个问题,很大的问题,对,非常大的问题".老鸟不急不慢的推了推2000度超级近视眼镜框,慢吞吞的说:"说来听听".        "我有一个100万数据量的表,有一个宽度为7500字段,不幸的是现在我需要根据这个字

数据库一个简单的计算列问题

问题描述 数据库一个简单的计算列问题 定义学生表,其中规定: ? ? 学号列是主关键字: 院系列为计算列(取学号列的第 3 和第 4 个字符) ,并且院系值参照院系表的编号值(院 系表是被参照表,主关键字是编号:参照表是学生表,外部关键字是院系) ,此约束说明 一名学生一定属于某个院系: ? ? ? 姓名列不允许为空值: 性别必须取值"男"或"女": 学生的学籍状态为正常.留级.休学或退学. 代码: create table student.学生 (学号 char(

datatable计算列-C# Winform项目,DataTable怎样能实现用SQL标量函数作为计算列公式?

问题描述 C# Winform项目,DataTable怎样能实现用SQL标量函数作为计算列公式? 当把DataTable的计算列公式设置为数据库标量函数dbo.function()时,运行程序是会提示function()不存在. DataTable的列可以实现和SQL一样的计算列规范吗?或者有其他可替代方案?研究好几天了,还是没头绪,求大神解惑. 解决方案 反正一般都用linq,当然你不会linq就没辙了.