- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我在 SQL Server 中使用一个查询,它需要一个范围来检查数字是否在该范围内(例如,在下面检查 DemographicGroupDimID
是否为(1,2 或 3) . 在做了一些谷歌搜索后,我发现能够做到这一点的唯一方法如下:
DECLARE @adults table (Id int)
INSERT INTO @adults VALUES (1), (2), (3)
SELECT [date], [station], [impression] = SUM([impressions]) / COUNT(DISTINCT [datetime] )
FROM
(SELECT [datetime] = DATEADD(minute,td.Minute,DATEADD(hour,td.NielsenLocalHour,CONVERT(smalldatetime, ddt.DateKey))), [date] = ddt.DateKey, [station] = nd.Name, [impressions] = SUM(naf.Impression)
FROM [Nielsen].[dbo].[NielsenAnalyticsFact] as naf
LEFT JOIN [dbo].[DateDim] AS ddt
ON naf.StartDateDimID = ddt.DateDimID
LEFT JOIN [dbo].NetworkDim as nd
ON naf.NetworkDimID = nd.NetworkDimID
LEFT JOIN [dbo].TimeDim as td
ON naf.QuarterHourDimID = td.TimeDimID
WHERE (naf.NielsenMarketDimID = 1
AND naf.RecordTypeDimID = 2
AND naf.AudienceEstimateTypeDimID = 1
AND naf.DailyOrWeeklyDimID = 1
AND naf.RecordSequenceCodeDimID = 5
AND naf.ViewingTypeDimID = 4
AND naf.QuarterHourDimID IS NOT NULL
AND naf.DemographicGroupDimID < 31
AND nd.Affiliation = 'Cable'
AND naf.NetworkDimID != 1278
AND naf.DemographicGroupDimID in (SELECT Id FROM @adults))
GROUP BY DATEADD(minute,td.Minute,DATEADD(hour,td.NielsenLocalHour,CONVERT(smalldatetime, ddt.DateKey))), nd.Name, ddt.DateKey)
AS grouped_table
GROUP BY [date], [station]
ORDER BY [date], [station]
如果我需要使用不同的范围动态执行此操作,则会失败,如下所示:
from queries import DB_CREDENTIALS
import pyodbc
import pandas as pd
sql_ = """DECLARE @adults table (Id int)
INSERT INTO @adults VALUES ?
SELECT [date], [station], [impression] = SUM([impressions]) / COUNT(DISTINCT [datetime] )
FROM
(SELECT [datetime] = DATEADD(minute,td.Minute,DATEADD(hour,td.NielsenLocalHour,CONVERT(smalldatetime, ddt.DateKey))), [date] = ddt.DateKey, [station] = nd.Name, [impressions] = SUM(naf.Impression)
FROM [Nielsen].[dbo].[NielsenAnalyticsFact] as naf
LEFT JOIN [dbo].[DateDim] AS ddt
ON naf.StartDateDimID = ddt.DateDimID
LEFT JOIN [dbo].NetworkDim as nd
ON naf.NetworkDimID = nd.NetworkDimID
LEFT JOIN [dbo].TimeDim as td
ON naf.QuarterHourDimID = td.TimeDimID
WHERE (naf.NielsenMarketDimID = 1
AND naf.RecordTypeDimID = 2
AND naf.AudienceEstimateTypeDimID = 1
AND naf.DailyOrWeeklyDimID = 1
AND naf.RecordSequenceCodeDimID = 5
AND naf.ViewingTypeDimID = 4
AND naf.QuarterHourDimID IS NOT NULL
AND naf.DemographicGroupDimID < 31
AND nd.Affiliation = 'Cable'
AND naf.NetworkDimID != 1278
AND naf.DemographicGroupDimID in (SELECT Id FROM @adults))
GROUP BY DATEADD(minute,td.Minute,DATEADD(hour,td.NielsenLocalHour,CONVERT(smalldatetime, ddt.DateKey))), nd.Name, ddt.DateKey)
AS grouped_table
GROUP BY [date], [station]
ORDER BY [date], [station]"""
with pyodbc.connect(DB_CREDENTIALS) as cnxn:
df = pd.read_sql(sql=sql_, con=cnxn, params=['(30)'])
---------------------------------------------------------------------------
DatabaseError Traceback (most recent call last)
<ipython-input-5-4b63847d007f> in <module>()
1 with pyodbc.connect(DB_CREDENTIALS) as cnxn:
----> 2 df = pd.read_sql(sql=sql_, con=cnxn, params=['(30)'])
C:\Users\mburke\AppData\Local\Continuum\Anaconda64\lib\site-packages\pandas\io\sql.pyc in read_sql(sql, con, index_col, coerce_float, params, parse_dates, columns, chunksize)
497 sql, index_col=index_col, params=params,
498 coerce_float=coerce_float, parse_dates=parse_dates,
--> 499 chunksize=chunksize)
500
501 try:
C:\Users\mburke\AppData\Local\Continuum\Anaconda64\lib\site-packages\pandas\io\sql.pyc in read_query(self, sql, index_col, coerce_float, params, parse_dates, chunksize)
1593
1594 args = _convert_params(sql, params)
-> 1595 cursor = self.execute(*args)
1596 columns = [col_desc[0] for col_desc in cursor.description]
1597
C:\Users\mburke\AppData\Local\Continuum\Anaconda64\lib\site-packages\pandas\io\sql.pyc in execute(self, *args, **kwargs)
1570 ex = DatabaseError(
1571 "Execution failed on sql '%s': %s" % (args[0], exc))
-> 1572 raise_with_traceback(ex)
1573
1574 @staticmethod
C:\Users\mburke\AppData\Local\Continuum\Anaconda64\lib\site-packages\pandas\io\sql.pyc in execute(self, *args, **kwargs)
1558 cur.execute(*args, **kwargs)
1559 else:
-> 1560 cur.execute(*args)
1561 return cur
1562 except Exception as exc:
DatabaseError: Execution failed on sql 'DECLARE @adults table (Id int)
INSERT INTO @adults VALUES ?
SELECT [date], [station], [impression] = SUM([impressions]) / COUNT(DISTINCT [datetime] )
FROM
(SELECT [datetime] = DATEADD(minute,td.Minute,DATEADD(hour,td.NielsenLocalHour,CONVERT(smalldatetime, ddt.DateKey))), [date] = ddt.DateKey, [station] = nd.Name, [impressions] = SUM(naf.Impression)
FROM [Nielsen].[dbo].[NielsenAnalyticsFact] as naf
LEFT JOIN [dbo].[DateDim] AS ddt
ON naf.StartDateDimID = ddt.DateDimID
LEFT JOIN [dbo].NetworkDim as nd
ON naf.NetworkDimID = nd.NetworkDimID
LEFT JOIN [dbo].TimeDim as td
ON naf.QuarterHourDimID = td.TimeDimID
WHERE (naf.NielsenMarketDimID = 1
AND naf.RecordTypeDimID = 2
AND naf.AudienceEstimateTypeDimID = 1
AND naf.DailyOrWeeklyDimID = 1
AND naf.RecordSequenceCodeDimID = 5
AND naf.ViewingTypeDimID = 4
AND naf.QuarterHourDimID IS NOT NULL
AND naf.DemographicGroupDimID < 31
AND nd.Affiliation = 'Cable'
AND naf.NetworkDimID != 1278
AND naf.DemographicGroupDimID in (SELECT Id FROM @adults))
GROUP BY DATEADD(minute,td.Minute,DATEADD(hour,td.NielsenLocalHour,CONVERT(smalldatetime, ddt.DateKey))), nd.Name, ddt.DateKey)
AS grouped_table
GROUP BY [date], [station]
ORDER BY [date], [station]': ('42000', "[42000] [Microsoft][ODBC SQL Server Driver][SQL Server]Incorrect syntax near '@P1'. (102) (SQLExecDirectW); [42000] [Microsoft][ODBC SQL Server Driver][SQL Server]Statement(s) could not be prepared. (8180)")
这是因为 declare 语句需要在 select 语句本身的范围内吗?我不确定 pandas
如何处理 pyodbc
游标对象,因此我不确定此错误的来源。
编辑:请注意,我在此实例中传递的参数是 (30)
只是为了使用范围内只有一个数字失败的简单情况。对于更复杂的字符串,例如 (1), (2), (3)
,它当然也会失败,就像上面的示例一样。
最佳答案
如果您使用 prepared statements在您的 SQL 中,您不能为一个占位符/参数/绑定(bind)变量放置多个值!
除此之外,您只能使用占位符/参数/绑定(bind)变量来代替 literals ,您不能将它用于 不是 文字的 SQL 语句的一部分。
在您的情况下,您尝试将 (
和 )
作为 SQL 的一部分,但不是文字作为参数。
使用参数/准备好的语句/绑定(bind)变量也可以保护您免受一些 SQL 注入(inject)。
就是说,尝试按如下方式更改您的代码:
改变
INSERT INTO @adults VALUES ?
到
INSERT INTO @adults VALUES (?)
和
df = pd.read_sql(sql=sql_, con=cnxn, params=['(30)'])
到
df = pd.read_sql(sql=sql_, con=cnxn, params=['30'])
更新:
你可以这样准备你的SQL:
In [9]: vals = [20,30,40]
In [32]: vals
Out[32]: [20, 30, 40]
In [33]: ' (?)' * len(vals)
Out[33]: ' (?) (?) (?)'
然后:
In [14]: sql_ = """DECLARE @adults table (Id int)
....: INSERT INTO @adults VALUES {}
....:
....: SELECT [date],
....: """
In [15]: sql_.format(' (?)' * len(vals))
Out[15]: 'DECLARE @adults table (Id int)\nINSERT INTO @adults VALUES (?) (?) (?)\n\nSELECT [date],\n'
注意生成的(?) (?) (?)
最后调用你的 SQL:
df = pd.read_sql(sql=sql_.format(' (?)' * len(vals)), con=cnxn, params=vals)
关于python - 使用参数 : pandas (or pyodbc) not functioning properly 从 SQL Server 读取,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37051297/
如果你有一个简单的程序名 pyodbcTest.py import pyodbc print "pass" 然后使用 cx_freeze 编译它 cxfreeze --targe-dir=cxTest
我有一个 FileMaker db 在 Mac Mini 上的 FileMaker Server 14 上运行,我正在尝试使用 pyodbc 来获取它。进展不顺利。 首先,什么有效: telnet 1
这里有点菜鸟。我正在尝试创建一个 python 脚本来对基于 SQL 的数据库进行压力测试。 我一直在尝试使用 pyodbc 连接到数据库,但是我在 pyodbc.connect(connection
我想使用 pyodbc 和 freetds 从 Python 连接到 SQL SERVER 数据库。 我的连接正常。 我的代码: class GetSystems(Resource): def get
我正在尝试将 pyodbc 库导入 google colab,但出现此错误。 以防万一,我在我的笔记本中安装了 Anaconda,而且我从来没有遇到过 pyodbc 的问题。 你能帮我吗? 太棒了!
我最近跑了pip install pyodbc 。这表明它在cmd上成功安装了pyodbc。 但是当我这样做时 import pyodbc在 IDLE 中,我收到一条错误消息“没有名为 pyodbc
我写了一个简短的 python 脚本,它试图导入 pyodbc 扩展包,以便我可以访问我的 SQL 表。 import pyodbc as pyodbc cnxn = pyodbc.connect('
我正在尝试使用列表作为值的源将数据插入到 Access mdb 文件中。 cursor.execute("select * from Components") cursor.executemany("
我使用 pyodbc 的时间很短,现在面临着从文件中应用 sql 脚本来执行一些 View 创建的问题。为了应用 sql 文件,我使用另一个线程中的示例 - Follow up: Execute .s
我传递给 pyodbc.connect(conn_str) 的 conn_str 是: conn_str = str("DRIVER={{ODBC Driver 13 for SQL Server}}
好的,我已经在 mac 上安装了 python3.8,分别安装了多个 odbcdrivers。 我创建了一个新的虚拟环境,并在该环境中运行: >>> import pyodbc >>> pyodbc.
我有一个在Mac Mini上的FileMaker Server 14上运行的FileMaker db,我试图通过pyodbc来解决。不好 首先,什么有效: telnet 192.169.19.3 23
我可以按照以下说明确定是否安装了 suds: checking suds version in python 即通过在 Python 控制台中运行以下命令: >>>> import suds >>>>
我可以按照以下说明确定是否安装了 suds: checking suds version in python 即通过在 Python 控制台中运行以下命令: >>>> import suds >>>>
当我使用 pyodbc 查询 SQL 服务器时,我只得到我请求的字段的名称(而不是值本身)。 import pyodbc conn = pyodbc.connect(connection_string
我尝试使用以下代码查询 SQL Server 2012 数据库: import pyodbc class sqlserverConnector: def __init__(self, conn
我目前正在编写一个程序,该程序将从 Excel 电子表格中获取数据并将其插入到我在程序中创建的 SQL Server 表中。 我之前已将日期时间列指定为 nvarchar(250) ,以便使整个程序正
如果有人能指出我正确的方向,我会很感兴趣。 我有一个很长的存储过程(其中还包含对其中其他存储过程的调用),用于更新各种表。 如果我在管理工作室中运行存储过程,它运行良好。如果我从 pyodbc 调用它
你好,我有一个带有 SQL 的 Python 脚本,我使用 fecthall() 方法提取我想要的元素,直到那里一切都很好,因为我想将两列(datetime.date 和 datetime.time)
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,
我是一名优秀的程序员,十分优秀!