使用 Python 客户端在 SQL 服务器中执行存储过程
Executing stored procedures in SQL Server using Python client
我从 Whosebug 上的 了解到如何在 python (pyodbc) 中调用 SQL 服务器上的存储过程。将我的代码修改为以下内容后,我可以从我创建的 db_engine
连接 运行 execute()
。
import pyodbc
import sqlalchemy as sal
from sqlalchemy import create_engine
import pandas as pd
import urllib
params = urllib.parse.quote_plus(
'DRIVER={ODBC Driver 17 for SQL Server};'
f'SERVER=myserver.com;'
f'DATABASE=mydb;'
f'UID=foo;'
f'PWD=bar')
cobnnection_string = f'mssql+pyodbc:///?odbc_connect={params}'
db_engine = create_engine(connection_string)
db_engine.execute("EXEC [dbo].[appDoThis] 'MYDB';")
<sqlalchemy.engine.result.ResultProxy at 0x1121f55e0>
db_engine.execute("EXEC [dbo].[appDoThat];")
<sqlalchemy.engine.result.ResultProxy at 0x1121f5610>
然而,即使运行在Python中执行上述代码后没有返回任何错误,当我检查数据库时,我确认没有执行任何操作(更能说明问题的是上述命令需要一两秒才能完成,而 运行在数据库管理工具上成功地安装这些存储过程大约需要 5 分钟)。
为了正确调试,我应该如何理解上述设置中哪些部分工作不正常?我从字面上 运行 通过我的数据库管理工具完全相同的代码没有问题 - 存储过程按预期执行。什么可以通过 Python 阻止这种情况发生?执行的SQL需要commit吗?有没有办法使用返回的 ResultProxy
进行调试?如有任何建议,我们将不胜感激。
直接在 Engine
对象上调用 .execute()
是一种过时的使用模式,并且会从 SQLAlchemy 1.4 版开始发出弃用警告。如今,首选方法是使用使用 engine.begin()
:
的上下文管理器(with
块)
import sqlalchemy as sa
# …
with engine.begin() as conn: # transaction starts here
conn.execute(sa.text("EXEC [dbo].[appDoThis] 'MYDB';"))
# On exiting the `with` block the transaction will automatically be committed
# if no errors have occurred. If an error has occurred the transaction will
# automatically be rolled back.
备注:
- 传递 SQL 命令字符串时,应将其包装在 SQLAlchemy
text()
对象中。
- SQL 服务器存储过程(和匿名代码块)在绝大多数情况下应该以
SET NOCOUNT ON;
开头。如果不这样做,可能会导致合法结果或错误“滞后于”DML 语句(如 INSERT
、UPDATE
或 DELETE
.[=27)可能发出的任何行计数=]
我从 Whosebug 上的 db_engine
连接 运行 execute()
。
import pyodbc
import sqlalchemy as sal
from sqlalchemy import create_engine
import pandas as pd
import urllib
params = urllib.parse.quote_plus(
'DRIVER={ODBC Driver 17 for SQL Server};'
f'SERVER=myserver.com;'
f'DATABASE=mydb;'
f'UID=foo;'
f'PWD=bar')
cobnnection_string = f'mssql+pyodbc:///?odbc_connect={params}'
db_engine = create_engine(connection_string)
db_engine.execute("EXEC [dbo].[appDoThis] 'MYDB';")
<sqlalchemy.engine.result.ResultProxy at 0x1121f55e0>
db_engine.execute("EXEC [dbo].[appDoThat];")
<sqlalchemy.engine.result.ResultProxy at 0x1121f5610>
然而,即使运行在Python中执行上述代码后没有返回任何错误,当我检查数据库时,我确认没有执行任何操作(更能说明问题的是上述命令需要一两秒才能完成,而 运行在数据库管理工具上成功地安装这些存储过程大约需要 5 分钟)。
为了正确调试,我应该如何理解上述设置中哪些部分工作不正常?我从字面上 运行 通过我的数据库管理工具完全相同的代码没有问题 - 存储过程按预期执行。什么可以通过 Python 阻止这种情况发生?执行的SQL需要commit吗?有没有办法使用返回的 ResultProxy
进行调试?如有任何建议,我们将不胜感激。
直接在 Engine
对象上调用 .execute()
是一种过时的使用模式,并且会从 SQLAlchemy 1.4 版开始发出弃用警告。如今,首选方法是使用使用 engine.begin()
:
with
块)
import sqlalchemy as sa
# …
with engine.begin() as conn: # transaction starts here
conn.execute(sa.text("EXEC [dbo].[appDoThis] 'MYDB';"))
# On exiting the `with` block the transaction will automatically be committed
# if no errors have occurred. If an error has occurred the transaction will
# automatically be rolled back.
备注:
- 传递 SQL 命令字符串时,应将其包装在 SQLAlchemy
text()
对象中。 - SQL 服务器存储过程(和匿名代码块)在绝大多数情况下应该以
SET NOCOUNT ON;
开头。如果不这样做,可能会导致合法结果或错误“滞后于”DML 语句(如INSERT
、UPDATE
或DELETE
.[=27)可能发出的任何行计数=]