番外:跨目录导入模块

关键词: 导入模块 模块

知识的领域是无限的,我们的学习也是无限期的。这篇文章主要讲述相关的知识,希望能为你提供帮助。

进击のpython


包&跨模块代码调用



  • 当你的模块文件越来越多,就需要对模块文件进行划分,比如把负责跟数据库交互的都放一个文件夹,把与页面交互相关的放一个文件夹,

    包(Package)

    1. my_proj/
    2. ├── apeland_web #代码目录
    3. ├── __init__.py
    4. ├── admin.py
    5. ├── apps.py
    6. ├── models.py
    7. ├── tests.py
    8. └── views.py
    9. ├── manage.py
    10. └── my_proj #配置文件目录
    11. ├── __init__.py
    12. ├── settings.py
    13. ├── urls.py
    14. └── wsgi.py

    像上面这样,一个文件夹管理多个模块文件,这个文件夹就被称为包
    一个包就是一个文件夹,但该文件夹下必须存在 init.py文件, 该文件的内容可以为空, init.py用于标识当前文件夹是一个包。
    这个 init.py的文件主要是用来对包进行一些初始化的,当当前这个package被别的程序调用时 init.py文件会先执行,一般为空, 一些你希望只要package被调用就立刻执行的代码可以放在 init.py里,一会后面会演示。


  • 目录结构如下

    跨模块导入

    1. my_proj
    2. ├── apeland_web
    3. ├── __init__.py
    4. ├── admin.py
    5. ├── apps.py
    6. ├── models.py
    7. ├── tests.py
    8. └── views.py
    9. ├── manage.py
    10. └── my_proj
    11. ├── settings.py
    12. ├── urls.py
    13. └── wsgi.py

    根据上面的结构,我怎么样可以在apeland_web/views.py里导入my_proj/settings.py模块呢?
    直接导入????
    跨目录导入模块 - 图1
    是会报错的,因为这两个模块,不在同一路径,所以找不到这个settings模块
    my_proj/settings.py 相当于是apeland_web/views.py的父亲(apeland_web)的兄弟(my_proj)的儿子(settings.py),settings.py算是views.py的表弟啦
    在views.py里只能导入同级别兄弟模块代码,或者子级别包里的模块,根本不知道表弟表哥的存在。这可怎么办呢?
    添加环境变量,把父亲级的路径添加到sys.path中,就可以了,这样导入就相当于从父亲级开始找模块了。
    跨目录导入模块 - 图2
    (标红是因为解释器的问题,不用管!强迫症去死!)
    那么问题来了!
    这是我的电脑,这是我的文件存放路径
    那到你电脑上,还是D盘????那不一定吧
    那怎么才能自动配置呢???
    还记得我们的os模块吗?
    是不是说用哪个记哪个?
    现在开始了奥!
    跨目录导入模块 - 图3
    这个方法,可以获取当前文件的相对路径
    但是为什么打印的是绝对路径呢?
    这是因为pycharm是从底层开始调用的,所以没有打印相对路径
    正常是相对路径:my_proj1/apeland_web/views.py
    然后我们只想要前面的my_proj1
    怎么做呢??????????
    利用os模块
    去掉了一层!但是我只想要my_proj1,怎么办??
    再来一遍呗!
    目的达到了吧!那我们怎么把它变成绝对路径呢???
    这样,因为你用的是动态的路径拼接,所以,即使放到你的电脑上
    这段程序也是能够使用的!


虽然通过添加环境变量的方式可以实现跨模块导入
但是官方不推荐这么干
因为这样就需要在每个目录下的每个程序里都写一遍添加环境变量的代码
官方推荐的玩法是,在项目里创建个入口程序,整个程序调用的开始应该是从入口程序发起,这个入口程序一般放在项目的顶级目录
这样做的好处是,项目中的二级目录 apeland_web/views.py中再调用他表亲my_proj/settings.py时就不用再添加环境变量了
原因是由于manage.py在顶层,manage.py启动时项目的环境变量路径就会自动变成….xxx/my_proj/这一级别
跨目录导入模块 - 图4
这样,就十分的简单啦!
主要就是因为当你程序执行的时候
就默认以你程序执行文件的位置作为起点
所以才可以这么玩!

官方推荐的跨目录导入方法

在项目根目录创建个入口程序,整个项目调用的开始应该是从入口程序发现。
在项目根目录创建个入口程序,整个项目调用的开始应该是从入口程序发现。
在项目根目录创建个入口程序,整个项目调用的开始应该是从入口程序发现。

这样做的好处是,在项目中的二级目录的模块中再调用其他二级目录的模块时就不用再添加环境变量了。

跨目录导入模块 - 图5


参考

https://www.yinxiang.com/everhub/note/13862e92-dfaa-4837-adc1-01c0e08b4d53

https://groups.google.com/g/python-cn/c/AmyjTNc5FkE django多次初始化的问题