8

[编辑 00]:我已经对帖子进行了多次编辑,现在甚至编辑了标题,请阅读以下内容。

我刚刚了解了格式字符串方法,以及它在字典中的使用,例如由 和 提供vars()locals()字典globals()

name = 'Ismael'
print 'My name is {name}.'.format(**vars())

但我想做:

name = 'Ismael'
print 'My name is {name}.' # Similar to ruby

所以我想出了这个:

def mprint(string='', dictionary=globals()):
    print string.format(**dictionary)

您可以在此处与代码交互:http: //labs.codecademy.com/BA0B/3# :workspace

最后,我想做的是将函数放在另一个名为 的文件中my_print.py,所以我可以这样做:

from my_print import mprint

name= 'Ismael'
mprint('Hello! My name is {name}.')

但是现在,范围存在问题,我如何从导入的 mprint 函数中获取主模块命名空间作为字典。(不是来自 的那个my_print.py

我希望我让自己明白,如果没有,请尝试从另一个模块导入该功能。(回溯在链接中)

它是从 dict 访问globals()dict my_print.py,但是变量名当然没有在那个范围内定义,关于如何实现这一点的任何想法?

如果函数在同一个模块中定义,则该函数有效,但请注意我必须如何使用globals(),因为如果不是,我只会得到一个包含mprint()范围内值的字典。

我曾尝试使用非本地和点符号来访问主模块变量,但我仍然无法弄清楚。


[编辑 01]:我想我已经找到了解决方案:

在 my_print.py 中:

def mprint(string='',dictionary=None):
    if dictionary is None:
        import sys
        caller = sys._getframe(1)
        dictionary = caller.f_locals
    print string.format(**dictionary)

在 test.py 中:

from my_print import mprint

name = 'Ismael'
country = 'Mexico'
languages = ['English', 'Spanish']

mprint("Hello! My name is {name}, I'm from {country}\n"
       "and I can speak {languages[1]} and {languages[0]}.")

它打印:

Hello! My name is Ismael, I'm from Mexico
and I can speak Spanish and English.

你们觉得怎么样?那对我来说是一个困难的!

我喜欢它,对我来说更具可读性。


[编辑 02]:我制作了一个模块,其中包含一个interpolate函数、一个Interpolate类和一个interpolate类似于该函数的类方法的尝试。

它有一个小型测试套件并记录在案!

我坚持方法实现,我不明白。

这是代码: http: //pastebin.com/N2WubRSB

你们觉得怎么样?


[编辑 03]:好的,我现在已经解决了这个interpolate()功能。

string_interpolation.py

import sys


def get_scope(scope):
    scope = scope.lower()
    caller = sys._getframe(2)
    options = ['l', 'local', 'g', 'global']

    if scope not in options[:2]:
        if scope in options[2:]:
            return caller.f_globals
        else:
            raise ValueError('invalid mode: {0}'.format(scope))
    return caller.f_locals


def interpolate(format_string=str(),sequence=None,scope='local',returns=False):
    if type(sequence) is str:
        scope = sequence
        sequence = get_scope(scope)
    else:
        if not sequence:
            sequence = get_scope(scope)

    format = 'format_string.format(**sequence)'
    if returns is False:
        print eval(format)

    elif returns is True:
        return eval(format)

再次感谢各位!有什么意见吗?


[编辑 04]:

这是我的最后一个版本,它有一个测试、文档字符串并描述了我发现的一些限制:http: //pastebin.com/ssqbbs57

您可以在这里快速测试代码:http: //labs.codecademy.com/BBMF# :workspace

并在此处克隆 grom git repo: https ://github.com/Ismael-VC/python_string_interpolation.git

4

3 回答 3

2

模块在 python 中不共享命名空间,所以globals()formy_print总是globals()my_print.py 文件的;即实际定义函数的位置。

def mprint(string='', dic = None):
    dictionary = dic if dic is not None else globals()
    print string.format(**dictionary)

您应该显式传递当前模块的 globals() 以使其工作。

Ans 在 python 函数中不使用可变对象作为默认值,它可能会导致意外结果。改为None用作默认值。

一个了解模块范围的简单示例:

文件:my_print.py

x = 10
def func():
    global x
    x += 1
    print x

文件:main.py

from my_print import *
x = 50
func()   #prints 11 because for func() global scope is still 
         #the global scope of my_print file
print x  #prints 50
于 2013-05-12T06:37:21.703 回答
1

您的问题的一部分——好吧,它不起作用的原因——在这个问题中突出显示。

您可以通过globals()作为第二个参数传入来让您的函数工作,mprint('Hello my name is {name}',globals()).

尽管在 Ruby 中可能很方便,但如果您想充分利用该语言,我建议您不要用 Python 编写 Ruby。

于 2013-05-12T06:33:50.307 回答
0

语言设计不仅仅是解决难题:;)

http://www.artima.com/forums/flat.jsp?forum=106&thread=147358

编辑: PEP-0498解决了这个问题!

模块中的Templatestring,也做了我需要的(但更类似于字符串format方法),最后它也有我所追求的可读性,它也有推荐的显式性,它在标准库中,也可以很容易定制和扩展。

http://docs.python.org/2/library/string.html?highlight=template#string.Template

from string import Template

name = 'Renata'
place = 'hospital'
job = 'Dr.'
how = 'glad'
header = '\nTo Ms. {name}:'

letter = Template("""
Hello Ms. $name.

I'm glad to inform, you've been
accepted in our $place, and $job Red
will ${how}ly recieve you tomorrow morning.
""")

print header.format(**vars())
print letter.substitute(vars())

有趣的是,现在我越来越喜欢使用{}而不是,$而且我仍然喜欢string_interpolation我提出的模块,因为从长远来看,它比任何一个都少打字。哈哈!

在此处运行代码:

http://labs.codecademy.com/BE3n/3#:workspace

于 2013-05-17T15:46:45.137 回答