逆向之旅:七个让Python编程更糟糕的小技

2024年 5月 6日 74.1k 0

文章带大家进入Python编程的奇特角落,探索那些完全出乎意料甚至无厘头的编程事实。在这里,你可以窥见如何在一行内完成复杂的函数,或者是如何完全无视掉代码中的assert语句。

逆向之旅:七个让Python编程更糟糕的小技-1

如果你想在Python编程方面变得更糟糕,那么,这篇文章就是为你量身定制的。

(1) _.__.___.____._____ 也能行

class Test:
    def __getattr__(self, key):
        return Test()

在一个类中,当1用于访问someattribute属性2时调用magic方法someattributes属性没有显式定义__getattribute__object.someattribute。

class Test:
    def __getattr__(self, key):
        return Test()


_ = Test()

print(_)
print(_.__)
print(_.__.___)
print(_.__.___.____)
print(_.__.___.____._____)

# 
# 
# 
# 
# 
  • _就是一个测试类对象
  • _.__调用,返回另一个 Test 对象__getattr__
  • _.__.___又一次调用,返回另一个Test 对象__getattr__
  • _.__.___.____再一次调用,返回另一个Test 对象__getattr__
  • _.__.___.____._____继续调用,返回另一个Test 对象__getattr__
  • 这就是可能的_.__.___.____._____

(2) 利用exec()在一行中写代码

如果你曾经想要学的是如何在一行代码中写出任何 Python 函数,那么这点就适合你。假设有一个多行函数,但是并不知道如何才能合法地把它压缩成一行代码

def hi():
    print('apple')
    print('orange')

为了便于理解,这里有一个简明易懂的例子。

exec("def hi():\n. print('apple')\n  print('orange')")

exec()会将字符串作为Python代码来执行 因此,如果将函数编成字符串格式并把提供给exec,就成功地用python的一行代码来写出这个函数了

(3) Dog()()()()可以是有效代码

class Dog:
    pass


dog = Dog()
print(dog())  # error

目前得到了一个错误,这是因为默认情况下,是不能将一个对象像一个函数一样去调用的。

class Dog:
    def __call__(self):
        return 'apple pie'


dog = Dog()
print(dog())  # apple pie

定义魔术方法__call__可以决定对象的调用行为,就像对函数进行调用一样。

class Dog:
    def __call__(self):
        return Dog()


dog = Dog()
print(dog())  # 

在这,让__call__返回一个新的 Dog 对象。

print(Dog()()()()())

由于每次都得到一个新的 Dog 对象,无论何时对Dog对象进行调用,都可以无限地调用下去。

(4) 多个负号的问题

x = -1--2---3----4-----5

这是有效的 Python 代码:

  • 1-2可以被解读为 1 减 2
  • -1则被看成负数(不是减法的负号)
  • --1是负负,就等于 1
  • ---1是负负负的,又变成了 -1
  • 这种情况就会一直持续下去

-1--2---3----4-----5:因此,这是合法的 Python 代码-1-(-2)-(--3)-(---4)-(----5)

(5)如何限制使用print()

print = None

print('hello world')

# TypeError: 'NoneType' object is not callable

在这,将print = None,将None赋值给了print。这覆盖了非常熟悉的常用函数,因此我们现在无法使用print()。

print = None

__builtins__.print('hello')  # hello

但可以用__builtins__.print来访问原来的print()功能!

print = None
__builtins__ = None

如果让__builtins__设置为None。现在没有人能再使用 print() 了。

(6)让全局变量都消失

a = 4
b = 5
c = 6

keys = list(globals().keys())
for key in keys:
    del globals()[key]
del globals()['keys']
del globals()['key']

print(globals())  # {}

print(a)  # 'a' 没有定义
  • globals()会返回含有所有全局变量的字典
  • 可以删掉所有在globals()中的键值对
  • 删完后,globals()变成了空的
  • 这意味着没有剩余的全局变量,并且之前定义的所有全局变量都一去不复返了

(7)可无视assert语句

你是不是经常assert代码语句困扰?别担心,因为有一种方法能让你完全无视掉assert语句。

# hello.py
assert 1 == 2

如果用python hello.py来运行它,由于 assert 语句,会收到一个 AssertionError。

# hello.py
assert 1 == 2
  • 也可以选择用python -O hello.py来运行它
  • -O标志让 Python 忽略所有的assert语句
  • 现在,可以让你的代码忽略所有 assert 语句,无论它们会捕获何种错误!毕竟,谁在乎呢?

结论

文章带大家进入Python编程的奇特角落,探索那些完全出乎意料甚至无厘头的编程事实。在这里,你可以窥见如何在一行内完成复杂的函数,或者是如何完全无视掉代码中的assert语句。

如果你对编程充满好奇或者想成为一个更糟糕的程序员,那么这篇文章将为你揭开一个被藏在细节里的惊人世界。

相关文章

JavaScript2024新功能:Object.groupBy、正则表达式v标志
PHP trim 函数对多字节字符的使用和限制
新函数 json_validate() 、randomizer 类扩展…20 个PHP 8.3 新特性全面解析
使用HTMX为WordPress增效:如何在不使用复杂框架的情况下增强平台功能
为React 19做准备:WordPress 6.6用户指南
如何删除WordPress中的所有评论

发布评论