资讯专栏INFORMATION COLUMN

Python小记(一):import和from。。。import区别?

番茄西红柿 / 3046人阅读

摘要:而当我们时这个是一个类我们称呼它为函数。后两者是常量用于忽略换行符和大小写。这就会导致很多难以觉察的潜在。结论无论是使用还是第一个要求是代码能够正常运行其次根据代码维护性团队编码风格来确定选择哪一种方案。

系统自带模块(库)

```cppimport retarget = 'abc1234xyz're.search('(/d+)', target)
但有时候,你可能会看到某些人这样写代码:```pythonfrom re import searchtarget = 'abc1234xyz'search('(/d+)', target)

那么这两种导入方式有什么区别呢?

我们分别使用type函数来看看他们的类型:

>>> import re>>> type(re)<class 'module'>>>> from re import search>>> type(search)<class 'function'>

可以看到,直接使用import re导入的re它是一个module类,也就是模块。我们把它成为正则表达式模块。而当我们from re import search时,这个search是一个function类,我们称呼它为search 函数。
一个模块里面可以包含多个函数。
如果在你的代码里面,你已经确定只使用search函数,不会再使用正则表达式里面的其他函数了,那么你使用两种方法都可以,没什么区别。
但是,如果你要使用正则表达式下面的多个函数,或者是一些常量,那么用第一种方案会更加简洁清晰。

例如:

import rere.search('c(.*?)x', flags=re.S)re.sub('[a-zA-Z0-9]', '***', target, flags=re.I)

在这个例子中,你分别使用了re.search,re.sub,re.S和re.I。后两者是常量,用于忽略换行符和大小写。
但是,如果你使用from re import search, sub, S, I来写代码,那么代码就会变成这样:

import research('c(.*?)x', flags=S)sub('[a-zA-Z0-9]', '***', target, flags=I)

看起来虽然简洁了,但是,一旦你的代码行数多了以后,你很容易忘记S和I这两个变量是什么东西。而且我们自己定义的函数,也很有可能取名为sub或者search,从而覆盖正则表达式模块下面的这两个同名函数。这就会导致很多难以觉察的潜在 bug。
再举一个例子。Python 的 datetime模块,我们可以直接import datetime,此时我们导入的是一个datetime模块,

输出为:class‘module’

但是如果你写为from datetime import datetime,那么你导入的datetime是一个type类:

输出为:classtype

因为这种方式导入的datetime,它就是Python 中的一种类型,用于表示包含日期和时间的数据。

这两种导入方式导入的datetime,虽然名字一样,但是他们的意义完全不一样,请大家观察下面两种写法:

import datetimenow = datetime.datetime.now()one_hour_ago = now - datetime.timedelta(hours=1)from datetime import datetime, timedeltanow = datetime.now()one_hour_ago = now - timedelta(hours=1)

第二种写法看似简单,但实则改动起来却更为麻烦。例如我还需要增加一个变量today用于记录今日的日期。

对于第一段代码,我们只需要增加一行即可:

today = datetime.date.today()

但对于第二行来说,我们需要首先修改导入部分的代码:

from datetime import datetime, timedelta, date
然后才能改代码:today = date.today()

这样一来你就要修改两个地方,反倒增加了负担。

第三方库

在使用某些第三方库的代码里面,我们会看到类似这样的写法:

from lxml.html import fromstring

selector = fromstring(HTML)
但是我们还可以写为:

from lxml import html

selector = html.fromstring(HTML)
但是,下面这种写法会导致报错:

import lxml
selector = lxml.html.fromstring(HTML)
那么这里的lxml.html又是什么东西呢?

这种情况多常见于一些特别大型的第三方库中,这种库能处理多种类型的数据。例如lxml它既能处理xml的数据,又能处理html的数据,于是这种库会划分子模块,lxml.html模块专门负责html相关的数据。

动手试试

在使用某些第三方库的代码里面,我们会看到类似这样的写法:

 from lxml.html import fromstring selector = fromstring(HTML)

但是我们还可以写为:

from lxml import htmlselector = html.fromstring(HTML)

但是,下面这种写法会导致报错:

import lxmlselector = lxml.html.fromstring(HTML)

那么这里的lxml.html又是什么东西呢?
这种情况多常见于一些特别大型的第三方库中,这种库能处理多种类型的数据。例如lxml它既能处理xml的数据,又能处理html的数据,于是这种库会划分子模块,lxml.html模块专门负责html相关的数据。

自己来实现多种导入方法
我们现在自己来写代码,实现这多种导入方法。
我们创建一个文件夹DocParser,在里面分别创建两个文件main.py和util.py,他们的内容如下:

util.py文件:

def write():    print('write 函数被调用!')main.py文件:import utilutil.write()

现在我们把main.py的导入方式修改一下(结果与上面相同):

