微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

FreeTDS ODBC + pyodbc 是否支持表值参数 (TVP)?

如何解决FreeTDS ODBC + pyodbc 是否支持表值参数 (TVP)?

我在 unix/rhel7 系统上工作。我已经为 FreeTDS、unixODBC 和 pyodbc 安装了 require 驱动程序。其他查询工作正常,但是当我尝试使用 TVP(表值参数)执行存储过程时,它给了我错误。有什么方法可以使用 python 中的 windows 服务帐户 连接 sql Server?

示例:

import pyodbc;

cnxn = pyodbc.connect('DRIVER=FreeTDS;SERVER=SERVERNAME;PORT=1234;UID=USERNAME;PWD=PASSWORD;DATABASE=dbnAME')
cnxn.cursor()


param_array = []
for i in range(3):
  param_array.append(['abc','adi','/somepath/','2021-01-04','NEW'])

result_array = cursor.execute("EXEC abc.stored_proc_name ?",[param_array]).fetchall()
cursor.commit()

cnxn.close()

错误

pyodbc.Error: ('HY004','[HY004] [FreeTDS][sql Server]无效数据类型 (0) (sqlBindParameter)')

那么有没有其他方法可以从支持 TVP 的 python 连接 sql 服务帐户?或者上面的例子有什么解决方案吗?

解决方法

FreeTDS ODBC 不直接支持表值参数 (TVP),如所讨论的 here。但是,我们可以使用临时表和匿名代码块来解决此问题。对于用户定义的表类型

USE [myDb]
GO

/****** Object:  UserDefinedTableType [dbo].[dboListInt]    Script Date: 2021-02-18 10:53:17 ******/
CREATE TYPE [dbo].[dboListInt] AS TABLE(
    [Id] [int] NOT NULL,PRIMARY KEY CLUSTERED 
(
    [Id] ASC
)WITH (IGNORE_DUP_KEY = OFF)
)
GO

以及接受该表类型的存储过程

USE [myDb]
GO
/****** Object:  StoredProcedure [dbo].[dboPyOdbcTestTvp]    Script Date: 2021-02-18 10:41:43 ******/
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
CREATE PROCEDURE [dbo].[dboPyOdbcTestTvp](@tvp [dbo].dboListInt READONLY)
AS
BEGIN
    SET NOCOUNT ON;

    SELECT * FROM @tvp

END

我们可以像这样调用存储过程并检索结果:

import pyodbc

cnxn = pyodbc.connect(
    "DRIVER=FreeTDS_1.2.18;"
    "SERVER=192.168.0.179;"
    "PORT=49242;"
    "DATABASE=myDb;"
    "UID=sa;PWD=_whatever_;"
)
crsr = cnxn.cursor()
crsr.execute("CREATE TABLE #tvp_data (Id int)")
tvp_data = [(123,),(234,(345,)]
crsr.executemany(
    "INSERT INTO #tvp_data (Id) VALUES (?)",tvp_data
)
crsr.execute("""\
SET NOCOUNT ON;
DECLARE @tvp dbo.dboListInt;
INSERT INTO @tvp (Id)
SELECT Id FROM #tvp_data;
EXEC dbo.dboPyOdbcTestTvp @tvp;
""")
print(crsr.fetchall())
# [(123,)]

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。