Наверняка вы замечали, что в Python есть удобная функция для получения переменной окружения
Почему так?
На самом деле энвайромент обновляется, но это значение не добавляется в словарь
Откройте исходник функции
Словарь
В тоже время, когда вы создаёте или изменяете ключ в
То есть, технически
Аналогично при удалении переменной вызывается еще одна built-in функция
Итого
▫️ Удобней всего явно обновлять переменные через
▫️
#basic
os.getenv(NAME)
И её "сестра" для создания или изменения переменных окруженияos.putenv(NAME, VALUE)
Но почему-то putenv() не работает как должно. Энвайромент не обновляется!os.putenv('MYVAR', '1')
print(os.getenv('MYVAR'))
... и ничего 😴Почему так?
На самом деле энвайромент обновляется, но это значение не добавляется в словарь
os.environ.Откройте исходник функции
os.getenv(). Это просто шорткат для os.environ.get()
В то время как putenv() это built-in С-функция.Словарь
os.environ (или точней класс из MutableMapping) создаётся из энвайромента в момент инициализации. Функция putenv() самостоятельно его не изменяет.В тоже время, когда вы создаёте или изменяете ключ в
os.environ, автоматически вызывается putenv() в методе __setitem__().То есть, технически
putenv() всё делает верно, но в os.environ это не отражается. Можно проверить так:>>> os.putenv('MYVAR', '123')
>>> os.system('python -c "import os;print(os.getenv(\'MYVAR\'))"')
123
Я объявил переменную в текущем процессе и вызвал дочерний процесс, который её унаследовал и получил в составе os.environ.Аналогично при удалении переменной вызывается еще одна built-in функция
unsetenv(), удаляющая переменную из системы.Итого
▫️ Удобней всего явно обновлять переменные через
os.environ
▫️ Есть способ неявно создать/удалить переменную через putenv/unsetenv, что не повлияет на os.environ но изменит энвайромент и передаст изменения сабпроцессам. Но так лучше не делать!▫️
os.environ это просто обертка для built-in функций putenv() и unsetenv(). #basic
GitHub
cpython/Lib/os.py at 3.10 · python/cpython
The Python programming language. Contribute to python/cpython development by creating an account on GitHub.
👍2
Метод строки
А вот что будет по умолчанию
По умолчанию в качестве разделителя используется любой пробельный символ, будь то табуляция или новая строка. Включая несколько таких символов идущих подряд. А также игнорируются пробельные символы по краям строки.
#tricks #basic
split() разделяет строку на несколько строк по указанному символу>>> "a_b_c".split('_')
['a', 'b', 'c']
Можно указать максимальное количество разделений>>> "a_b_c".split('_', 1)
['a', 'b_c']
Или резать с другой стороны с помощью rsplit() (right split)>>> "a_b_c".rsplit('_', 1)
['a_b', 'c']
А что будет если оставить аргументы пустыми?>>> "a_b_c".split()Получаем список с одним элементом, потому что по умолчанию используется пробельный символ.
['a_b_c']
>>> "a b c".split()То есть это равнозначно такому вызову?
['a', 'b', 'c']
>>> "a b c".split(" ")
['a', 'b', 'c']
Кажется да, но нет! Давайте попробуем добавить пробелов между буквами>>> "a b c".split(" ")
['a', '', '', 'b', '', '', 'c']
И вот картина уже не так предсказуема 😕А вот что будет по умолчанию
>>> "a b c".split()Всё снова красиво! 🤩
['a', 'b', 'c']
По умолчанию в качестве разделителя используется любой пробельный символ, будь то табуляция или новая строка. Включая несколько таких символов идущих подряд. А также игнорируются пробельные символы по краям строки.
>>> "a\t b\n c ".split()Аналогичный способ можно собрать с помощью регулярного выражения. Но пробелы по краям строки придется обрабатывать дополнительно.
['a', 'b', 'c']
>>> import reЗдесь тоже можно указать количество разделений
>>> re.split(r"\s+", ' a b c '.strip())
['a', 'b', 'c']
>>> re.split(r"\s+", 'a b c', 1)А что если мы хотим написать красиво, то есть
['a', 'b c']
split() без аргументов, но при этом указать количество разделений? В этом случае первым аргументом передаём None
>>> "a\n b c".split(None, 1)Данный метод не учитывает строки с пробелами, взятые в кавычки
['a', 'b c']
'a "b c" '.split()Но для таких случаев есть другие способы.
['a', '"b', 'c"']
#tricks #basic
Telegram
Python Заметки
В посте про правильное использование аргумента shell упоминалось что в некоторых случаях атрибуты следует отправлять списком а не строкой. Что делать, если команда приходит именно строкой? Как её преобразовать в список?
Ответ очевиден
>>> cmd_str = 'ls …
Ответ очевиден
>>> cmd_str = 'ls …
👍25😱1😢1
Репозитори на почитать для расширения кругозора.
▫️ Большой ликбез по экосистеме Python
https://github.com/brunocampos01/understanding-the-python-ecosystem
▫️Подборка самых популярных полезностей для веб разработчиков
https://github.com/ml-tooling/best-of-web-python
▫️Подборка библиотек для тех кто уважает типизацию в Python
https://github.com/typeddjango/awesome-python-typing
#basic #libs
▫️ Большой ликбез по экосистеме Python
https://github.com/brunocampos01/understanding-the-python-ecosystem
▫️Подборка самых популярных полезностей для веб разработчиков
https://github.com/ml-tooling/best-of-web-python
▫️Подборка библиотек для тех кто уважает типизацию в Python
https://github.com/typeddjango/awesome-python-typing
#basic #libs
👍9❤6
Функция
Ранее я писал про функцию
Скорее всего вы уже знаете как использовать функцию
dir() - удобна для получения списка атрибутов у любого объекта.Ранее я писал про функцию
__dir__() в модуле (не путайте её с переменной __all__(), которая указывает список объектов для импорта если встречается конструкция from module import *). Скорее всего вы уже знаете как использовать функцию
dir(). Любой объект может реализовать метод __dir__() чтобы указать список имеющийхся и динамических атрибутов. И функция dir() поможет получить список этих атрибутов.>>> dir(str)У этой функции есть еще один способ применения. Её можно вызвать без аргумента, и в таком случае она вернёт список имён в текущем неймспейсе.
['__add__', '__class__', '__contains__', ...]
>>> dir()#basic #tricks
['__builtins__', '__doc__', '__file__', ...]
>>> def test():
>>> x = 1
>>> print(dir())
>>> test()
['x']
Telegram
Python Заметки
Знаете ли вы про "магические" методы классов ˍˍgetattributeˍˍ() и ˍˍgetattrˍˍ()?
ˍˍgetattributeˍˍ вызывается всякий раз когда идёт обращение к атрибуту объекта. Например метод или какая-то переменная.
ˍˍgetattrˍˍ вызывается когда атрибут не найден.
И…
ˍˍgetattributeˍˍ вызывается всякий раз когда идёт обращение к атрибуту объекта. Например метод или какая-то переменная.
ˍˍgetattrˍˍ вызывается когда атрибут не найден.
И…
👍7