from util import writewrite()

现在,我们来创建一个文件夹microsoft,里面再添加一个文件parse.py:

def read():    print('我是 microsoft 文件夹下面的 parse.py 中的 read函数')

此时我们在 main.py中对它进行调用:

from microsoft import parseparse.read()

我们也可以用另一种方法:

from microsoft.parse import readread()

但是,你不能直接导入microsoft
import microsoft
microsoft.parse.read

无论你使用的是import xxx还是from xxx.yyy.zzz.www import qqq,你导入进来的东西,要不就是一个模块(对应到.py 文件的文件名),或者是某个.py 文件中的函数名、类名、变量名。

无论是import xxx还是from xxx import yyy,你导入进来的都不能是一个文件夹的名字。

可能有这样一种情况,就是某个函数名与文件的名字相同,例如:

在 microsoft文件夹里面有一个microsoft.py文件,这个文件里面有一个函数叫做microsoft,那么你的代码可以写为:

from microsoft import microsoft`
microsoft.microsoft()
但请注意分辨,这里你导入的还是模块,只不过microsoft.py文件名与它所在的文件夹名恰好相同而已。

结论

无论是使用import还是from import,第一个要求是代码能够正常运行,其次,根据代码维护性,团队编码风格来确定选择哪一种方案。

如果我们只会使用到某个模块下面的一个函数(或者常量、类)并且名字不会产生混淆,可识别性高,那么from 模块名 import 函数名这没有什么问题。

如果我们会用到一个模块下面的多个函数,或者是我们将要使用的函数名、常量名、类名可能会让人产生混淆(例如 re.S、re.I),那么这种情况下,import 模块名然后再 模块名.xxx来调用会让代码更加清晰,更好维护。

但无论什么情况下,都禁止使用from xxx import *这种写法,它会给你带来无穷无尽的噩梦。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/119990.html

相关文章

  • python小记

    摘要:提示由于水平有限,如发现有疑问或错误的地方请毫不客气的提出讨论我会在第一时间回复感谢在先重要的事情说三遍查看文档的时候方法执行一定添加查看文档的时候方法执行一定添加获取帮助方法查看另一种方法查看中内置方法问题描述在树莓派支持的上面最后提示安 提示:由于水平有限,如发现有疑问或错误的地方请毫不客气的提出、讨论,我会在第一时间回复,感谢在先 重要的事情说三遍 查看module文档的时候方...

    baiy 评论0 收藏0
  • react-router 升级小记

    摘要:前言最近将公司项目的从版本升到了版本,跟完全不兼容,是一次彻底的重写。升级过程中踩了不少的坑,也有一些值得分享的点。没有就会匹配所有路由最后不得不说升级很困难,坑也很多。 前言 最近将公司项目的 react-router 从 v3 版本升到了 v4 版本,react-router v4 跟 v3 完全不兼容,是一次彻底的重写。这也给升级造成了极大的困难,与其说升级不如说是对 route...

    isLishude 评论0 收藏0
  • Vuex 升级入坑小记

    摘要:升级入坑小记场景描述引入的版本为,开启调试工具默认升级后可以调试。遂升级,发现大量使用失效,报,的中文文档,没有及时更新。机票订单和用户信息。 Vuex 升级入坑小记 场景描述 引入Vuex的版本为0.3,开启调试工具默认升级后可以调试Vuex。给作者一个大大的赞。为提高开发体验也是操碎了心 (๑•̀ㅂ•́)و✧ (8。安利下(Vue Devtools)。 Vue Devtools ...

    ziwenxie 评论0 收藏0
  • Vue2学习小记-给Vue2路由导航钩子axios拦截器做个封装

    摘要:写在前面最近在学习,遇到有些页面请求数据需要用户登录权限服务器响应不符预期的问题,但是总不能每个页面都做单独处理吧,于是想到提供了拦截器这个好东西,再于是就出现了本文。 1.写在前面 最近在学习Vue2,遇到有些页面请求数据需要用户登录权限、服务器响应不符预期的问题,但是总不能每个页面都做单独处理吧,于是想到axios提供了拦截器这个好东西,再于是就出现了本文。 2.具体需求 用户鉴...

    Tikitoo 评论0 收藏0
  • 微信公众号开发小记——2.80端口上的服务

    摘要:微信在做一些操作是需要用到生成二维码等而每天接口的调用上限为,需要自己做缓存文档讲了几种方式,我觉得放在中拿比较妥当。微信菜单会缓存分钟,你可以取消关注,然后在关注查看菜单变化效果。 描述 微信公众号开发基本分为2大种类型 1.用户直接做了某些操作(回复信息、订阅、扫码、发语音、点按钮等),此时这些信息微信会发送到微信服务器的80端口,这是一种开发类型;2.通过连接(按钮、文章)引导用...

    frank_fun 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<