python: BytesIO 中 read 用法


最近在用 Flask 寫一個項目,后台管理用的插件暫時是 flask-admin。想實現的效果:在后台管理頁面中,把提交到后端的圖片不保存在 static 文件夾下面,而是通過后端代碼把這個文件對象上傳到 AWS 的 S3中存儲。

通過flask-admin 上傳到后端的文件對象的類型是:

FileStorage    # werkzeug.datastructures.FileStorage
# flask 中的 request.files 獲取到的類型也是 FileStorage

 

所以先從提交到后端的 form 表單中獲取到該文件對象,例如為: img_obj。 現在剛需要把類型為  FileStorage 的 img_obj 轉化為 file-like object (AWS S3  boto3 中的 upload_fileobj 接口需要這樣的參數)。轉化的過程用到了 shutil 的copyfileobj 和 BytesIO, 如下:

from shutil import copyfileobj

temp_file = BytesIO()

copyfileobj(img_obj.stream, temp_file)    # img_obj.stream 應該是能獲取到對象中的數據流; 然后把 imb_obj 中的數據流copy到 temp_file 中

 

然后,問題來了。 利用下面的 S3 upload_fileobj接口把文件上傳到 S3后,對應的文件一直都是 0 比特。代碼如下:

from shutil import copyfileobj

temp_file = BytesIO()

copyfileobj(img_obj.stream, temp_file)

client.upload_fileobj(temp_file, "bucket-name", Key="static/%s" % img_obj.filename)        # 利用這個接口把文件上傳到服務器后一直都是0比特

 

蛋疼。。。

查詢資料發現原因。

我們先來看下 shutil.copyfileobj 的源碼:

def copyfileobj(fsrc, fdst, length=16*1024):
    """copy data from file-like object fsrc to file-like object fdst"""
    while 1:
        buf = fsrc.read(length)
        if not buf:
            break
        fdst.write(buf)

"""
從上述代碼的最后一行看,fdst.write(buf) ,此時寫“文件”的游標已經到“文件”的最后
"""

 

我們再來看下面有關 BytesIO 的的一些用法:

In [1]: from io import BytesIO                                                                                                                                

In [2]: f = BytesIO()                                                                                                                                         

In [3]: f.write(b'abc')         # 把byte 寫入到 f 中,此時 游標已經到f的最后位置                                                                                                                             
Out[3]: 3

In [4]: f.read()     # 由於此時游標是從f 的 最后的位置開始 read,那么后面的內容肯定是空                                                                                                                                        
Out[4]: b''

In [5]: f.tell()                                                                                                                                           
Out[5]: 3             # 說明游標是在f最后的位置 

In [6]: f.seek(0)        # 利用 seek(0) 把游標的位置放到f的 0 位置處                                                                                                                                
Out[6]: 0

In [7]: f.read()      # 此時再 read 就能看到全部內容                                                                                                                                        
Out[7]: b'abc'
 
"""
getvalue() 是獲取全部內容;
read() 是從游標的當前位置往后讀
"""

 

所以上面問題的原因也是:

"""
copyfileobj 中的 fdst.write(buf) 寫完后,此時游標在“文件”最后一個位置;而由於 S3 的 upload_fileobj 接口中的第一個參數是file-like object, 而且upload_fileobj會調用 這個 file-like object 的 read() 方法,read 出來的內容會上傳到 S3 上。 所以,解決辦法就是利用 seek(0) 把游標位置再次放到 0 處
""" 

正確代碼如下:

from shutil import copyfileobj

temp_file = BytesIO()

copyfileobj(img_obj.stream, temp_file)

temp_file.seek(0)    # 讓游標回到0處

client.upload_fileobj(temp_file, "bucket-name", Key="static/%s" % img_obj.filename)        

 

或者直接把利用 FileStorage 的 stream 屬性把文件上傳到 S3,代碼如下:

client.upload_fileobj(img_obj.stream, "bucket-name", Key="static/%s" % img_obj.filename)     

 

 

BytesIO 參考鏈接:https://www.thetopsites.net/article/53485708.shtml

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM