如何解决如何显示其自身源代码的副本
问题
我试图了解程序是否可以打印自己的源代码,
我找到了以下源代码:
print((lambda str='print((lambda str=%r: (str %% str))())': (str % str))())
运行时,输出与源代码相同:
print((lambda str='print((lambda str=%r: (str %% str))())': (str % str))())
-
有人可以详细解释源代码的语法吗?
-
程序如何打印自己的源代码?
解决方法
首先要记住的是:此表达式不会显示您放置在其中的任何文件的完整源代码。仅在这种特殊情况下,当源代码仅包含此行时(后跟换行符) ,如果我们正在做书呆子),则在运行该文件时,它将打印自己的源代码。
现在,让我们从头开始进行区分(运行此命令的输出在每个阶段都应该是相同的):
print((lambda str='print(lambda str=%r: (str %% str))()': (str % str))())
首先,让我们将lambda表达式与print语句分开(您可以阅读有关lambda表达式here的信息,它们本质上是用于定义“小型匿名函数”的简洁表示法):
f = lambda str='print(lambda str=%r: (str %% str))()': (str % str)
print(f())
现在,为了更具可读性,我们可以将lambda表达式替换为等效函数:
def f(str='print(lambda str=%r: (str %% str))()'):
return str % str
print(f())
现在我们可以看到print
语句与此等效:
str = 'print(lambda str=%r: (str %% str))()'
print(str % str)
这是旧式字符串格式的示例,您可以阅读有关here的更多信息。本质上,当字符串后跟%
运算符时,它执行的操作在概念上类似于string.format
方法。
特别是,如果我们有一个类似str1 % obj
的表达式,则"%%"
中子字符串str1
的任何实例都将被子字符串"%"
替换,并且如果子字符串"%r"
仅出现一次,则将其替换为调用repr(obj)
的结果。举一个具体的例子,调用print("Test string: %% %r %%" % 23)
的输出为Test string: % 23 %
。
现在,回到Quine,我们拥有的就是这个:
r_string = repr('print(lambda str=%r: (str %% str))()')
print('print(lambda str={}: (str {} str))()'.format(r_string,"%"))
如果您对Python有一定的了解(您可能希望查找the repr
function和the string.format
method),这些语句应该很有意义,并且输出为:
print((lambda str='print(lambda str=%r: (str %% str))()': (str % str))())
这是原始源代码。
PS,如果您不是在问“此语法的工作原理”,而是“首先是某人如何提出此特定示例”的问题,那么(我不屑一顾)我不知道。
但是,通常来说,程序很容易打印自己的源代码。它所要做的只是包含语句print(*open(__file__),sep="")
。在此示例中,保存该语句的文件也可以包含其他命令。
__file__
是Python中的一个特殊变量,它以字符串形式包含源代码的文件名,而*
则进行“可迭代的拆包”。
编辑:我认为您的原始代码可能缺少一些括号?我认为应该是这样的(请注意,字符串文字中print
之后有2个大括号,而字符串文字末尾还有一个额外的右括号):
print((lambda str='print((lambda str=%r: (str %% str))())': (str % str))())
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。