FreeTDS-unixODBC-MSSQL2012
以下形态:
/etc/odbcinst.ini等

[FreeTDS]
Description = FreeTDS Driver
Driver = /usr/local/freetds/lib/libtdsodbc.so
Setup = /usr/lib64/libtdsS.so.2
FileUsage = 1
CPTimeout = 5
CRReuse = 5

/等/odbc.ini
[MSSQLTEST]
driver=FreeTDS
server=10.10.1.16
port=1433
database=ACCOUNT
client_charset = UTF-8
tds_version = 8.0

/etc/freetds.conf文件
[MSSQLTEST]
host = 10.10.1.16
port = 1433
tds version = 8.0

我用pyodbc连接MSSQL2012
import sys
reload(sys)
sys.setdefaultencoding('utf-8')

import pyodbc
conn =  pyodbc.connect("DSN=MSSQLTEST;UID=sa;PWD=xxxxx;database=PAYON;CHARSET=UTF8")
cursor = conn.cursor()
self.cursor.execute("{call insert_name('안녕')}")

结果:MSSQL2012中的name列为空值,而不是错误值。
self.cursor.execute("{call insert_name('123')}")

结果:MSSQL2012中的name列为123。
名字是nvarchar(50)。我认为这是字符集/编码问题。
Linux是cenos6.5,$LANG是ko_KR.UTF-8。
但我不知道问题的关键。
附加信息
编码问题?mssql设置?
我在下面写代码:
# -*- coding:utf-8 -*-

import sys
reload(sys)
sys.setdefaultencoding('utf-8')


import pyodbc
conn = pyodbc.connect("DSN=MSSQLTEST;UID=sa;PWD=xxxxx;database=PAYON")
cursor = conn.cursor()

cursor.execute("insert into tpayon_test (name) values(N'안녕')")
cursor.commit()
conn.close()

和unixODBC日志:
[ODBC][28783][1414141316.359552][__handles.c][450]
    Exit:[SQL_SUCCESS]
        Environment = 0x1987c70
[ODBC][28783][1414141316.359682][SQLSetEnvAttr.c][182]
    Entry:
        Environment = 0x1987c70
        Attribute = SQL_ATTR_ODBC_VERSION
        Value = 0x3
        StrLen = 4
[ODBC][28783][1414141316.359740][SQLSetEnvAttr.c][349]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.359792][SQLAllocHandle.c][364]
    Entry:
        Handle Type = 2
        Input Handle = 0x1987c70
[ODBC][28783][1414141316.359865][SQLAllocHandle.c][482]
    Exit:[SQL_SUCCESS]
        Output Handle = 0x19848f0
[ODBC][28783][1414141316.359930][SQLDriverConnectW.c][286]
    Entry:
        Connection = 0x19848f0
        Window Hdl = (nil)
        Str In = [DSN=MSSQLTEST;UID=sa;PWD=xxxxx;database=PAYON][length = 51]
        Str Out = (nil)
        Str Out Max = 0
        Str Out Ptr = (nil)
        Completion = 0
    UNICODE Using encoding ASCII 'ISO8859-1' and UNICODE 'UCS-2LE'

[ODBC][28783][1414141316.368017][SQLDriverConnectW.c][842]
    Exit:[SQL_SUCCESS]
        Connection Out [[NULL]]
[ODBC][28783][1414141316.368172][SQLSetConnectAttr.c][321]
    Entry:
        Connection = 0x19848f0
        Attribute = SQL_ATTR_AUTOCOMMIT
        Value = (nil)
        StrLen = -5
[ODBC][28783][1414141316.368872][SQLSetConnectAttr.c][675]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.368977][SQLGetInfo.c][546]
    Entry:
        Connection = 0x19848f0
        Info Type = SQL_DRIVER_ODBC_VER (77)
        Info Value = 0x7fffe24b61c0
        Buffer Length = 20
        StrLen = 0x7fffe24b61be
[ODBC][28783][1414141316.369032][SQLGetInfo.c][608]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.369121][SQLGetInfo.c][546]
    Entry:
        Connection = 0x19848f0
        Info Type = SQL_DESCRIBE_PARAMETER (10002)
        Info Value = 0x7fffe24b61a0
        Buffer Length = 2
        StrLen = 0x7fffe24b61be
[ODBC][28783][1414141316.369204][SQLGetInfo.c][608]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.369255][SQLGetInfo.c][546]
    Entry:
        Connection = 0x19848f0
        Info Type = SQL_NEED_LONG_DATA_LEN (111)
        Info Value = 0x7fffe24b61a0
        Buffer Length = 2
        StrLen = 0x7fffe24b61be
[ODBC][28783][1414141316.369304][SQLGetInfo.c][608]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.369347][SQLAllocHandle.c][529]
    Entry:
        Handle Type = 3
        Input Handle = 0x19848f0
[ODBC][28783][1414141316.369411][SQLAllocHandle.c][1064]
    Exit:[SQL_SUCCESS]
        Output Handle = 0x19db6d0
[ODBC][28783][1414141316.369463][SQLGetTypeInfo.c][164]
    Entry:
        Statement = 0x19db6d0
        Data Type = SQL_TYPE_TIMESTAMP
[ODBC][28783][1414141316.391748][SQLGetTypeInfo.c][314]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.391810][SQLFetch.c][158]
    Entry:
        Statement = 0x19db6d0
[ODBC][28783][1414141316.391864][SQLFetch.c][340]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.391910][SQLGetData.c][233]
    Entry:
        Statement = 0x19db6d0
        Column Number = 3
        Target Type = 4 SQL_INTEGER
        Buffer Length = 4
        Target Value = 0x7fffe24b61b8
        StrLen Or Ind = (nil)
