zoukankan      html  css  js  c++  java
  • python—模块和包

    模块的概念

    在计算机程序的开发过程中,随着程序代码越写越多,在一个文件里代码就会越来越长,越来越不容易维护。为了编写可维护的代码,我们把很多函数分组,分别放到不同的文件里,这样,每个文件包含的代码就相对较少,很多编程语言都采用这种组织代码的方式。在Python中,一个.py文件就称之为一个模块(Module)。

    模块一共三种:

    • python标准库
    • 第三方模块
    • 应用程序自定义模块

    另外,使用模块还可以避免函数名和变量名冲突。相同名字的函数和变量完全可以分别存在不同的模块中,因此,我们自己在编写模块时,不必考虑名字会与其他模块冲突。但是也要注意,尽量不要与内置函数名字冲突。

    模块:

      import:

          1 执行对应文件

          2 引入变量名

    1.模块的调用

      (1)import moudle_name1,moudle_name2...

    import cal
    
    print(cal.add(5,3))
    
    print(cal.sub(5,4))

                调用方式: moudle_name.函数名  如cal.add()

      (2)from moudle_name import name1,name2...    这个声明不会把整个moudle_name 模块导入当前的明明空间中,只会将里面的name1或name2单个引入到执行这个声明的的模块的全局符号表  

         或者 from moudle_name import *    (此处 * 代表全部函数) 但是此种方法不推荐使用

    from cal import add
    from cal import  sub
    
    print(add(5,3))        注意此处与(1)的区别
    print(sub(5,4))

        调用方式: 函数名   如add()    

     

        运行本质:无论(1)还是(2),首先通过sys.path找到test.py,然后执行test脚本(全部执行),区别是(1)会将test这个变量名加载到名字空间,而(2)只会将add这个变量名加载进来。  

      (3)from file_name import moudle_name

                                 

           运行 bin文件内的程序,会报错,原因在于 main文件中无法引入 cal 模块,这是因为解释器(pycharm)只会加载到执行文件所在的路径(运行bin文件 只会加载到 ‘模块’ 这一路径),因此并不能在

          bin文件中通过 import main来调用模块,因为解释器无法找到my_moudle 下的 main (此时解释器只认识 ‘模块’ 下的一级目录),此时就可以通过 from my_moudle import main 来调用 main,

          因为尽管解释器不认识main ,但是认识my_moudle, 可以通过此方法调用 my_moudle 下的模块。但是此程序依然是错误的,原因在于 main 文件 ,尽管 main 与 cal 在同级目录中,但此时main

          是引入文件,因此解释器在找 cal 时,仍然是在 ‘模块’ 这一层找( 解释器只认识执行文件(bin)这一层的文件)。因此,正确的程序应该把 main 中的import cal 改为 from moudle import cal。

          也可以说,方法(1)只能调用执行文件同级目录内的模块。

      (4)调用模块 补充:

        若存在以下情况

              

        在 bin文件中调用 my_module 下的 main ,因为运行bin 时,会将执行目录添加到环境变量中,此时目录为 F:\pycharm\Pycharm练习\day22_模块2\bin ,因此找不到上一级的my_module,

        (day22_模块才能找到mny_module)也就是说,在这种情况下,无法通过 from my_module import main 来调用main

        解决方法  通过 os.path.dirname() 方法来实现

                      os.path.dirname() 的作用

    import sys,os
    
    print(sys.path)
    print(os.path.dirname(os.path.dirname(__file__)))
    print(__file__)
    运行结果:
    ['F:\pycharm\Pycharm练习\day22_模块2\bin', 'F:\pycharm\Pycharm练习', 'F:\python\python36.zip', 'F:\python\DLLs', 'F:\python\lib', 'F:\python', 'F:\python\lib\site-packages']
    F:/pycharm/Pycharm练习/day22_模块2
    F:/pycharm/Pycharm练习/day22_模块2/bin/bin.py

               可以看出os.path.diename() 的作用是返回文件目录的上一层目录,因此,可以用两次该方法 即os.path.dirname(os.path.diename(__file__)) 来获取 F:/pycharm/Pycharm练习/day22_模块2 ,

                  sys.path 的作用:以列表的形式返回当前文件的执行目录

               在将其添加到环境变量中即可通过 from my_module import main  调用main

    import sys,os
    
    
    BASE_DIR = os.path.dirname(os.path.dirname(__file__))  #
    sys.path.append(BASE_DIR)
    
    from my_module import main
    
    if __name__ == '__main__':
        print(main.run())

              此程序仍然存在问题,原因在于 __file__ 事实上知识返回了文件名(bin.py),而pycharm 自动进行了拼接,因此此程序在终端上运行时会报错

              正确处理方法:

    import sys,os
    
    BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
    
    sys.path.append(BASE_DIR)
    
    from my_module import main
    
    if __name__ == '__main__':
        print(main.run())

            os.path.abspath(__file__) 返回文件的绝对路径

            sys.path.append()将括号内的目录添加到环境变量中

                

    包:

      如果不同的人编写的模块名相同怎么办?为了避免模块名冲突,Python又引入了按目录来组织模块的方法,称为包(Package)。举个例子,一个abc.py的文件就是一个名字叫abc的模块,

      一个xyz.py的文 件就是一个名字叫xyz的模块。现在,假设我们的abcxyz这两个模块名字与其他模块冲突了,于是我们可以通过包来组织模块,避免冲突。方法是选择一个顶层包名:

                                                                                       

      引入了包以后,只要顶层的包名不与别人冲突,那所有模块都不会与别人冲突。现在,view.py模块的名字就变成了hello_django.app01.views,类似的,manage.py的模块名则是hello_django.manage。

      请注意,每一个包目录下面都会有一个__init__.py的文件,这个文件是必须存在的,否则,Python就把这个目录当成普通目录(文件夹),而不是一个包。__init__.py可以是空文件,也可以有Python代

      码,因为__init__.py本身就是一个模块,而它的模块名就是对应包的名字

     

      (1)调用包内的模块

    # ------------方法一----------------from web.web1.web2.web3 import cal
    
    
    print(cal.add(5,3))
    
    # ------------------方法二-------------
    from web.web1.web2.web3.cal import add
    
    print(add(5,3))

    __name__  的作用

               如果我们是直接执行某个.py文件的时候,该文件中那么”__name__ == '__main__'“是True,但是我们如果从另外一个.py文件通过import导入该文件的时候,这时__name__的值就是我们这个py文件的名字而不是__main__。

              这个功能还有一个用处:调试代码的时候,在”if __name__ == '__main__'“中加入一些我们的调试代码,我们可以让外部模块调用的时候不执行我们的调试代码,但是如果我们想排查问题的时候,直接执行该模块文件,调试代码能够正常运行

  • 相关阅读:
    Begin Example with Override Encoded SOAP XML Serialization
    State Machine Terminology
    How to: Specify an Alternate Element Name for an XML Stream
    How to: Publish Metadata for a WCF Service.(What is the Metadata Exchange Endpoint purpose.)
    Beginning Guide With Controlling XML Serialization Using Attributes(XmlSerializaiton of Array)
    Workflow 4.0 Hosting Extensions
    What can we do in the CacheMetaData Method of Activity
    How and Why to use the System.servicemodel.MessageParameterAttribute in WCF
    How to: Begin Sample with Serialization and Deserialization an Object
    A Test WCF Service without anything of config.
  • 原文地址:https://www.cnblogs.com/jgua/p/13636234.html
Copyright © 2011-2022 走看看