0

我正在使用 Access DB 作为经典 asp 网页的后端。我正在创建一个使用参数的已保存查询。我在 Access 2000 的查询生成器中进行测试。字段都是文本,默认值为 NULL,[in_b] 允许长度为零,其中有 9 个,分别命名为 B160、B80、B40、B30 等。

我的查询是:

 SELECT COUNT([in_b]) AS BCnt FROM tblScore 
 WHERE UCALL=[in_call] and NOT ISNULL([in_b]);

这将返回给定 UCALL 的所有行的计数。如果我将 2[in_b] 中的任何一个更改为实际的列名,查询就会给我我想要的(在这种情况下为 0),但是我必须运行 9 个查询。

表格示例

Call  Zone B160   B80  B40
NF4L   1    NULL   X    NULL
NF4L   6    Null  Null  NULL
NF4L   20     X     X   Null
WA4B   2    NULL  NULL   X

If in_call is NF4L and in_b is B160, I expect 1
If in_call is NF4L and in_b is B80,  I expect 2
If in_call is NF4L and in_b is B40,  I expect 0

目标是对给定调用的所有非空“B”列进行计数。

4

2 回答 2

0

在这种情况下,您使用参数 ([in_b]) 作为对列的引用,而不是实际数据值。我认为您不会对这种设计走得太远。

除了构建 9 个不同的查询(不要这样做!)之外,您还可以尝试另外两个选项:

  1. 将您的表格设计更改为更多的键值对
  2. 根据提供的条件动态构建查询

对于选项 1,您可以将表更改为如下所示:

Call | Zone | BField | BFieldValue
NF4L | 1    | B80    | X
NF4L | 1    | B160   | NULL
NF4L | 1    | B40    | NULL
NF4L | 6    | B160   | NULL
NF4L | 6    | B80    | NULL
NF4L | 6    | B40    | NULL
NF4L | 20   | B160   | X
NF4L | 20   | B80    | X
NF4L | 20   | B40    | NULL
...

您的查询可能如下所示:

SELECT BField, COUNT([BFieldValue]) AS BCnt 
FROM tblScore 
WHERE UCALL=[in_call] and [BFieldValue] IS NOT NULL
GROUP BY BField;

此选项是两者中可扩展性更强的一个。如果您需要在某些时候添加额外的“BFields”,这样做是微不足道的,但代价是必须重新设计表格,如何将数据添加到表格中。

选项 2 是在运行时动态构建查询,以便您可以根据您概述的条件替换实际的列名。例如:

Dim SQL as String
Dim BField as String
Dim InCall as String

'These would typically be passed in, not hardcoded as they are here
Set InCall = "NF4L"
Set BField = "B160"

SET SQL = "SELECT Count([" & BField & "] FROM tblScore WHERE [Call] ='" & InCall & "' AND NOT ISNULL([" & BField & "]);"

'From here, either open up a recordset based on this SQL or create a new QueryDef object and execute it.
于 2013-01-31T14:00:26.397 回答
0

我知道使用 SQL Server,您可以传递参数并使用CASE语句,如下所示:

SELECT COUNT(*) 
FROM Test
WHERE CASE 
   WHEN @col = 'col1'
   THEN col1 
   WHEN @col = 'col2' 
   THEN col2 END IS NOT NULL

其中@col 是传入的参数。我认为您可以在 MS Access 中使用IIF-- 试一试:

SELECT COUNT(*) 
FROM Test
WHERE NOT ISNULL(IIF(@col = 'col1', col1, IIF(@col = 'col2', col2, ...))

或者,您可以使用动态 SQL 执行此服务器端,但请确保使用参数化查询。这是一个很好的例子:

Const ad_nVarChar = 202
Const ad_ParamInput = 1

strSQL = "SELECT Count(*) FROM tblScore WHERE [Call] = ? AND NOT ISNULL(?); "

Set cmd = Server.CreateObject("ADODB.Command")    
cmd.ActiveConnection = connStr
cmd.CommandText = strSQL
cmd.CommandType = adCmdText
cmd.Parameters(0) = "NF4L"
cmd.Parameters(1) = "B160"

Set rst = cmd.Execute()

祝你好运。

于 2013-01-31T14:12:32.713 回答