16

出于某种奇怪的原因,我无法从 Python 测试应用程序中的 callproc 调用中获得结果。MqSQL 5.2.47 中的存储过程如下所示:

CREATE PROCEDURE `mytestdb`.`getperson` (IN personid INT)
BEGIN
   select person.person_id,
          person.person_fname,
          person.person_mi,
          person.person_lname,
          person.persongender_id,
          person.personjob_id
     from person
    where person.person_id = personid;
END

现在,将 PyCharm 与 Python 3.3 一起使用,调用此存储过程时我似乎无法检索任何内容。这段代码让我得到了想要的结果:

import mysql.connector

cnx = mysql.connector.connect(user='root', host='127.0.0.1', database='mytestdb')
cnx._open_connection()
cursor = cnx.cursor()

cursor.execute("select * from person where person.person_id = 1")
people = cursor.fetchall()

for person in people:
    print(person)

cnx.close()

但是这段代码使用 cursor.fetchall() 或 cursor.fetchone()...

import mysql.connector

cnx = mysql.connector.connect(user='root', host='127.0.0.1', database='mytestdb')
cnx._open_connection()
cursor = cnx.cursor()

cursor.callproc("getperson", [1])
people = cursor.fetchall()

for person in people:
    print(person)

cnx.close()

...返回“mysql.connector.errors.InterfaceError:没有要从中获取的结果集。” 像这样使用 cursor.execute() 方法还有一个额外的奇怪行为......

import mysql.connector

cnx = mysql.connector.connect(user='root', host='127.0.0.1', database='mytestdb')
cnx._open_connection()
cursor = cnx.cursor()

cursor.execute("call getperson(1)")
people = cursor.fetchall()

for person in people:
    print(person)

cnx.close()

...因为它产生“mysql.connector.errors.InterfaceError: Use cmd_query_iter for statements with multiple queries”后跟“mysql.connector.errors.InterfaceError: Use multi=True when execution multiple statements”尽管我是只返回一个查询结果而不是多个结果集。MySQL Python 连接器是否将存储过程上的执行调用视为双重查询?我怎样才能调用存储过程并取回我的结果?我真的不想在我的代码中使用动态 SQL。提前感谢您的任何建议!

4

3 回答 3

19

您是否尝试过选择其中一个结果集?

for result in cursor.stored_results():
    people = result.fetchall()

即使您只有一个SELECTstmt,它也可能分配给多个结果集。我知道在 PHP 的 MySQLi 存储过程中这样做是为了允许 INOUT 和 OUT 变量返回(同样,你没有,但也许它无论如何都在分配)。

我正在使用的完整代码(正在运行)是:

import mysql.connector

cnx = mysql.connector.connect(user='me',password='pw',host='localhost',database='mydb')
cnx._open_connection()
cursor = cnx.cursor()

cursor.callproc("getperson",[1])

for result in cursor.stored_results():
    people=result.fetchall()

for person in people:
    print person

cnx.close()
于 2013-03-10T08:36:52.640 回答
8

调用后获取存储过程的结果cursor.callproc取决于以下因素:

  • 调用过程的结果是否分配给 INOUT 或 OUT 参数
  • 结果是由单行还是一个结果集(或多个结果集)组成
  • 用于进行调用的 python 包

DBAPI规范有这样的说法cursor.callproc

调用具有给定名称的存储数据库过程。对于过程期望的每个参数,参数序列必须包含一个条目。调用的结果作为输入序列的修改副本返回。输入参数保持不变,输出和输入/输出参数替换为可能的新值。

该过程还可以提供结果集作为输出。然后必须通过标准的 .fetch*() 方法使其可用。

实际上,cursor.callproc只有当过程返回单行且列数与 INOUT 和 OUT 参数的数量相匹配时,才能使用 的返回值,因此处理结果的方式会有所不同。


以下是主要的 MySQL Python 连接器包 - MySQL Connectormysqlclient (MySQLdb)PyMySQL是如何处理这些情况的。

单行结果,通过 INOUT 或 OUT 参数返回

  • MySQL 连接器返回输入序列的修改副本作为cursor.callproc;的返回值 该值是一个元组。

    params = [in_param, out_param1, out_param2]
    in_, out1, out2 = cursor.callproc("test_proc", params) 
    
  • mysqlclientPyMySQL要求向数据库查询输出参数,然后通过游标获取结果;该值是元组的元组。要查询的参数名称的形式为'@_{procedure_name}_{params.index(param)}'

    cursor.callproc("test_proc", params)
    cursor.execute("""SELECT @_test_proc_0, @_test_proc_1""")
    result = cursor.fetchall()
    

单个结果集中的一行或多行,未定义 INOUT 或 OUT 参数

  • MySQL 连接器通过游标的stored_results方法公开结果(cursor.stored_results不是 DBAPI 规范的一部分)

    cursor.callproc("test_proc", params)
    results = [r.fetchall() for r in cursor.stored_results()]
    
  • mysqlclientPyMySQL通过游标的 fetch* 方法公开结果

    cursor.callproc("test_proc", params)
    results = cursor.fetchall()
    

多个结果集,未定义 INOUT 或 OUT 参数

  • MySQL 连接器stored_results通过游标的方法公开结果

    cursor.callproc("test_proc", params)
    results = [r.fetchall() for r in cursor.stored_results()]
    
  • mysqlclientPyMySQL要求通过游标获取每个结果集,同时调用cursor.nextset前进到下一个结果集。请注意,可能会返回一个额外的空结果集,这是调用过程的结果(如果结果集是通过cursor.nextset而不是仅调用cursor.fetchall一次来检索的,那么在前面的示例中也会发生这种情况)。

    cursor.callproc("test_proc", params)
    results = [cursor.fetchall()]
    while cursor.nextset():
        results.append(cursor.fetchall())
    

版本信息

$ mysql --version
mysql  Ver 15.1 Distrib 10.1.41-MariaDB, for debian-linux-gnu (x86_64) using readline 5.2

$ pip list | grep -i mysql
mysql-connector-python 8.0.18 
mysqlclient            1.4.6  
PyMySQL                0.9.3  
于 2019-12-31T15:53:26.857 回答
0

为什么不这样尝试

cursor.callproc("getperson", ['1'])
于 2013-03-10T08:22:58.180 回答