Django-10-導(dǎo)入導(dǎo)出功能優(yōu)化

當(dāng)我們需要用django的中文界面時(shí)粉渠,可以設(shè)置LANGUAGE_CODE = 'zh-hans'
在admin頁(yè)面需要展示列表字段為中文名時(shí),可以在模型中為各字段添加verbose_name派哲。
而在使用導(dǎo)出功能中發(fā)現(xiàn),導(dǎo)出的excel掺喻,csv表格中列名會(huì)默認(rèn)成模型中的英文字段名芭届。

為了更易讀,現(xiàn)在將表格中的列名改為字段的verbose_name感耙。

1. 導(dǎo)出表格中的列名顯示

首先喉脖,我們先在DeviceResource的init中新建一個(gè)全局字段vname_dict,獲取Device模型中的所有字段對(duì)應(yīng)的verbose_name抑月。

#myproject/admin.py
class DeviceResource(resources.ModelResource):
    ...
    def __init__(self):
        super(DeviceResource, self).__init__()
        # 獲取所以字段的verbose_name并存放在字典
        field_list = apps.get_model('deviceManager', 'Device')._meta.fields
        self.vname_dict = {}
        for i in field_list:
            self.vname_dict[i.name] = i.verbose_name

然后,查看resource.py源碼舆蝴,可以發(fā)現(xiàn)導(dǎo)出數(shù)據(jù)前谦絮,是通過get_export_fields方法獲取要導(dǎo)出的數(shù)據(jù),而該方法其實(shí)只是直接調(diào)用了get_fields方法:獲取數(shù)據(jù)并按export_order的順序排列洁仗,返回一個(gè)列表层皱。

# import-export/resource.py
...
def get_export_fields(self):
    return self.get_fields()

我們直接重寫get_export_fields方法,在這里修改導(dǎo)出數(shù)據(jù)的列名就可以了赠潦。

# myapp/admin.py
# 默認(rèn)導(dǎo)入導(dǎo)出field的column_name為字段的名稱叫胖,這里修改為字段的verbose_name
def get_export_fields(self):
    fields = self.get_fields()
    for field in fields:
        field_name = self.get_field_name(field)
        # 如果我們?cè)O(shè)置過verbose_name,則將column_name替換為verbose_name她奥。否則維持原有的字段名
        if field_name in self.vname_dict.keys():  
            field.column_name = self.vname_dict[field_name]
    return fields

再使用導(dǎo)入導(dǎo)出時(shí)瓮增,列名就變?yōu)槲覀優(yōu)槟P椭凶侄卧O(shè)置的verbose_name啦!

列名為verbose_name
2. 導(dǎo)出表格中的外鍵顯示

接著發(fā)現(xiàn)一個(gè)問題:我們導(dǎo)出的表格中哩俭,所有的外鍵字段會(huì)顯示外鍵的id绷跑,而不是我們想看到的文字。查看resource.py源文件凡资,發(fā)現(xiàn)有一個(gè)before_export方法砸捏,官方注釋寫著可以重寫這個(gè)方法去自定義一些導(dǎo)出前的操作。但是,我發(fā)現(xiàn)before_export中只能獲取到queryset垦藏,也是就從數(shù)據(jù)庫(kù)查詢后的一個(gè)列表梆暖,是不可更改的,我根本沒法在這里修改將要導(dǎo)出的數(shù)據(jù)掂骏。于是只能重寫export方法了轰驳。

首先,我們需要知道哪些字段是外鍵芭挽。在DeviceResource的init方法中再新建一個(gè)全局變量self.fkey』希現(xiàn)在,init變?yōu)椋?/p>

#myproject/admin.py
class DeviceResource(resources.ModelResource):
    ...
    def __init__(self):
        super(DeviceResource, self).__init__()
        #獲取deviceManager應(yīng)用下Device模型中的所有字段
        field_list = apps.get_model('deviceManager', 'Device')._meta.fields  
        self.vname_dict = {}
        self.fkey = []
        for i in field_list:
            self.vname_dict[i.name] = i.verbose_name    # 獲取所有字段的verbose_name并存放在字典
            if(isinstance(i, ForeignKey)):
                self.fkey.append(i.name)    # 獲取所有ForeignKey字段的name存放在列表

