解決使用Pandas 讀取超過65536行的Excel文件問題
場(chǎng)景
今天需要合并天貓訂單數(shù)據(jù),由于前期6.18活動(dòng)有很多數(shù)據(jù)需要處理,將幾個(gè)月份合并一起,結(jié)果報(bào)錯(cuò)。
問題分析
Excel 文件的格式曾經(jīng)發(fā)生過一次變化,在 Excel 2007 以前,使用擴(kuò)展名為 .xls 格式的文件,這種文件格式是一種特定的二進(jìn)制格式,最多支持 65,536 行,256 列表格。從 Excel 2007 版開始,默認(rèn)采用了基于 XML 的新的文件格式 .xlsx ,支持的表格行數(shù)達(dá)到了 1,048,576,列數(shù)達(dá)到了 16,384。需要注意的是,將 .xlsx 格式的文件轉(zhuǎn)換為 .xls 格式的文件時(shí),65536 行和 256 列之后的數(shù)據(jù)都會(huì)被丟棄。
Pandas 讀取 Excel 文件的引擎是 xlrd , xlrd 雖然同時(shí)支持 .xlsx 和 .xls 兩種文件格式,但是在源碼文件 xlrd/sheet.py 中限制了讀取的 Excel 文件行數(shù)必須小于 65536,列數(shù)必須小于 256。
if self.biff_version >= 80: self.utter_max_rows = 65536 else: self.utter_max_rows = 16384 self.utter_max_cols = 256
這就導(dǎo)致,即使是 .xlsx 格式的文件, xlrd 依然不支持讀取 65536 行以上的 Excel 文件(源碼中還有一個(gè)行數(shù)限制是 16384,這是因?yàn)?Excel 95 時(shí)代, xls 文件所支持的最大行數(shù)是 16384)。
解決辦法
openpyxl 是一個(gè)專門用來操作 .xlsx 格式文件的 Python 庫(kù),和 xlrd 相比它對(duì)于最大行列數(shù)的支持和 .xlsx 文件所定義的最大行列數(shù)一致。
首先安裝 openpyxl :
pip install openpyxl
Pandas 的 read_excel 方法中,有 engine 字段,可以指定所使用的處理 Excel 文件的引擎,填入 openpyxl ,再讀取文件就可以了。
import os import pandas as pd # 將文件讀取出來放一個(gè)列表里面 pwd = '1' # 獲取文件目錄 # 新建列表,存放文件名 file_list = [] # 新建列表存放每個(gè)文件數(shù)據(jù)(依次讀取多個(gè)相同結(jié)構(gòu)的Excel文件并創(chuàng)建DataFrame) dfs = [] for root,dirs,files in os.walk(pwd): # 第一個(gè)為起始路徑,第二個(gè)為起始路徑下的文件夾,第三個(gè)是起始路徑下的文件。 for file in files: file_path = os.path.join(root, file) file_list.append(file_path) # 使用os.path.join(dirpath, name)得到全路徑 df = pd.read_excel(file_path) # 導(dǎo)入xlsx文件,將excel轉(zhuǎn)換成DataFrame dfs.append(df) # 將多個(gè)DataFrame合并為一個(gè) df = pd.concat(dfs) # 數(shù)據(jù)輸出,寫入excel文件,不包含索引數(shù)據(jù) # 數(shù)據(jù)寫入 Excel,需要首先安裝一個(gè) engine,由 engine 負(fù)責(zé)將數(shù)據(jù)寫入 Excel,pandas 使用 openpyx 或 xlsxwriter 作為寫入引擎。 df.to_excel('test\\1.xlsx', index=False,engine='openpyxl') # 導(dǎo)出 Excel,一般不需要索引,將 index 參數(shù)設(shè)為 False
補(bǔ)充知識(shí):python使用xlrd讀取excel數(shù)據(jù)作為requests的請(qǐng)求參數(shù),并把返回的數(shù)據(jù)寫入excel中
實(shí)現(xiàn)功能:
從excel中的第一列數(shù)據(jù)作為post請(qǐng)求的數(shù)據(jù),數(shù)據(jù)為json格式;把post返回的結(jié)果寫入到excel的第二列數(shù)據(jù)中,并把返回?cái)?shù)據(jù)與excel中的預(yù)期結(jié)果做比較,如果與預(yù)期一致則在案例執(zhí)行結(jié)果中寫入成功,否則寫入失敗。
每一行的數(shù)據(jù)都不一樣,可實(shí)現(xiàn)循環(huán)調(diào)用
# !/usr/bin/env python # -*- coding:utf-8 -*- #import xlwt #這個(gè)專門用于寫入excel的庫(kù)沒有用到 import xlrd from xlutils.copy import copy import requests import json old_excel = xlrd.open_workbook('excel.xls') sheet = old_excel.sheets()[0] url = 'http://10.1.1.32:1380/service/allocFk2' headers = {'Content-Type': 'application/json'} i = 0 new_excel = copy(old_excel) for row in sheet.get_rows(): data = row[0].value response = requests.post(url=url, headers=headers, data=data) text = response.text #使用json.loads可以把Unicode類型,即json類型轉(zhuǎn)換成dict類型 text = json.loads(text)["returnMsg"] #屏蔽這行代碼即可把返回的完整數(shù)據(jù)寫入文件中 ws = new_excel.get_sheet(0) ws.write(i,1,text) new_excel.save('excel.xls') old_excel = xlrd.open_workbook('excel.xls') new_excel = copy(old_excel) i = i+1
執(zhí)行前的excel格式:
發(fā)送報(bào)文 | 返回報(bào)文 | 校驗(yàn)字符 | 案例執(zhí)行結(jié)果 |
{ "projectId" :"0070", "projectAllocBatch" :"1", "serviceCode" :"GT012", "seqNo" :"180800272201GT51286712", "tranTimeStamp" :"20180817102244", "sign" :"2dbb89a6bd86b2af1ff6a76c35c05284" } | 交易失敗 | ||
{ "projectId" :"0070", "projectAllocBatch" :"1", "serviceCode" :"GT012", "seqNo" :"180800272201GT51286713", "tranTimeStamp" :"20180817102244", "sign" :"2dbb89a6bd86b2af1ff6a76c35c05284" } | 交易失敗 | ||
{ "projectId" :"0070", "projectAllocBatch" :"1", "serviceCode" :"GT012", "seqNo" :"180800272201GT51286713", "tranTimeStamp" :"20180817102244", "sign" :"2dbb89a6bd86b2af1ff6a76c35c05284" } | 交易成功 |
執(zhí)行后的結(jié)果:
調(diào)試過程中遇到的問題:
1、一開始在for循環(huán)的最后沒有增加這兩行代碼
old_excel = xlrd.open_workbook('excel.xls')
new_excel = copy(old_excel)
這樣的話new_excel永遠(yuǎn)都是一開始獲取到的那一個(gè),只會(huì)把最后一個(gè)循環(huán)返回的結(jié)果寫入文件,因?yàn)橹暗娜慷急灰婚_始獲取的那個(gè)old_excel給覆蓋了,所以每次執(zhí)行完寫入操作以后都要重新做一次copy操作,這樣就能保證new_excel是最新的。
2、注意執(zhí)行程序之前要把excel關(guān)閉,否則會(huì)報(bào)錯(cuò)
以上這篇解決使用Pandas 讀取超過65536行的Excel文件問題就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
celery實(shí)現(xiàn)動(dòng)態(tài)設(shè)置定時(shí)任務(wù)
這篇文章主要為大家詳細(xì)介紹了celery實(shí)現(xiàn)動(dòng)態(tài)設(shè)置定時(shí)任務(wù),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-03-03Python實(shí)現(xiàn)Pig Latin小游戲?qū)嵗a
這篇文章主要介紹了Python實(shí)現(xiàn)Pig Latin小游戲?qū)嵗a,分享了相關(guān)代碼示例,小編覺得還是挺不錯(cuò)的,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-02-02在pycharm中創(chuàng)建django項(xiàng)目的示例代碼
這篇文章主要介紹了在pycharm中創(chuàng)建django項(xiàng)目的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05使用Python實(shí)現(xiàn)批量修改文件的修改日期功能
在日常的文件管理中,您可能需要批量修改文件的修改日期,比如,您可能希望將某個(gè)文件夾中的所有文件的修改日期隨機(jī)設(shè)置為6到8月份之間的日期,這在數(shù)據(jù)整理中可能非常有用,本文將詳細(xì)介紹如何使用Python實(shí)現(xiàn)這一功能,需要的朋友可以參考下2024-10-10pandas參數(shù)設(shè)置的實(shí)用小技巧
這篇文章主要給大家介紹了關(guān)于pandas參數(shù)設(shè)置的實(shí)用小技巧,文中通過實(shí)例代碼結(jié)束的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用pandas具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08