python:關於py文件之間相互import的問題


問題背景

調試腳本時,遇到一個問題:ImportError: cannot import name 'A' from 'study_case.a' (/Users/rchera/PycharmProjects/test/study_case/a.py)

具體情況是這樣嬸兒的:

前些日子寫了一個py文件,它的功能主要是創建數據(暫且稱為create_data.py,每條數據會生成一個唯一的id);

同時寫了另一個py文件,它的功能主要是操作數據,例如對數據進行編輯、刪除等(暫且稱為operate_data.py);

要對數據進行操作的話,必須傳入數據對應的唯一id,所以在operate_data.py導入create_data.py生成數據的方法

至此,一切都很順利,還沒有什么問題

 

由於一些情況,我需要在生成一條數據后,緊接着要把它刪除,

開始打算直接在create_data.py中再寫一個刪除數據的方法,不過后來想到之前已經在operate_data.py中定義好刪除數據的方法了,

這樣的話,直接在create_data.py導入operate_data.py,引用其刪除數據的方法應該也行

 

但是實際卻並未按照預想的那樣,按照上面的想法調好腳本后,create_data.py和operate_data.py都無法運行了,會提示頂部列出的錯誤~

 

在網上搜索了一下,大致原因是:create_data.py和operate_data.py兩個文件互相調用了

當然並不是說python不能循環引用,只是我的寫法有問題,然后我繼續深入搜索了一番,在一些大神的博客里找到了解決方法

參考文章:

https://fishc.com.cn/forum.php?mod=viewthread&tid=60873&page=1#pid2337627

https://www.zhihu.com/question/19887316

https://blog.igevin.info/posts/how-to-avoid-python-circle-import-error/

ps.大神都建議不要在python中交叉循環引用(即不要在a中調用b,然后b中又調用a),最好規划好代碼的層級,哪些是公共方法,哪些是業務邏輯,把這些提前想好,有利於后期的維護~

 

OK,接下來按照網上給出的方法,用一個例子演示下如何解決python中循環引用的問題

實例演示

准備2個py文件

a.py

from study_case.b import B


class A:
    @staticmethod
    def a1(number):
        return number * number

    @staticmethod
    def a2():
        value = B().b(1)
        return value


if __name__ == '__main__':
    t = A()
    print(t.a1(2))

b.py

from study_case.a import A


class B:
    @staticmethod
    def b(x):
        value = A().a1(1)
        return value + x


if __name__ == '__main__':
    t = B()
    print(t.b(1))

文件結構如下

 

按照上面這種寫法,運行a或b,都會報錯,如下

 

 解決方法1: 在文件的頂部引入,不要用from,使用絕對引入

例如,修改b.py文件內容如下,直接引入a文件

import study_case.a


class B:
    @staticmethod
    def b(x):
        value = study_case.a.A().a1(1)   
        return value + x


if __name__ == '__main__':
    t = B()
    print(t.b(1))

不過我試了下,如果用from的話,只導入a文件這個層級,也不會報錯,如下

from study_case import a


class B:
    @staticmethod
    def b(x):
        value = a.A().a1(1)
        return value + x


if __name__ == '__main__':
    t = B()
    print(t.b(1))

 

解決方法2: 函數頂部引入,可以用from

例如,修改b.py文件內容如下,在函數名下方第一行引入a文件的類

class B:
    @staticmethod
    def b(x):
        from study_case.a import A
        value = A().a1(1)
        return value + x


if __name__ == '__main__':
    t = B()
    print(t.b(1))

 


免責聲明!

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



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