如何显示其自身源代码的副本

问题描述

问题

我试图了解程序是否可以打印自己的源代码

我找到了以下源代码

print((lambda str='print((lambda str=%r: (str %% str))())': (str % str))())

运行时,输出与源代码相同:

print((lambda str='print((lambda str=%r: (str %% str))())': (str % str))())
  1. 有人可以详细解释源代码的语法吗?

  2. 程序如何打印自己的源代码

解决方法

首先要记住的是:此表达式不会显示您放置在其中的任何文件的完整源代码。仅在这种特殊情况下,当源代码仅包含此行时(后跟换行符) ,如果我们正在做书呆子),则在运行该文件时,它将打印自己的源代码。

现在,让我们从头开始进行区分(运行此命令的输出在每个阶段都应该是相同的):

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 functionthe 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))())