pyodbc - 从 MS Access (MDB) 数据库读取主键

pyodbc - read primary keys from MS Access (MDB) database

当我尝试使用 cursor.primaryKeys("tablename") 时发生异常:

Error: ('IM001', '[IM001] [Microsoft][ODBC Driver Manager] Driver does not support this function (0) (SQLPrimaryKeys)')

list(cursor.columns(table='tablename')) 也不显示主键。

这里是使用 pythonnet 和 Oledb Jet 驱动程序的解决方案。请注意,这不会将主键的顺序保留为列:

import clr
import System
import System.Data.OleDb
from System.Data.OleDb import OleDbSchemaGuid

def getKeyNames(tableName, mdbname):
    conn = System.Data.OleDb.OleDbConnection()
    conn.ConnectionString = ("Provider=Microsoft.Jet.OLEDB.4.0;"
                         "Data source={}".format(mdbname))
    conn.Open()
    returnList=[]
    mySchema = (conn).GetOleDbSchemaTable(OleDbSchemaGuid.Primary_Keys,
        [None, None, tableName])
    columnOrdinalForName = mySchema.Columns["COLUMN_NAME"].Ordinal
    for r in mySchema.Rows:
        returnList.append(r.ItemArray[columnOrdinalForName])
        conn.Close()
    return returnList

getKeyNames(table_name,mdbname)

对于 Access ODBC,我们通常可以* 通过 pyodbc cursor 对象的 .statistics() 方法获取主键列:

crsr = conn.cursor()
table_name = 'MyTable'
# dict comprehension: {ordinal_position: col_name}
pk_cols = {row[7]: row[8] for row in crsr.statistics(table_name) if row[5]=='PrimaryKey'}
print(pk_cols)  # e.g., {1: 'InvID', 2: 'LineItem'}

*编辑:此方法假定 table 的主键索引名为 PrimaryKey。如果 table 是使用 MS Access table 生成器 (GUI) 创建的,则为真,但如果 table 是使用 DDL 创建的,则 不是 (即 CREATE TABLE …)。在这些情况下,主键索引的名称将类似于 Index_EA5344E1_0942_445C,因此上述方法将不起作用,但我们可以改用 ACE DAO:

import win32com.client  # needs `pip install pywin32`


def get_access_primary_key_columns(db_path, table_name):
    db_engine = win32com.client.Dispatch("DAO.DBEngine.120")
    db = db_engine.OpenDatabase(db_path)
    tbd = db.TableDefs(table_name)
    for idx in tbd.Indexes:
        if idx.Primary:
            return [fld.Name for fld in idx.Fields]


if __name__ == "__main__":
    print(
        get_access_primary_key_columns(
            r"C:\Users\Public\Database1.accdb", "team"
        )
    )
    # ['city', 'prov']