python模块化是什么_Python-模块化

1、模块化

一般来说,编程语言中,库,包,模块是同一个概念,是代码组织方式。

Python中,只有一种模块对象类型,但是为了模块化组织模块的遍历,提供了‘包’的概念。

模块 module,指的是Python的源代码文件。

包package,指的是模块组织在一起的和包同名的目录及其相关文件。

2、导入语句 :

2.1:import

import 模块1[, 模块2 ]  ------完全导入,当然EPE8  推荐多个模块分别导入

import... as....              -------模块别名

import语句:

1、找到指定的模块,加载和初始化它,生成模块对象,找不到,抛出importError异常

2、在import所在的作用域的局部命名空间中,增加名称和上一步创建的对象关联。

测试1:

1 print(dir())2 '''

3 ['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'partial']4

5 '''

6

7 importfunctools8 print(dir())9 print(functools)10 print(functools.wraps)11

12 '''

13 ['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'functools', 'partial']14 15 16 '''

17

18 importos.path19 print(dir())20 print(os)21 print(os.path)22

23 '''

24 ['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'os', 'partial']25 26 27 '''

28

29 importos.path as osp30 print(dir())31 print(osp)32

33 '''

34 ['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'osp', 'partial']35

各自单独运行

总结:

导入顶级模块,其名称会加入到本地名词空间中,并绑定到其模块对象。这里说的模块对象,就是导入的模块

导入非顶级模块,值将其顶级模块名称加入到本地名词空间中,导入的模块必须使用完全限定名称类访问。

print(path)报错,

print(os.path) 正确

如果使用了as,as后的名称直接绑定到导入的模块对象,并将该名称加入到本地名词空间中。

2.2、form...import...

from...import...  部分导入

from...import... as... 别名

测试2:

1 from pathlib import Path, PosixPath #当前名词空间导入该模块指定成员。

2 print(dir())3 '''

