在"python之禅"那几句话中有一句:namespace is a good thing.
python对于命名空间的处理非常简单,下面的内容不一定真实,完全是我根据现象推测出来的.

一.上来直接调用全局变量

x=[1]
def haha():
   #x=[2]
   x.append('baga')

haha()
print(x)

如果没有注释那句话,输出为[1];如果注释了那句话,输出为[1,'baga'].
这说明python中,有一个命名列表,从内层到外层挨个寻找,直到找到为止.即便整个命名列表中包含很多重复,那也没啥关系,因为默认为最内层的那一个.对于每一层,一个变量名只能对应一个变量,不允许存在多个变量名,即每一层作用域都相当于一个HashMap即class Namespace extends HashMap<String,Object>{},而多层作用域组织起来就是一个栈.Stack<Namespace>namespaceStack.当遇到一个缩进时,向namespaceStack中插入一个Namespace结点,当退回一个缩进时,弹栈一个Namespace结点.
当查找某个变量时,从栈顶向栈底层层搜索各个Namespace,直到找到该变量名为止.

二.进入一个作用域时立即建立Namespace对象且在运行过程中不添加新元素

x = "weidiao"

def haha():
   y=x.upper()
   #x=y

haha()
print(x)

如果注释x=y,代码正常运行;如果不注释这句话,上述代码会报错UnboundLocalError: local variable 'x' referenced before assignment,说x.upper()中的x无法解析.
这说明当python进入一个作用域时,会首先扫描一遍这个作用域,把变量名映射建立起来,但是它们的值都是未绑定状态.

三.告诉python我用的是全局变量

x = "weidiao"

def haha():
   #global x
   x='haha'

haha()
print(x)

如果注释global x,那么haha()函数中定义的x='haha'就会插入到haha的Namespace中去.如果不注释global语句,那么global x相当于告诉编译器,在本作用域内不插入x,这样一来编译器即便扫描到x=也不会将x插入到Namespace中去.因为本作用域内没有x,所以遇到x=这样的赋值语句时,执行的操作就相当于直接对外层变量进行操作.
如果在函数中先定义x=,然后又把global x就会报错,因为在执行global语句时发现没法执行.

四.global是一种指令,一种行为.它影响的是整个函数的Namespace,而不是内层的Namespace

x = "weidiao"

def haha():
   for i in range(3):
      global x
      print(x.upper())
   x='haha'

haha()
print(x)

程序输出为haha
我是在for循环作用域内声明global的,但是这个global却影响了整个函数.这表明global总是作用于函数的Namespace,于是可见Namespace也是分成很多类别的.

五.Namespace的分类

  • 函数的Namespace,称为function类型的namespace
  • for,if,while的Namespace,统一为block类型的namespace
  • 类的Namespace,称为class类型的namespace

在java中,function的namespace不能跟block的namespace有同名元素(编译报错),function namespace可以跟class namespace有同名元素,block namespace也可以跟class namespace有重名元素.
在python中,block namespace和function namespace可以包含同名元素.一切都是简单化处理.

六.伪代码实现python中的命名作用域机制

class Namespace(set):
   def __init__(self, type):
      self.type = type  # 定义命名空间的类型

   def insert(self, name):
      self[name] = unasigned  # 插入一个变量名,默认它是未赋值的

class FunctionNameSpace(Namespace):
   def __init__(self):
      Namespace.__init__("function")
      self.globalList = []  # 函数命名空间都有一个globalList

stack = Stack < Namespace > ()

def getVar(name):
   for namespace in stack:
      if namespace.contains(name):
         return namespace[name]
   raise("undefined variable %s"%name)

def processLine(line):
   if line is "global sentence":
      ns = stack.getTopFunctionNamespace()  # 获取栈顶第一个函数命名空间
      if ns.contains(line.globalName):
         if ns[line.globalName].unasigned:  # 如果存在这样的局部变量但是未曾赋值则删掉
            ns.remove(line.globalName)
         else:  # 已经存在的局部变量且已赋值,则报错无法声明global
            raise Exception("SyntaxWarning: name '%s' is assigned to before global" % line.globalName)
      else:
         ns.globalList.append(line.globalName)  # 添加到全局变量表中
   else:
      pass

def parseFunction(src):
   for line in src:
      if line.indent.delta == 0:#本行缩进变化量为0
         process(line)
      elif line.indent.delta == 1:#本行缩进变化量为1,即多缩进了一个tab
         ns = Namespace("block")
         for l in line.nextLinesWithSameIndent:
            if l is "assign sentence":
               ns.push(l.name)
         stack.push(ns)
         process(line)
      elif line.indent.delta < 0:#本行缩进变化量为负数,即回退了delta个,需要弹栈
         for i in range(line.indent.delta, 0):
            stack.pop()
         process(line)

