8

MySQLdb を使用して mysql データベースと通信しており、すべての結果セットを動的に取得できます。

私の問題は、結果セットを取得すると、mysql でタイムスタンプとして宣言されている列がいくつかあるが、取得すると None になることです。

2 つの列があり、どちらもタイムスタンプが宣言されていますが、一方は正しいデータを返し、他方は None を返します。utime と enddate は両方ともタイムスタンプとして宣言されていますが、utime は正しく返されませんが、enddate は返されます。

['utime', 'userstr', 'vstr_client', 'enddate']

((None, '000102030ff43260gg0809000000000004', '7.7.0', '1970-01-01 12:00:00.000000'))

def parse_data_and_description(cursor, data):

    res = []
    cols = [d[0] for d in cursor.description]
    print cols
    print data

    for i in data:
        res.append(OrderedDict(zip(cols, i)))
    return res

def call_multi_rs(sp, args):

    rs_id=0;
    conn = connect()
    cursor = conn.cursor()
    try:
        conn.autocommit(True)
        cursor.execute ("CALL %s%s" % (sp, args))
        while True:
            rs_id+=1
            data = cursor.fetchone( )
            listout = parse_data_and_description(cursor, data)
            print listout
            if cursor.nextset( )==None:
            # This means no more recordsets available
            break
4

2 に答える 2

8

最後に、誰も答えなかったり、より多くの情報を見つけようとしたりした後、さらに解決策を探したところ、MySQLdb ライブラリがデータ型を sql から python に変換し、タイムスタンプを変換しないバグがあることがわかりました。

なぜ一方が変換され、もう一方が変換されないのかはまだわかりません。誰かがそれを理解できる場合は、これを更新してください。

ただし、mysql データベースに接続するときに行う必要がある変更は次のとおりです。MySQLdb は Python 日時オブジェクトをシリアライズできません

try:
    import MySQLdb.converters
except ImportError:
    _connarg('conv')

def connect(host='abc.dev.local', user='abc', passwd='def', db='myabc', port=3306):

    try:
        orig_conv = MySQLdb.converters.conversions
        conv_iter = iter(orig_conv)
        convert = dict(zip(conv_iter, [str,] * len(orig_conv.keys())))
        print "Connecting host=%s user=%s db=%s port=%d" % (host, user, db, port)
        conn = MySQLdb.connect(host, user, passwd, db, port, conv=convert)
    except MySQLdb.Error, e:
        print "Error connecting %d: %s" % (e.args[0], e.args[1])
    return conn
于 2013-03-14T23:34:56.403 に答える
3

私は同じ問題に出くわしました: DATETIME(1)-type returnsのデータを取得していますNone

一部の調査により、MySQLdb-Bug #325が発生しました。そのバグトラッカーによると、問題はまだ未解決です (2 年以上経過しています) が、コメントは有効な解決策を提供します。

MySQLdb パッケージの times.py で、次のようにマイクロ秒を処理するためにいくつかの行を挿入する必要があります。

def DateTime_or_None(s):
    if ' ' in s:
       sep = ' '
    elif 'T' in s:
        sep = 'T'
    else:
        return Date_or_None(s)

    try:
        d, t = s.split(sep, 1)
        if '.' in t:
            t, ms = t.split('.',1)
            ms = ms.ljust(6, '0')
        else:
            ms = 0
        return datetime(*[ int(x) for x in d.split('-')+t.split(':')+[ms] ])
    except (SystemExit, KeyboardInterrupt):
        raise
    except:
        return Date_or_None(s)

def TimeDelta_or_None(s):
    try:
        h, m, s = s.split(':')
        if '.' in s:
            s, ms = s.split('.')
            ms = ms.ljust(6, '0')
        else:
            ms = 0
        h, m, s, ms = int(h), int(m), int(s), int(ms)
        td = timedelta(hours=abs(h), minutes=m, seconds=s,
                       microseconds=ms)
        if h < 0:
            return -td
        else:
            return td
    except ValueError:
        # unpacking or int/float conversion failed
        return None

def Time_or_None(s):
    try:
        h, m, s = s.split(':')
        if '.' in s:
            s, ms = s.split('.')
            ms = ms.ljust(6, '0')
        else:
            ms = 0
        h, m, s, ms = int(h), int(m), int(s), int(ms)
        return time(hour=h, minute=m, second=s, microsecond=ms)
    except ValueError:
        return None

ただし、説明できないのは、元のクエリが一方の列で機能していて、もう一方の列では機能していないことです..おそらく、2番目の列にはマイクロ秒の情報が含まれていませんか?

于 2016-09-06T07:36:34.737 に答える