4 ['Path', 'PosixPath', '__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'partial']5 '''

6 from pathlib import * #在当前名词空间导入该模块所有公共成员(非下划线开头成员)或指定成员

7 print(dir())8 '''

9 ['Path', 'PosixPath', 'PurePath', 'PurePosixPath', 'PureWindowsPath', 'WindowsPath', '__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'partial']10 '''

11

12 from functools importwraps as wr, partial13 print(dir())14 '''

15 ['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'partial', 'wr']16 '''

17

18 from os.path import exists #加载,初始化os, os.path模块, exists 加入bending名词空间并绑定

19

20 if exists('f:/'):21 print('found')22 else:23 print('not found')24

25 print(dir())26 print(exists)27

28 '''

29 found30 ['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'exists', 'partial']31 32

33 '''

34 importos35

36 print(dir())37 print(os.path.exists)38 #print(exists) # NameError: name 'exists' is not defined 除非 from os.path import exists

39 print(type(os.path)) #

40 print(os.path.__dict__['exists'])41 print(getattr(os.path, 'exists'))42

43 '''

44 ['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'os', 'partial']45 46 47 48 '''

49 importsys50 print(sys.modules)51

52 '''

53 {'sys': , 'builtins': , '_frozen_importlib': , '_imp': , '_thread': , '_warnings': , '_weakref': , 'zipimport': , '_frozen_importlib_external': , '_io': , 'marshal': , 'nt': , 'winreg': , 'encodings': , 'codecs': , '_codecs': , 'encodings.aliases': , 'encodings.utf_8': , '_signal': , '__main__': , 'encodings.latin_1': , 'io': , 'abc': , '_abc': , 'site': , 'os': , 'stat': , '_stat': , 'ntpath': , 'genericpath': , 'os.path': , '_collections_abc': , '_sitebuiltins': , '_bootlocale': , '_locale': , 'encodings.gbk': , '_codecs_cn': , '_multibytecodec': , 'types': , 'importlib': , 'importlib._bootstrap': , 'importlib._bootstrap_external': , 'warnings': , 'importlib.util': , 'importlib.abc': , 'importlib.machinery': , 'contextlib': , 'collections': , 'operator': , '_operator': , 'keyword': , 'heapq': , '_heapq': , 'itertools': , 'reprlib': , '_collections': , 'functools': , '_functools': , 'mpl_toolkits': }54 '''

测试

总结:

找到from子句中指定的模块,加载并初始化它(注意不是导入)

对于import子句后的名称

先查from子句导入的模块是否具有该名称的属性

如果不是,则尝试导入该名称的子模块

还没有找到,就抛出ImportError异常

这个名称保存到本地名词空间总,如果有as 子句,则使用as子句后的名称。

举例:

from os importpath

按照步骤:

首先加载os模块,

先查os下面有没有path属性,如果没有

再找os下面有没有 path的模块,如果没有就报错

找到后,就把这个名词,保存在本地名词空间中

测试 3:因为类似于 类的方法,只需要定义一次,被不同的实例调用。所以模块下函数只需要生成一次

不同的导入方式,只生成一个,被调用。通过某种机制,保证生成一个类,一个方法等。

1 #from pathlib import Path

2 #print(Path, id(Path)) # 可以看到是 同一个对象。

3 #4 #import pathlib as p

5 #print(dir())

6 #print(p)

7 #print(p.Path, id(p.Path)) # # 可以看到是 同一个对象。

8 #9 #'''

10 # 42273688

11 #['Path', '__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'p', 'partial']

12 #

13 # 42273688

14 #'''

产生同一个对象

globals(), locals(), dir() 的作用范围总结1:

3、自定义类

测试:

自定义模块命名规范:

模块名就是文件名

模块名必须符合标识符的要求

不要使用系统模块名来避免冲突,除非你明确知道这个模块的用途

通常模块名都是小写,下划线分割。

4、模块搜索顺序:

使用 sys.path 查看搜索顺序 :现在自己项目中找,在去系统库,再去第三方库。

结果为 :Python模块的路径搜索顺序。

上面的是有问题的,因为我之前的代码都在code_pycharm 下的 code_test_daye下.这并不是项目目录

所谓的项目目录是在 code_pycharm下,也就是跟 code_test_daye同一层,

当加载一个模块的时候,需要从这些搜索路径中从前到后依次查找,并不搜索这些目录的子目录。

搜索到模块就加载,搜索不到就抛异常

路径也可以为字典,zip文件,egg文件。libzip可以直接访问该文件。

.egg文件,由setuptools库创建的包,第三方库常用的格式,添加了元数据(版本号,依赖项等)信息的zip文件

路径顺序为:

程序主目录,程序运行的主程序脚本所在的目录

PYTHON目录,环境变量PYTHON设置的目录也是搜索模块的路径

所以打印sys.path 有时候会出现两个项目目录,是因为加入了环境变量。

标准库目录,Python自带的库模块所在的目录

sys.path可以被修改,增加新的目录(一般不要修改)

5、模块的重复导入

从执行结果来看,不会产生重复导入现象。

也就是说,模块加载一会,产生模块对象,所以模块内定义的属性也只加载一次。

所有加载的模块都会记录在sys.modules中,sys.modules是存储已经加载过的所有模块的字典。

打印sys.modules可以看到os, os.path都已经加载了。就像一个已加载模块的cache

1 importsys2

3 print(sys.modules)4

5 {'sys': , 'builtins': , '_frozen_importlib': , '_imp': , '_thread': , '_warnings': , '_weakref': , 'zipimport': , '_frozen_importlib_external': , '_io': , 'marshal': , 'nt': , 'winreg': , 'encodings': , 'codecs': , '_codecs': , 'encodings.aliases': , 'encodings.utf_8': , '_signal': , '__main__': , 'encodings.latin_1': , 'io': , 'abc': , '_abc': , 'site': , 'os': , 'stat': , '_stat': , 'ntpath': , 'genericpath': , 'os.path': , '_collections_abc': , '_sitebuiltins': , '_bootlocale': , '_locale': , 'encodings.gbk': , '_codecs_cn': , '_multibytecodec': , 'types': , 'importlib': , 'importlib._bootstrap': , 'importlib._bootstrap_external': , 'warnings': , 'importlib.util': , 'importlib.abc': , 'importlib.machinery': , 'contextlib': , 'collections': , 'operator': , '_operator': , 'keyword': , 'heapq': , '_heapq': , 'itertools': , 'reprlib': , '_collections': , 'functools': , '_functools': , 'mpl_toolkits': }

sys.modules

6、模块运行

__name__ 每个模块都会定义一个__name__特殊变量来存储当前模块的名称,如果不指定,则默认为源代码文件名,如果是包则有限定名 m.m1.m2

解释器初始化的时候,会初始化sys.modules字典(保存已加载的模块),加载builtion(全局函数,常量)模块,__main__模块,sys模块,以及初始化模块搜索路劲sys.pah

也就是说:如果用sys.path,首先没有被加载,才会去找。

Python 是脚本语言,任何y一个脚本都可以直接执行,也可以作为模块被导入,除了包的__init__.py

当从标准输入(命令行方式敲代码) ,脚本($python test.py)或 交互式读取的时候,会将模块的__name__设置为__main__,模块的顶层代码就在__main__这个作用域中执行,顶层代码:模块中缩进最外层的代码,

如果import导入的,其__name__默认就是模块名,虽然可以修改但是不建议修改。

测试:__name__

1 #t1.py

2 print('#### t1 #####')3

4 if __name__ == '__main__':5 print('main')6 else:7 print('in import module')8

9

10

11 #t2.py

12 print('#### t2 #####')13 import t1

__name__

结果:

1 #### t2 #####

2 #### t1 #####

3 in import module

if __name__ == '__main__': 用途

本模块的功能测试,对于非本主模块,用来测试本模块内的函数,类

避免主模块变更的副作用顶层代码,没有封装,当主模块使用时没有问题,

但是一旦有了新的主模块,老的主模块成了被导入模块,由于原来代码没有封装,一并执行了。

把之前的 旧主模块的代码放到 if __name__ == '__main__' 下,封装起来

判断模块是否以程序的方式运行: $ python test.py

7、模块的属性:

属性

含义

__file__

字符串,源文件路径

__cached__

字符串,编译后的字节码文件路径

__spec__

显示模块的规模

__name__

模块名

__package__

当模块是包,同__name__,否则,可以设置为顶级模块的空字符串

1 importtt32

3 print(tt3.__cached__)4 print(tt3.__file__)5 print(__file__)6 print(tt3.__spec__)7 print(tt3.__name__)8

9 print(tt3.__package__)10

11

12

13 E:\code_pycharm\__pycache__\tt3.cpython-37.pyc14 E:\code_pycharm\tt3.py15 E:/code_pycharm/m/m2/m22.py16 ModuleSpec(name='tt3', loader=<_frozen_importlib_external.sourcefileloader object at>, origin='E:\\code_pycharm\\tt3.py')17 tt3

View Code

8、包

包,特殊的模块。

Python模块支持目录吗?

pycharm中,创建Directory 和 Python package 不同,前者是创建普通目录,后者是创建一个带有__init__.py文件的目录 即 包

python中,目录可以作为模块,这就是包,不过代码需要写到该目录下的 __init__.py中

9、子模块

关于locals() globals() dir()区别和联系:

总结:

首次,三者都不会突破 所在模块的边界,从上面的大一可以看出。

但dir()没传参,和locals() 是一样的,都是收集局部作用域中的变量,loacls() 是字典

globals() 则永远都是全局的,顶级的。

如果dir()给定了参数,就只收集所给参数的变量或属性。

主模块打印会有:'__annotations__'

子模块:

包目录下的py文件,子目录都是其子模块

测试:每个模块都 写: print(__name__)

1 print('~===========~', __name__)2

3 importsys4 importm.m15 importos.path6 importm.m2.m227

8 print(os.path.exists)9 print(m.m2.m22.show)10

11 print(dir())12

13 print(sorted(filter(lambda x:x.startswith('m'), sys.modules)))14

15 print(m.m1.show) #如果不是 import m.m1,就报错

16 #按照查找顺序,先看有没有m1属性,但是m 模块没m1,所以找m的子模块

17 #但是从 sys.modules 看,并没有加载,所以报错。

18 #所以只能先加载 import m.m1 或者m中有m1,如果导入后,m的属性字典中是有m1的

19

20 #搭配

21 #from m.m2 import m22

22 #print(m22.show)

t1.py 跟m在同一级

结果:

1 ~===========~ __main__

2 m3 m.m14 m.m25 m.m2.m226

7

8 ['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'm', 'os', 'sys']9 ['m', 'm.m1', 'm.m2', 'm.m2.m22', 'marshal', 'mpl_toolkits']10

结果

注意:import os 直接可以os.path,因为sys就直接导入了,在modules中可以看到,

但是必须导入os,才可以调用,dir()如果没有os,就无法调用

__file__ 只能留给包,__file__ 只能是模块

请保留__init__.py 文件,2版本并不能删,3版本虽然可以删,但不要删。。。。。

10、模块和包的总结:

包能够更好的最值模块,尤其是大的模块,其代码行数很多,可以用把它拆分成很多子模块,便于使用某些功能就加载相应的子模块。

包目录中__init__.py 是在包第一次导入的时候就会执行,内容可以为空,也可以是用于该报初始化工作的代码,最好不要删除它

导入子模块一定会加载福模块,但是导入父模块, 一定 不会导入子模块

包目录之间只能使用点号作为分割符,表示模块及其子模块的底层关系

模块也是封装,如同类,函数,不过他能够封装变量,类,函数。

模块就是命名空间,其内部的顶层标识符,都是它的属性,可以通过dict 或dir(模块名)查看

包也是模块,但是模块不一定是包,包是特殊的模块,是一种组织方式,是目录,它包含__path__, __file__属性等。

1 print(__file__)2 print(m.__file__)3 print(m.__path__)4

5 '''

6 E:/code_pycharm/code_test_daye/t1.py7 E:\code_pycharm\code_test_daye\m\__init__.py8 ['E:\\code_pycharm\\code_test_daye\\m']9 '''

1 print(__file__, type(__file__))2 print(m.__file__, type(m.__file__))3 print('E:\\code_pycharm\\code_test_daye\\m\\__init__.py')4

5 print(m.__path__, type(m.__path__[0]))6

7 '''

8 E:/code_pycharm/code_test_daye/t1.py 9 E:\code_pycharm\code_test_daye\m\__init__.py 10 E:\code_pycharm\code_test_daye\m\__init__.py11 ['E:\\code_pycharm\\code_test_daye\\m'] 12 '''

注意,都是字符串,win反斜杠

问题:

from json import encoder之后,json.dump 函数用不了。因为名词空间没有json

import json.encoder之后,json.dump可以,encoder是类,可以导入,其次,名词空间中是json

11、绝对导入,相对导入

在import语句或者from导入模块,模块名称最前面是不是以 . 开头的

绝对导入总是去模块搜索路径中找,当然会查看一下该模块是否已经加载。

相对导入:

只能在包内使用,只能在from语句中使用。

使用 . 点号,表示当前目录内

.. 表示上一级目录

不要在顶层模块中使用相对导入

有相对导入语句的模块不能直接运行,不能当主模块运行!!!!!!!!!!

因为使用了相对导入的模块就是为了内部相互的引用资源的,不是为了直接运行的,对于包来说,正确的使用方式还是在顶级模块使用这些包

在包内使用 相对导入,即便包名改变也不会影响内部资源引用

12、访问控制

下划线开头的模块名?

_ 或 __ 开头的模块,或变量

1 print(__name__)2 a = 5

3 _b = 6

4 __c = 7

5 __my__ = 8

测试:import访问

1 print(__name__)2

3 importt24

5 importsys6

7 print(sorted(sys.modules.keys()))8 print(dir())9 print(t2.a, t2._b, t2.__c, t2.__my__)10

11

12 '''

13 __main__14 t215 ['__main__', '_abc', '_bootlocale', '_codecs', '_codecs_cn', '_collections', '_collections_abc', '_frozen_importlib', '_frozen_importlib_external', '_functools', '_heapq', '_imp', '_io', '_locale', '_multibytecodec', '_operator', '_signal', '_sitebuiltins', '_stat', '_thread', '_warnings', '_weakref', 'abc', 'builtins', 'codecs', 'collections', 'contextlib', 'encodings', 'encodings.aliases', 'encodings.gbk', 'encodings.latin_1', 'encodings.utf_8', 'functools', 'genericpath', 'heapq', 'importlib', 'importlib._bootstrap', 'importlib._bootstrap_external', 'importlib.abc', 'importlib.machinery', 'importlib.util', 'io', 'itertools', 'keyword', 'marshal', 'mpl_toolkits', 'nt', 'ntpath', 'operator', 'os', 'os.path', 'reprlib', 'site', 'stat', 'sys', 't2', 'types', 'warnings', 'winreg', 'zipimport']16 ['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'sys', 't2']17 5 6 7 818 '''

View Code

普通变量,保护变量,私有变量,特殊变量,都没有被隐藏,也就是说模块内没有私有的变量,在模块中定义不做特殊处理

测试2:from访问,结果还是可以访问。

1 print(__name__)2

3 from t2 import a, _b as b, __c as c, __my__as my4

5 importsys6

7 print(sorted(sys.modules.keys()))8 print(dir())9 print(a, b, c, my)10

11

12 '''

13 __main__14 t215 ['__main__', '_abc', '_bootlocale', '_codecs', '_codecs_cn', '_collections', '_collections_abc', '_frozen_importlib', '_frozen_importlib_external', '_functools', '_heapq', '_imp', '_io', '_locale', '_multibytecodec', '_operator', '_signal', '_sitebuiltins', '_stat', '_thread', '_warnings', '_weakref', 'abc', 'builtins', 'codecs', 'collections', 'contextlib', 'encodings', 'encodings.aliases', 'encodings.gbk', 'encodings.latin_1', 'encodings.utf_8', 'functools', 'genericpath', 'heapq', 'importlib', 'importlib._bootstrap', 'importlib._bootstrap_external', 'importlib.abc', 'importlib.machinery', 'importlib.util', 'io', 'itertools', 'keyword', 'marshal', 'mpl_toolkits', 'nt', 'ntpath', 'operator', 'os', 'os.path', 'reprlib', 'site', 'stat', 'sys', 't2', 'types', 'warnings', 'winreg', 'zipimport']16 ['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'a', 'b', 'c', 'my', 'sys']17 5 6 7 818 '''

View Code

from ... import  *  和 __all__

使用from ... import  * 导入

测试1:结果,t2中的下划线开头的都没有导入

1 print(__name__)2

3 from t2 import *

4

5 importsys6

7 print(sorted(sys.modules.keys()))8 print(dir())9 print(locals()['a'])10 #print(locals()['_b']) # 访问不到

11

12 a = 1888888888888888888

13 print(locals()['a'])14

15 '''

16 __main__17 t218 ['__main__', '_abc', '_bootlocale', '_codecs', '_codecs_cn', '_collections', '_collections_abc', '_frozen_importlib', '_frozen_importlib_external', '_functools', '_heapq', '_imp', '_io', '_locale', '_multibytecodec', '_operator', '_signal', '_sitebuiltins', '_stat', '_thread', '_warnings', '_weakref', 'abc', 'builtins', 'codecs', 'collections', 'contextlib', 'encodings', 'encodings.aliases', 'encodings.gbk', 'encodings.latin_1', 'encodings.utf_8', 'functools', 'genericpath', 'heapq', 'importlib', 'importlib._bootstrap', 'importlib._bootstrap_external', 'importlib.abc', 'importlib.machinery', 'importlib.util', 'io', 'itertools', 'keyword', 'marshal', 'mpl_toolkits', 'nt', 'ntpath', 'operator', 'os', 'os.path', 'reprlib', 'site', 'stat', 'sys', 't2', 'types', 'warnings', 'winreg', 'zipimport']19 ['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', 'a', 'sys']20 521 188888888888888888822 '''

没有使用__all__

测试2:t2 使用了__all__

1 print(__name__)2

3 __all__ = ['x', 'y','a','_b']4 a = 5

5 _b = 6

6 __c = 7

7 __my__ = 8

8

9 x = 9

10 y = 10

1 print(__name__)2

3 from t2 import *

4

5 importsys6

7 print(sorted(sys.modules.keys()))8 print(dir())9 print(locals()['a'])10 print(locals()['x'])11 print(locals()['_b'])12

13

14

15 '''

16 __main__17 t218 ['__main__', '_abc', '_bootlocale', '_codecs', '_codecs_cn', '_collections', '_collections_abc', '_frozen_importlib', '_frozen_importlib_external', '_functools', '_heapq', '_imp', '_io', '_locale', '_multibytecodec', '_operator', '_signal', '_sitebuiltins', '_stat', '_thread', '_warnings', '_weakref', 'abc', 'builtins', 'codecs', 'collections', 'contextlib', 'encodings', 'encodings.aliases', 'encodings.gbk', 'encodings.latin_1', 'encodings.utf_8', 'functools', 'genericpath', 'heapq', 'importlib', 'importlib._bootstrap', 'importlib._bootstrap_external', 'importlib.abc', 'importlib.machinery', 'importlib.util', 'io', 'itertools', 'keyword', 'marshal', 'mpl_toolkits', 'nt', 'ntpath', 'operator', 'os', 'os.path', 'reprlib', 'site', 'stat', 'sys', 't2', 'types', 'warnings', 'winreg', 'zipimport']19 ['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__', '_b', 'a', 'sys', 'x', 'y']20 521 922 623

24 '''

View Code

__all__ 的意思是,宣称当前模块有多少属性公布出去

13、包和子模块

测试;

           

在__init__.py 中有什么变量,则使用ffrom m  import * 加载什么变量,这依然符合模块访问控制

总结:

14、模块变量的修改

总结:

模块对象是同一个,因此模块的变量也是同一个,对模块变量的修改,会影响所有使用者。

除非万不得已,或明确知道自己在做什么,否则不要修改模块的变量

前面说的猴子补丁,也可以通过打补丁的方式,修改模块的变量,类,函数等内容。

补充:

1、

import os

import os.path

两者明面上一样,通过dir(),都是os,但是后者导入的是, os 和 os.path

2、

import os.path 名词空间只有os 这种只将os 导入

import os.path as osp 名词空间 osp ,也就是说,直接将 os.path 导入

import os.path as osp

print(osp.exists)

print(os.path) # 所以这样是报错的,因为名词空间中没有os 只有osp

3、

import 后面只能导入 module

from ... import ... import 后面可以导入属性,函数等from后面只能是module

理解:从 。。。 导入。。。 出错一般是说 从。。。 出现问题,是模块吗,问自己

4、

from后面紧跟的,是不会导入到名词空间,只将import后面的加入到名词空间中

import单独使用,只导入顶级模块,除了别名外。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值