然后袜爪,把resource.py中的export方法全部拷貝過來(lái)后進(jìn)行修改蠕趁,如下。
備注: # ---------- #之間的是自己添加的代碼辛馆,其它為export原本的代碼俺陋。

#myproject/admin.py
class DeviceResource(resources.ModelResource):
    ...
    # 重載resources.py的export方法,修改將要導(dǎo)出的data的某些外鍵相關(guān)數(shù)據(jù)昙篙。默認(rèn)導(dǎo)出外鍵id腊状,這里修改為導(dǎo)出外鍵對(duì)應(yīng)的值
    def export(self, queryset=None, *args, **kwargs):
        self.before_export(queryset, *args, **kwargs)

        if queryset is None:
            queryset = self.get_queryset()
        headers = self.get_export_headers()
        data = tablib.Dataset(headers=headers)

        # --------------------- #
        # 獲取所有外鍵名稱在headers中的位置
        fk_index = {}
        for fk in self.fkey:
            fk_index[fk] = headers.index(self.vname_dict[fk])
        # --------------------- #

        if isinstance(queryset, QuerySet):
            # Iterate without the queryset cache, to avoid wasting memory when
            # exporting large datasets.
            iterable = queryset.iterator()
        else:
            iterable = queryset
        for obj in iterable:
            # --------------------- #
            # 獲取將要導(dǎo)出的源數(shù)據(jù),這里export_resource返回的是列表苔可,便于更改缴挖。替換到外鍵的值
            res = self.export_resource(obj)
            res[fk_index['area_company']] = Group.objects.get(id=res[fk_index['area_company']]).name
            res[fk_index['site']] = Site.objects.get(id=res[fk_index['site']]).site
            res[fk_index['device_type']] = DeviceType.objects.get(id=res[fk_index['device_type']]).type
            res[fk_index['device_model']] = Model.objects.get(id=res[fk_index['device_model']]).model
            res[fk_index['supplier']] = Supplier.objects.get(id=res[fk_index['supplier']]).supplier
            data.append(res)
            # --------------------- #
        self.after_export(queryset, data, *args, **kwargs)
        return data

這樣,導(dǎo)出表格中的外鍵字段就不會(huì)再顯示為id啦焚辅。對(duì)比如下:

修改前:

外鍵字段顯示的全是數(shù)字

修改后:

外鍵字段顯示為我們想看到的文字信息了
3. 導(dǎo)入時(shí)外鍵字段的轉(zhuǎn)換

對(duì)于外鍵的處理映屋,導(dǎo)入時(shí)存在同樣的問題。我們自己填寫表格時(shí)同蜻,肯定不會(huì)填寫這個(gè)值對(duì)應(yīng)在另一個(gè)數(shù)據(jù)表里的id棚点。所以執(zhí)行導(dǎo)入操作前,我們也需要將數(shù)據(jù)源中實(shí)際輸入的值轉(zhuǎn)換為對(duì)應(yīng)的外鍵id湾蔓。

查看resource.py源碼瘫析,發(fā)現(xiàn)我們可以直接重寫before_import方法去修改數(shù)據(jù)源dataset,代碼如下默责。

#myproject/admin.py
class DeviceResource(resources.ModelResource):
    ...
    # import相關(guān)贬循。將導(dǎo)入的表格中的一些文字通過數(shù)據(jù)庫(kù)查詢一些外鍵field對(duì)應(yīng)的id。也可通過設(shè)置use_transactions=True,重寫before_save_instance等方法
    def before_import(self, dataset, using_transactions, dry_run, **kwargs):
        dict = []
        id = Device.objects.latest('id').id+1
        for row in dataset.dict:
            tmp = OrderedDict()
            tmp['ID'] = id
            for item in row:
                if item == self.vname_dict['area_company']:
                    tmp[item] = Group.objects.get(name=row[self.vname_dict['area_company']]).id
                elif item == self.vname_dict['site']:
                    tmp[item] = Site.objects.get(site=row[self.vname_dict['site']]).id
                elif item == self.vname_dict['device_type']:
                    tmp[item] = DeviceType.objects.get(type=row[self.vname_dict['device_type']]).id
                elif item == self.vname_dict['device_model']:
                    tmp[item] = Model.objects.get(model=row[self.vname_dict['device_model']]).id
                elif item == self.vname_dict['supplier']:
                    tmp[item] = Supplier.objects.get(supplier=row[self.vname_dict['supplier']]).id
                else:
                    tmp[item] = row[item]
            id = id+1
            dict.append(tmp)
        dataset.dict = dict