[ODBC][28783][1414141316.391970][SQLGetData.c][497]
    Exit:[SQL_SUCCESS]
        Buffer = [23]
        Strlen Or Ind = NULLPTR
[ODBC][28783][1414141316.392015][SQLGetTypeInfo.c][164]
    Entry:
        Statement = 0x19db6d0
        Data Type = SQL_VARCHAR
[ODBC][28783][1414141316.392056][SQLGetTypeInfo.c][186]Error: 24000
[ODBC][28783][1414141316.392109][SQLGetTypeInfo.c][164]
    Entry:
        Statement = 0x19db6d0
        Data Type = Unknown(-9)
[ODBC][28783][1414141316.392196][SQLGetTypeInfo.c][186]Error: 24000
[ODBC][28783][1414141316.392275][SQLGetTypeInfo.c][164]
    Entry:
        Statement = 0x19db6d0
        Data Type = SQL_BINARY
[ODBC][28783][1414141316.392326][SQLGetTypeInfo.c][186]Error: 24000
[ODBC][28783][1414141316.392386][SQLFreeStmt.c][140]
    Entry:
        Statement = 0x19db6d0
        Option = 0
[ODBC][28783][1414141316.392542][SQLFreeStmt.c][246]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.392610][SQLAllocHandle.c][529]
    Entry:
        Handle Type = 3
        Input Handle = 0x19848f0
[ODBC][28783][1414141316.392665][SQLAllocHandle.c][1064]
    Exit:[SQL_SUCCESS]
        Output Handle = 0x1a057f0
[ODBC][28783][1414141316.392722][SQLFreeStmt.c][140]
    Entry:
        Statement = 0x1a057f0
        Option = 0
[ODBC][28783][1414141316.392767][SQLFreeStmt.c][246]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.392814][SQLExecDirect.c][236]
    Entry:
        Statement = 0x1a057f0
        SQL = [insert into tpayon_test (name) values(N'안녕')][length = 48 (SQL_NTS)]
[ODBC][28783][1414141316.394015][SQLExecDirect.c][499]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.394075][SQLRowCount.c][169]
    Entry:
        Statement = 0x1a057f0
        Row Count = 0x7fffe24b6300
[ODBC][28783][1414141316.394122][SQLRowCount.c][240]
    Exit:[SQL_SUCCESS]
        Row Count = 0x7fffe24b6300 -> 1
[ODBC][28783][1414141316.394169][SQLNumResultCols.c][152]
    Entry:
        Statement = 0x1a057f0
        Column Count = 0x7fffe24b631e
[ODBC][28783][1414141316.394214][SQLNumResultCols.c][244]
    Exit:[SQL_SUCCESS]
        Count = 0x7fffe24b631e -> 0
[ODBC][28783][1414141316.394265][SQLEndTran.c][318]
    Entry:
        Connection = 0x19848f0
        Completion Type = 0
[ODBC][28783][1414141316.394971][SQLGetInfo.c][546]
    Entry:
        Connection = 0x19848f0
        Info Type = SQL_CURSOR_COMMIT_BEHAVIOR (23)
        Info Value = 0x1985d60
        Buffer Length = 2
        StrLen = 0x7fffe24b621e
[ODBC][28783][1414141316.395160][SQLGetInfo.c][608]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.395211][SQLGetInfo.c][546]
    Entry:
        Connection = 0x19848f0
        Info Type = SQL_CURSOR_ROLLBACK_BEHAVIOR (24)
        Info Value = 0x1985d62
        Buffer Length = 2
        StrLen = 0x7fffe24b621e
[ODBC][28783][1414141316.395260][SQLGetInfo.c][608]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.395302][SQLEndTran.c][504]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.395349][SQLEndTran.c][318]
    Entry:
        Connection = 0x19848f0
        Completion Type = 1
[ODBC][28783][1414141316.408656][SQLEndTran.c][504]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.408716][SQLDisconnect.c][204]
    Entry:
        Connection = 0x19848f0
[ODBC][28783][1414141316.408893][SQLDisconnect.c][341]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.409055][SQLFreeHandle.c][279]
    Entry:
        Handle Type = 2
        Input Handle = 0x19848f0
[ODBC][28783][1414141316.409117][SQLFreeHandle.c][330]
    Exit:[SQL_SUCCESS]

最佳答案

我用pyodbc和FreeTDS在我的Xububtu 14.04框上做了一些测试,发现如果我使用参数化查询,我的代码可以正常工作,但是它不能处理文本查询。也就是说,使用与您相同的FreeTDS/ODBC配置,此方法失败(插入了不同的字符):

# -*- coding:utf-8 -*-
import pyodbc
conn = pyodbc.connect("DSN=MSSQLTEST;UID=sa;PWD=whatever;database=myDb")
cursor = conn.cursor()

cursor.execute(u"insert into tpayon_test (name) values (N'안녕')")

cursor.commit()
conn.close()

虽然这种方法很有效:
# -*- coding:utf-8 -*-
import pyodbc
conn = pyodbc.connect("DSN=MSSQLTEST;UID=sa;PWD=whatever;database=myDb")
cursor = conn.cursor()

sql = "insert into tpayon_test (name) values (?)"
parameters = [u"안녕"]
cursor.execute(sql, parameters)

cursor.commit()
conn.close()

同样,这(及其变体)也不起作用
cursor.execute(u"{call insert_name(N'안녕')}")

但事实上:
sql = "{call insert_name(?)}"
parameters = [u"안녕"]
cursor.execute(sql, parameters)

09-07 10:16