Задать вопрос
Ответы пользователя по тегу Python
  • Как вывести списки в 2 столбца?

    Vindicar
    @Vindicar
    RTFM!
    По-моему, второй for должен быть на одном уровне с первым, а не вложен в первый.
    Проверь отступы.
    Ответ написан
    1 комментарий
  • Как сделать относительный импорт без родительского пакета?

    Vindicar
    @Vindicar
    RTFM!
    Никак. Пакеты могут импортировать другие пакеты на одном с ними уровне или уровнем ниже, но не могут импортировать файлы из пакетов уровнем выше.

    Если, конечно, не манипулировать механизмом импорта. Он это позволяет, но обычно овчинка выделки не стоит.
    Так что проще закинуть module1 в каталог(пакет) и импортировать его как package1.module1.

    А вообще очень странное решение - почему скрипт верхнего уровня в каталоге, а импортируемые модули - нет?
    Ответ написан
    Комментировать
  • Как заставить def работать?

    Vindicar
    @Vindicar
    RTFM!
    Поучи язык. Просто поучи, без ботов.
    def объявляет функцию, но не вызывает её.
    Ты её объявляешь внутри record_volume(), но ничего с ней не делаешь.
    Ответ написан
    Комментировать
  • Как задать несколько значений одной команде?

    Vindicar
    @Vindicar
    RTFM!
    cоздать несколько отдельных BACK, но не меняя самого текстового значения?


    Освой finite state machines (FSM).
    Ответ написан
    2 комментария
  • Как добавить таймер?

    Vindicar
    @Vindicar
    RTFM!
    Elacov_top, для извлечения фрагментов из текста можно использовать регулярные выражения. Это позволит разбирать только фиксированный набор фраз, но этого зачастую хватает. Хуже с тем. что слово "пять" надо будет превратить в 5. Тут я даже навскидку не подскажу. Хотя есть средства разбора естественных языков, типа такого.

    У тебя другая проблема, поважнее. Ты представляешь, как вырастет твой if-if-if спустя 5-10 фич?
    Нужно декомпозировать простыню кода на отдельные части.
    Ответ написан
  • Почему дублируются байты, которые не влезли в размер буффера, при получении с помощью socket.recv?

    Vindicar
    @Vindicar
    RTFM!
    try:
                                data = self.sock.recv(constants.BUF_SIZE)
                            except socket.timeout:
                                message_ended = True
    
                            buff += data

    Если отработает except, предыдущее значение data не изменится, и к buff оно ВСЁ РАВНО приклеится.
    Закинь buff += data в ветку try-else.
    Ответ написан
    Комментировать
  • Json ключ value не парсится?

    Vindicar
    @Vindicar
    RTFM!
    Ну наверно нужно не цикл вкладывать в try, а наоборот?
    Ответ написан
  • Можно ли прервать input()?

    Vindicar
    @Vindicar
    RTFM!
    Нет, стандартный input() этого не предусматривает.
    Но есть сторонние варианты, например, pytimedinput.
    Ответ написан
    Комментировать
  • Как реализовать остановку цикла for?

    Vindicar
    @Vindicar
    RTFM!
    Если тебе нужна проверка на останов после каждой операции - придётся-таки эту проверку производить после каждой операции. Да, можно схитрить, но от этого не уйдёшь.

    Я могу посоветовать убрать time.sleep() и заменить их на threading.Event.wait() с таймаутом - в этом случае, если Event будет взведено, то не придётся "досыпать". Но проверять исход ожидания всё равно придётся самому.

    Альтернативное, но ОЧЕНЬ кардинальное решение - сделать код асинхронным. Причина в том, что у асинхронных подпрограмм, завёрнутых в Task, есть механизм отмены, позволяющий принудительно выкинуть исключение в ходе выполнения любой длительной операции (любого await-вызова). По сути, это та же самая проверка - но встроенная.

    Это работает, так как асинхронный код выполняется в одном потоке. Провернуть такой же фокус, скажем, в многопоточном приложении - нетривиально и довольно рискованно.
    Ответ написан
    2 комментария
  • Почему не работает IF-ELSE?

    Vindicar
    @Vindicar
    RTFM!
    1. Как проверял? Просто пытался угадать? Рандом - штука такая, могло просто не везти.
    2. Если пользователь с первого раза введёт правильный key, условие цикла сразу будет False, и цикл не выполнится - а значит, не выполнятся и ifы.

    Я бы посоветовал сделать так:
    max_tries = 5  # число попыток
    key = -1  # заведомо некорректное значение
    for i in range(1, max_tries+1):
        key = int(input(f'Попытка №{i}: '))
        if key == lock:
            print(f'Угадал c {i}-й попытки!')
            break
    else:  # этот else относится к for! 
        # он выполнится, если for НЕ БЫЛ прерван по break
        print('Попытки кончились. =(')
    Ответ написан
    1 комментарий
  • Как инициализировать декоратор в классе, а внутри использовать этот декоратор?

    Vindicar
    @Vindicar
    RTFM!
    Ты хочешь поместить хэндлеры в класс?
    Я в таких случаях делаю немного иначе.
    Делаю свой декоратор, который принимает те же параметры, что и ботовый, но просто сохраняет их в отдельном атрибуте декорируемого метода. Благо методу можно создать новый атрибут через setattr() или простым присваиванием.
    При конструировании экземпляра класса через dir() перечисляю содержимое класса, ищу методы, среди них ищу методы с моим атрибутом (т.е. те, которые были декорированы). Для каждого такого метода получаю bound method (через getattr(self, method_name)) и вызываю оригинальный декоратор на нём с сохранёнными параметрами. Ведь декоратор - это функция, его можно вызывать как функцию.
    Часть, связанную с конструированием, можно спрятать или в родительском абстрактном классе, или в метаклассе, чтобы не повторять для каждого класса с обработчиками.
    Ответ написан
    Комментировать
  • Как можно сделать виртуальную «флешку», которая будет отображаться в проводнике windows 10?

    Vindicar
    @Vindicar
    RTFM!
    Вариант А. Написать и установить драйвер виртуальной файловой системы. Разумеется, не на питоне.

    Вариант Б. Создать сетевой ресурс (можно на localhost, т.е. "как бы" сетевой), поддерживающий протокол WebDAV или иной, поддерживаемый виндой (как делает та же Samba) и примонтировать его через системную команду net use.
    Это если нужен именно виртуальный диск.

    Вариант В. Есть ещё Windows Shell Namespace Extension для создания виртуальных папок. Вроде как не рекомендуется.

    Вариант Г. Cloud Sync Engine.

    Я бы сказал, вариант Б наиболее реализуем именно на питоне. Плюс он более кроссплатформенный, и в принципе можно разместить скрипт не на локальной машине, а в сети.
    Ответ написан
    Комментировать
  • Ошибка can't concat list to bytes?

    Vindicar
    @Vindicar
    RTFM!
    Проверяй, что творится в user[0]. Какие значения, какие типы данных, и т.п.
    Ответ написан
  • В чем может быть ошибка?

    Vindicar
    @Vindicar
    RTFM!
    Вроде обрабатываю исключение на остановку программы

    Нет, не обрабатываешь. Ты ловишь Exception, а исключение KeyboardInterrupt наследуется не от него, а от BaseException - как раз, чтобы его случайно не поймать, когда не надо. Так что KeyboardInterrupt лучше ловить отдельно.

    Если тебе нужно гарантированно выполнить код при завершении работы программы, используй try...finally. В крайнем случае - модуль atexit.
    Ответ написан
  • Как правильно обрабатывать сообщения пользователя телеграмм боту?

    Vindicar
    @Vindicar
    RTFM!
    Ищи инфу по finite state machine (FSM) и как она реализуется в pyTelegramBotAPI.

    Если коротко - для каждого пользователя нужно хранить его текущий шаг (начало, выбор предмета, выбор задания, и т.п.) и ассоциированные с этим шагом данные. Т.е. нужно хранилище вида ключ-значение. Ключом будет ID пользователя.
    В качестве такого хранилища можно использовать и обычный словарь (если допустимо, чтобы бот всё забывал при перезапуске), или базу данных.

    И да, ты спалил токен бота. Меняй его теперь.
    Ответ написан
    1 комментарий
  • 'Handler' object is not callable как решить?

    Vindicar
    @Vindicar
    RTFM!
    callback_query_handlers
    Может, всё-таки callback_query_handler?
    Ответ написан
    Комментировать
  • Почему пайтон не видит глобальную переменную?

    Vindicar
    @Vindicar
    RTFM!
    1. Судя по отступам, у тебя get_answer() заканчивается на строке answer = input("Введите слово: ")
    2. Прежде чем упоминать переменную как глобал, она должна быть глобально определена, т.е. ей должно быть присвоено значение вне функции.
    3. Нафига тут вообще глобальная переменная?
    Ответ написан
  • Как цифру в строке преобразовать в числовой формат?

    Vindicar
    @Vindicar
    RTFM!
    Затем все приводится к формату строки типа ключ=значение

    Как можно значение оставить типа float?


    Во-первых, определись, о чем всё-таки речь - о формате, или о типе данных? Это разные вещи.
    Если о формате, что читай, как форматировать значения в f-строках.
    Если о типе, то ты просишь превратить число в строку, но оставить его числом. Т.е. взаимоисключающие вещи.
    "Нарисуйте семь перпендикулярных красных линий, но из них три синие должны быть параллельны, а ещё две - прозрачные."
    Ответ написан
    4 комментария
  • Как запустить выполнение асинхронной функции с определенной частотой выполнения?

    Vindicar
    @Vindicar
    RTFM!
    Храни в asyncio.Queue очередь запросов к API. Отдельная задача пусть выбирает запросы из очереди, отправляет, получает ответ и оповещает о результате. Например, так.
    import asyncio
    import typing
    
    class ThrottledResource:
        def __init__(self, delay: float):
            self._delay = delay
            self._queue = asyncio.Queue()
            self._task = None
        
        def start(self):
            self._task = asyncio.create_task(self._work_loop)
        
        def stop(self):
            self._task.cancel()
            self._task = None
    
        # этот метод вызывается клиентским кодом, получает параметры и возвращает отклик спустя время.
        async def query(self, params):
            future = asyncio.Future()  # Future просигналит, когда наш запрос будет обслужен
            await self._queue.put((future, params))
            result = await future  # корутина спит, пока запрос не обслужат
            return result
    
        async def _work_loop(self):
            while True:
                future, params = await. self._queue.get()  # ждем, пока не придёт запрос
                try:
                    result = await call_api(params)  # тут делаем асинхронное обращение к сервису
                except Exception as err:
                    future.set_exception(err)  # была ошибка - теперь await future выкинет исключение
                else:
                    future.set_result(result)  # полуен результат - await future вернёт его
                self._queue.task_done()  # каждому успешному get() соответствует task_done()
                asyncio.sleep(self._delay)  # можно учесть, сколько времени делался запрос. Но стоит ли?

    Код примерный, но идею передаёт. Использоваться будет как-то так
    api = ThrottledResource(delay=1.0)
    api.start()
    ...
    result = await api.query(params)  # await подождёт, пока не дойдёт очередь до нашего запроса


    Нужно добавить обработку ошибок, корректное завершение работы при наличии задач в очереди, и так далее.
    Технически вместо класса можно было реализовать это всё в виде декоратора над replier(), но это уже на вкус и цвет.
    Ответ написан
    6 комментариев
  • Почему пайчарм даёт ошибку, но когда делаю то, что он просит опять ошибка?

    Vindicar
    @Vindicar
    RTFM!
    @bot.callback_query_handler(func=lambda call: True)
    def work(call):


    Не надо описывать обработчики внутри других обработчиков. Это не будет работать так, как ты этого ожидаешь. Однажды прописанный, обработчик остаётся до конца работы программы.
    И уж ТОЧНО не нужно делать самому вызов work(call).
    Ответ написан