修改前桃序,如果導(dǎo)入外鍵字段不是填int的表格甘有,會(huì)報(bào)錯(cuò)。例如葡缰,如果導(dǎo)入以下表格:

需導(dǎo)入的表格

會(huì)出現(xiàn)如下報(bào)錯(cuò):

導(dǎo)入報(bào)錯(cuò)

修改后:


修改后

導(dǎo)入導(dǎo)出功能的優(yōu)化就搞定啦亏掀!

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末忱反,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子滤愕,更是在濱河造成了極大的恐慌温算,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,204評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件间影,死亡現(xiàn)場(chǎng)離奇詭異注竿,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)魂贬,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,091評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門巩割,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人付燥,你說我怎么就攤上這事宣谈。” “怎么了键科?”我有些...
    開封第一講書人閱讀 164,548評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵闻丑,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我勋颖,道長(zhǎng)嗦嗡,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,657評(píng)論 1 293
  • 正文 為了忘掉前任饭玲,我火速辦了婚禮侥祭,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘茄厘。我一直安慰自己矮冬,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,689評(píng)論 6 392
  • 文/花漫 我一把揭開白布蚕断。 她就那樣靜靜地躺著,像睡著了一般入挣。 火紅的嫁衣襯著肌膚如雪亿乳。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,554評(píng)論 1 305
  • 那天径筏,我揣著相機(jī)與錄音葛假,去河邊找鬼。 笑死滋恬,一個(gè)胖子當(dāng)著我的面吹牛聊训,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播恢氯,決...
    沈念sama閱讀 40,302評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼带斑,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼鼓寺!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起勋磕,我...
    開封第一講書人閱讀 39,216評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤妈候,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后挂滓,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體苦银,經(jīng)...
    沈念sama閱讀 45,661評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,851評(píng)論 3 336
  • 正文 我和宋清朗相戀三年赶站,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了幔虏。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,977評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡贝椿,死狀恐怖想括,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情团秽,我是刑警寧澤主胧,帶...
    沈念sama閱讀 35,697評(píng)論 5 347
  • 正文 年R本政府宣布,位于F島的核電站习勤,受9級(jí)特大地震影響踪栋,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜图毕,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,306評(píng)論 3 330
  • 文/蒙蒙 一夷都、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧予颤,春花似錦囤官、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,898評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至驳庭,卻和暖如春刑顺,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背饲常。 一陣腳步聲響...
    開封第一講書人閱讀 33,019評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工蹲堂, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人贝淤。 一個(gè)月前我還...
    沈念sama閱讀 48,138評(píng)論 3 370
  • 正文 我出身青樓柒竞,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親播聪。 傳聞我的和親對(duì)象是個(gè)殘疾皇子朽基,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,927評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理布隔,服務(wù)發(fā)現(xiàn),斷路器踩晶,智...
    卡卡羅2017閱讀 134,657評(píng)論 18 139
  • 關(guān)于Mongodb的全面總結(jié) MongoDB的內(nèi)部構(gòu)造《MongoDB The Definitive Guide》...
    中v中閱讀 31,934評(píng)論 2 89
  • 深沉厚重是第一等資質(zhì)执泰,磊落豪雄是第二等資質(zhì),聰明才辯是第三等資質(zhì)渡蜻。領(lǐng)導(dǎo)者首要資質(zhì)是具備時(shí)常深入思考事務(wù)本質(zhì)的厚...
    徐曉琳111閱讀 116評(píng)論 0 1
  • 2017.11.22 文/琴音 今天是一個(gè)好日子术吝。小櫻桃奶奶的生日,又是小櫻桃代表班級(jí)參...
    王燕惠閱讀 523評(píng)論 1 2
  • 我們后天學(xué)習(xí)的東西茸苇,都是理性排苍,理性是把人往回拉的力量。但是驅(qū)動(dòng)一個(gè)人的学密,其實(shí)是他的內(nèi)在感受淘衙,他的情緒。 知識(shí)的調(diào)用...
    ProblemSolver閱讀 240評(píng)論 0 0