使用 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.

备注:

  1. 传递 SQL 命令字符串时,应将其包装在 SQLAlchemy text() 对象中。
  2. SQL 服务器存储过程(和匿名代码块)在绝大多数情况下应该以 SET NOCOUNT ON; 开头。如果不这样做,可能会导致合法结果或错误“滞后于”DML 语句(如 INSERTUPDATEDELETE.[=27)可能发出的任何行计数=]