类型错误:无法将字节与整数连接

1 投票
2 回答
22011 浏览
提问于 2025-04-18 11:24

我正在尝试对文件名进行哈希处理,然后把它保存到数据库里。我使用的是Flask和Python 3.4。

这是我用来上传的代码:

@app.route('/', methods=['GET', 'POST'])
def upload_pic():
    if request.method == 'POST':
        file = request.files['file']
        try:
            extension = file.filename.rsplit('.', 1)[1].lower()
        except IndexError as e:
            abort(404)
        if file and check_extension(extension):
            # Salt and hash the file contents
            filename = md5(file.read() + str(round(time.time() * 1000))).hexdigest() + '.' + extension
            file.seek(0)  # Move cursor back to beginning so we can write to disk
            file.save(os.path.join(app.config['UPLOAD_DIR'], filename))
            add_pic(filename)
            gen_thumbnail(filename)
            return redirect(url_for('show_pic', filename=filename))
        else:  # Bad file extension
            abort(404)
    else:
        return render_template('upload.html')

当我提交表单时,出现了这个错误信息。

Traceback (most recent call last):
  File "C:\Python34\lib\site-packages\flask\app.py", line 1836, in __call__
    return self.wsgi_app(environ, start_response)
  File "C:\Python34\lib\site-packages\flask\app.py", line 1820, in wsgi_app
    response = self.make_response(self.handle_exception(e))
  File "C:\Python34\lib\site-packages\flask\app.py", line 1403, in handle_exception
    reraise(exc_type, exc_value, tb)
  File "C:\Python34\lib\site-packages\flask\_compat.py", line 33, in reraise
    raise value
  File "C:\Python34\lib\site-packages\flask\app.py", line 1817, in wsgi_app
    response = self.full_dispatch_request()
  File "C:\Python34\lib\site-packages\flask\app.py", line 1477, in full_dispatch_request
    rv = self.handle_user_exception(e)
  File "C:\Python34\lib\site-packages\flask\app.py", line 1381, in handle_user_exception
    reraise(exc_type, exc_value, tb)
  File "C:\Python34\lib\site-packages\flask\_compat.py", line 33, in reraise
    raise value
  File "C:\Python34\lib\site-packages\flask\app.py", line 1475, in full_dispatch_request
    rv = self.dispatch_request()
  File "C:\Python34\lib\site-packages\flask\app.py", line 1461, in dispatch_request
    return self.view_functions[rule.endpoint](**req.view_args)
  File "C:\Users\Ajay\PycharmProjects\codehackr-upload\flaskgur.py", line 70, in upload_pic
    filename = md5(file.read() + (round(time.time() * 1000))).hexdigest() + '.' + extension
TypeError: can't concat bytes to int

我不知道出了什么问题,请帮我解答一下。

谢谢。

2 个回答

1

没错,正如julienc所说,问题出在试图把file.read()的结果(类型是bytes)和round(time.time() * 1000)的结果(类型是int)拼接在一起。

但是,不要按照其他回答中的做法去做!保持冷静,继续往下看...

如何在Python 3中创建字节(bytes)

# (1)
bytes(12)                == b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'

# (2)
bytes([12])              == b'\x0c'
bytes([444])             -> ValueError("bytes must be in range(0, 256)")
bytes([0x01, 0xBC])      == b'\x01\xbc'

# (3)
( 12).to_bytes(8, 'big') == b'\x00\x00\x00\x00\x00\x00\x00\x0c'
(444).to_bytes(8, 'big') == b'\x00\x00\x00\x00\x00\x00\x01\xbc'

# (4)
str(12).encode('utf8')   == b'12'

(1) 创建长度为n的空字节(全是零)

这通常不是你想要的。要小心!

在这个问题的背景下,md5(file.read() + bytes(round(time.time() * 1000)))会对文件内容进行哈希处理,然后再加上大约1.5万亿个空字节。技术上来说,这样做是可行的,但计算成本可真高啊 :-)

(2) 创建长度为n的字节,包含指定的值

每个源值必须在0到255之间,才能放进一个字节里。

当然,也可以用元组来写,但对于长度为1的情况就显得有点丑陋了:bytes((12,))

(3) 将整数转换为字节的原生/二进制表示

这就是最初问题的答案,如何把一个整数转换成字节流等等。

不过现在你需要考虑一些底层的东西,比如输出的字节数和字节顺序(endianness)。想了解更多细节,可以查看int.to_bytes()的文档

当然,在这个例子中,额外的括号只在字面值12时需要。如果使用变量或其他表达式,就可以省略这些括号。

(4) 从指定文本(str)创建字节

这是上述方法的另一种选择,有时候你不需要或不想考虑底层的问题,这时可以通过简单地对值的文本表示进行哈希(或其他操作)来实现相同的目标。

当然,在这个例子中,可以用'ascii'或其他编码替代'utf8'。但为什么不训练一下肌肉记忆,使用在其他地方几乎通用的编码呢?

5

你正在尝试把file.read()的结果和round(time.time() * 1000)的结果连接在一起,而后者是一个整数。这就是你收到错误的原因:

TypeError: can't concat bytes to int

试着把这个整数转换一下:

file.read() + bytes(round(time.time() * 1000)))

不过正如@Lev Levitsky所指出的,你的代码似乎已经有了这个修正:你确定是在运行你代码的最新版本吗?

撰写回答