python命名空间的更多相关文章

  1. python命名空间与作用域

    python命名空间与作用域   命名空间是名称与对象之间的关系,可以将命名空间看做是字典,其中的键是名称,值是对象. 命名空间不共享名称. 在命名空间中的名称能将任何python对象作为值,在不同的 ...

  2. python命名空间的本质

    Python的命名空间是Python程序猿必须了解的内容,对Python命名空间的学习,将使我们在本质上掌握一些Python中的琐碎的规则. 接下来我将分四部分揭示Python命名空间的本质:一.命名 ...

  3. [Python] 命名空间&作用域

    Python的类语句不会创建实例 类会创建命名空间,通过对象访问类的属性和方法 类不会创建作用域,对方法和属性的引用必须加以限定(如在方法中必须通过self引用实例的属性) class My1(): ...

  4. Python 命名空间

    通俗的来说,Python中所谓的命名空间可以理解为一个容器.在这个容器中可以装许多标识符.不同容器中的同名的标识符是不会相互冲突的.理解python的命名空间需要掌握三条规则: 第一,赋值(包括显式赋 ...

  5. 详解python命名空间和作用域

    1.典型案例 先从几个典型的案例来看下名称空间及作用域对python代码运行的影响,请看下面几个代码实例及其执行结果,是否符合你的预期. 代码1:块作用域 if True: i = 1 print i ...

  6. python类和模块区别,python命名空间

    在python中,类可以提供模块级别之下的命名空间. 如果一个模块写很多函数,某些函数之间共同完成一组功能,用类会看起来更清晰,在调用时候也会更好,对于ide补全有更小范围的限定提示. 类提供 继承 ...

  7. python—命名空间、作用域查找顺序、闭包

    名称空间 name space,如下图: x = 1, 1存放在内存中,1 会有一个内存地址,x 则 存放在 name space 里,并同时记录了 1的内存地址, 即 名称空间是存放了变量x与1绑定 ...

  8. python命名空间、作用域、闭包与传值传引用

    (以下内容,均基于python3) 最近在看python函数部分,讲到了python的作用域问题,然后又讲了Python的闭包问题. 在做作业的时候,我遇到了几个问题,下面先来看作业. 一. 作业1: ...

  9. Python命名空间和作用域

    准备知识: 1.在Python解释器开始执行之后,机会在内存中开辟一个空间,每当遇到 一个变量的时候,就把变量和值之间的关系记录下来,但是当遇到函数定义 的时候,解释器只是把函数名读入内存,表示这个函 ...

随机推荐

  1. html+ccs3太阳系行星运转动画之土星有个环,地球有颗小卫星

    在上一篇<html+ccs3太阳系行星运转动画>中实现了太阳系八大行星的基本运转动画. 太阳系又何止这些内容,为丰富一下动画,接下来增加“土星环”和“月球”来充盈太阳系动画. 下面是充盈后 ...

  2. .Net语言 APP开发平台——Smobiler学习日志:如何实现快速跳转网页

    Smobiler是一个在VS环境中使用.Net语言来开发APP的开发平台,也许比Xamarin更方便 样式一 一.跳转网页代码(Button的Click事件) Private Sub Button1_ ...

  3. c/c++常见面试题

    1. C中static有什么作用 (1)隐藏. 当我们同时编译多个文件时,所有未加static前缀的全局变量和函数都具有全局可见性,故使用static在不同的文件中定义同名函数和同名变量,而不必担心命 ...

  4. Android ORM -- Litepal(1)

    ORM,即Object Relation Mapping,对象关系映射,实现了程序里面的类和数据库里面的数据之间的对应关系,对数据库的操作可以通过对类的操作去实现,不用再写SQL语句,从而提高了开发效 ...

  5. html5+jqueryMobile编写App推广注册页

    html5+jqueryMobile的组合可以直接开发web版的app,所以用到我当前app中的推广注册页的编写是很恰当的,其实只要你熟悉html4+jquery的组合开发,那么html5+jquer ...

  6. nginx php-fpm 输出php错误日志

    nginx是一个web服务器,因此nginx的access日志只有对访问页面的记录,不会有php 的 error log信息. nginx把对php的请求发给php-fpm fastcgi进程来处理, ...

  7. px、dp和sp,这些单位有什么区别?

    DP 这个是最常用但也最难理解的尺寸单位.它与“像素密度”密切相关,所以 首先我们解释一下什么是像素密度.假设有一部手机,屏幕的物理尺寸为1.5英寸x2英寸,屏幕分辨率为240x320,则我们可以计算 ...

  8. NodeJs端口被占用的情况

    在NodeJs运行的时候,我们往往会遇到一个问题:“端口被占用”,这个问题,我们的处理办法有哪些呢? 这里我只介绍一下linux下的方法: 1.使用nodeme(忘记是不是这个啦,回去查下,这个要安装 ...

  9. 使用python处理子域名爆破工具subdomainsbrute结果txt

    近期学习了一段时间python,结合自己的安全从业经验,越来越感觉到安全测试是一个体力活.如果没有良好的coding能力去自动化的话,无疑会把安全测试效率变得很低. 作为安全测试而言,第一步往往要通过 ...

  10. Oracle Sales Cloud:报告和分析(BIEE)小细节2——利用变量和过滤器传参(例如,根据提示展示不同部门的数据)

    在上一篇随笔中,我们建立了部门和子部门的双提示,并将部门和子部门做了关联.那么,本篇随笔我们重点介绍利用建好的双提示进行传参. 在操作之前,我们来看一个报告和分析的具体需求: [1] 两个有关联的提示 ...