Задать вопрос
Ответы пользователя по тегу Python
  • Как запустить функцию в отдельном потоке второй раз?

    Vindicar
    @Vindicar
    RTFM!
    1.
    "RuntimeError: threads can only be started once"
    Можно ли это как-то обойти понятным для начинающего в python способом?

    Можно перестать хотеть странного и принять как данность, что завершившийся поток - завершился. С ним уже ничего не сделаешь. Это так на уровне ОС, если что. Так что создавай экземпляр Thread тогда, когда надо запустить поток.
    2.
    Может можно как-то по-другому стартовать, без доп функции?

    И как ты это себе представляешь? Tkinter должен знать, какой твой код выполнить при нажатии кнопки. Единственный способ это сделать - передать ему функцию или иной callable object. Да, можно использовать лямбду, но в лямбду много не уместишь, так что лучше функция.
    А когда придёшь к оборачиванию GUI в классы, поймёшь, что передача функции/метода - наиболее простой и практичный подход.
    Ответ написан
    1 комментарий
  • Как сделать exe файл в python?

    Vindicar
    @Vindicar
    RTFM!
    Для начала можешь переименовать питоновский файл в pyw - это означает запуск без окна консоли.
    А далее, разбираешься с pyinstaller, а лучше сразу с его обёрткой auto-yp-to-exe, там есть опция для оконных приложений.
    Файлы данных внутрь exe я бы не вносил, они всё равно будут распакованы во временный каталог при запуске, так что только запуск замедлишь. Но если очень хочется, такая опция тоже есть, читай доки на вышеупомянутое.
    Ответ написан
    Комментировать
  • Как убрать ошибку кода в exe?

    Vindicar
    @Vindicar
    RTFM!
    1. почитать внимательно про auto-py-to-exe и осознать, что он базируется на pyinstaller
    2. почитать, как указать pyinstaller модуль, импорта которого он не увидел (hidden import)
    3. найти, где в интерфейсе auto-py-to-exe можно указать дополнительные опции для pyinstaller (подсказка: секция Advanced)
    4. Вписать в нужный параметр имя comtypes (возможно, придётся указать именно comtypes.stream)
    Ответ написан
    8 комментариев
  • Как исправить код?

    Vindicar
    @Vindicar
    RTFM!
    Ну ты уже вот тут глупость сделал.
    self.layer_1 = nn.Linear(inp1, inp2)  # Почему inp2 ?!

    Первый параметр Linear - число входов слоя. Для первого слоя должно совпадать с числом значений для данного входа сети. Второй параметр - число нейронов, т.е. число выходов слоя. Должно совпадать с числом входов следующего слоя сети.
    Вообще правило простое - сколько на выходе одного слоя, столько на входе следующего. Так как concatenate тупо дописывает данные одного входа в конец другого, то для него сумма длин входов должна быть равна выходу (а выход - входу следующего).

    Например, в твоём случае на входе первого Linear должно быть inp1 нейронов, а на выходе - 72 - inp2 нейронов. После concatenate получится (72 - inp2) + inp2 = 72 нейрона - ровно столько, сколько на входе второго Linear.

    Код исправишь сам.
    Ответ написан
    Комментировать
  • Как сделать подсчет скорости нажатий на пробел?

    Vindicar
    @Vindicar
    RTFM!
    Одним голым питоном - трудно, потому что консольный ввод буферизуется. Условно, я зажму пробел, а программа будет видеть как я ввожу один пробел за другим. Да и интервал между нажатиями теряется.
    Тебе нужно ловить пары событий нажатие-отпускание. Ты можешь попробовать использовать tkinter (он в питоне доступен сразу), создать окно, и привязать к нему обработчики соответствующих событий. Причём нужна именно пара событий <KeyPress> и <KeyPress>, а не просто <Key>.
    А реакция на эти события... зависит от того, что тебе нужно. Например, заводишь переменную-счётчик, при отпускании клавиши наращиваешь его на 1, а раз в секунду (в ткинтере есть метод .after(), он хорошо подходит для задач вида "сделай вот это через заданное время") выводишь этот счётчик пользователю, и тут же сбрасываешь его в ноль.
    Ответ написан
    Комментировать
  • Как решать такие задачи?

    Vindicar
    @Vindicar
    RTFM!
    Размер выхода слоя равен размеру входа следующего слоя.
    Вот как узнать размер выхода последнего слоя - я фз, тут недостаточно данных.

    EDIT: ой, блин, я невнимательный. Не увидел, что слои идут не 1->2->3->4, а 4->3->4.
    Тогда ещё проще, выход 4 должен совпадать по размеру со входом 3, а вход 4 - с выходом 3.
    Ответ написан
    2 комментария
  • Что использовать в качестве workerА для отложенных задач?

    Vindicar
    @Vindicar
    RTFM!
    Я бы вообще изобрёл велосипед, если честно.
    У тебя есть список или база активных (неслучившихся) событий. Пусть это будет пара ID-метка времени, остальная инфа нам без надобности.
    При запуске бота выбираем из списка ближайшее событие, и вычисляем время ожидания (с небольшим запасом) и спим в корутине. В случае, если ожидание прервано раньше - повторяем то же самое, чтобы либо "доспать", либо переключиться на ожидание более близкого события.
    Если же ожидание закончилось успешно, обрабатываем событие и снова повторяем поиск ближайшего события.
    Если список событий изменился, прерываем текущее ожидание, чтобы корутина ожидания могла адаптироваться к изменениям.

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

    Vindicar
    @Vindicar
    RTFM!
    Если Normalize может работать нормально с uint8 - то преобразования типа не нужны.
    Если не может - то никакой другой порядок и не заработает.
    Ответ написан
    Комментировать
  • Как в openai обойти ошибку openai.PermissionDeniedError?

    Vindicar
    @Vindicar
    RTFM!
    Сделать так, чтобы openai не понял, что вы в России.
    Как хотите, но ваш туннель должен терминироваться на зарубежном хосте, и только потом идти к ним.
    Либо ставьте бота туда, либо поднимайте там прокси с авторизацией, либо городите VPN...
    Ответ написан
  • Как сжать и закодировать данные так, чтобы их можно было отправить в теле запроса на сервер?

    Vindicar
    @Vindicar
    RTFM!
    У тебя различие появляется в том, что gzip.compress() даёт не тот же результат, что был подан в gzip.decompress().
    Видимо, разные реализации алгоритма gzip. Но если данные успешно разжимаются, то не всё ли равно?

    EDIT: я посмотрел инфу о заголовке Gzip. Создать идентичный заголовок несложно, но проблема идёт дальше: сами данные непохожи. Я раскопал, как получить "голый" deflate-поток без заголовков, и попытался сравнить его с тем, что идёт после gzip-заголовка в твоих данных. В твоём случае заголовок минималистичный, поэтому занимает всего 10 байт, а дальше идёт deflate-поток. Он успешно разжимается в те же данные, которые даёт gzip, так что я вроде не ошибся.

    Но попытка перебора вариантов сжатия так, чтобы получить хоть что-то похожее, результатов не дала.
    Вот мой код, увы, он выводит Match not found.
    import binascii, gzip, zlib, sys
    
    def deflate(data, compresslevel=9, strategy=0, wbits=zlib.MAX_WBITS, memlevel=zlib.DEF_MEM_LEVEL):
        compress = zlib.compressobj(
                compresslevel,        # level: 0-9
                zlib.DEFLATED,        # method: must be DEFLATED
                -wbits,               # window size in bits:
                                      #   -15..-8: negate, suppress header
                                      #   8..15: normal
                                      #   16..30: subtract 16, gzip header
                memlevel,             # mem level: 1..8/9
                strategy              # strategy:
                                      #   0 = Z_DEFAULT_STRATEGY
                                      #   1 = Z_FILTERED
                                      #   2 = Z_HUFFMAN_ONLY
                                      #   3 = Z_RLE
                                      #   4 = Z_FIXED
        )
        deflated = compress.compress(data)
        deflated += compress.flush()
        return deflated
    
    def inflate(data):
        decompress = zlib.decompressobj(
                -zlib.MAX_WBITS  # see above
        )
        inflated = decompress.decompress(data)
        inflated += decompress.flush()
        return inflated
    
    
    input_data = b"H4sIAAAAAAAA_2SNvU7EMBAG32V7To7txMleB-JKWugsc-wllhI78k8iQLw7sgPVdauZHX0GFX5H7BHMumpnFoJzRIGw03u5muYwG4VovStIIaiT4OLE6gNDuE7GOZq1_SiEdwiZ2wf59Rjy8-vLdnm67Ref_e1t_BxrI2uTtMlp8qF2FlkxLcI1h0Au6RzpX_VStQ1X8li3iZY_IQbVDl3HWX-OOCBEmuf76ucXAAD__wEAAP__grfYVeYAAAA"
    gzipped_data = binascii.a2b_base64(input_data.replace(b"_", b"/").replace(b"-", b"+") + b"=")
    raw_data = gzip.decompress(gzipped_data)
    print('RAW', raw_data)
    
    deflate_stream = gzipped_data[10:]
    inflated = inflate(deflate_stream)
    print('DEFLATED DATA', deflate_stream)
    print('RESTORED DATA', inflated)
    print('Restored data matches original:', inflated == raw_data)
    
    for level in range(1, 10):
        for strategy in range(0, 5):
            for wbits in range(9, 16):
                for memlevel in range(1, 10):
                    test = deflate(raw_data, level, strategy, wbits, memlevel)
                    if deflate_stream.startswith(test) or test.startswith(deflate_stream):
                        print(level, strategy, wbits, memlevel)
                        sys.exit(0)
    else:
        print('No match found')
    Ответ написан
    1 комментарий
  • Задание из практикума, не могу сделать уже целую неделю. Какие есть ошибки?

    Vindicar
    @Vindicar
    RTFM!
    У тебя как минимум один косяк в expire().
    for title, batches in items.items():
        for batch in batches:
            if batch['expiration_date'] and batch['expiration_date'] < today:
                expiring_items.append({'title': title, 'expiration_date': batch['expiration_date']})
                break  # <--- ???

    Ты находишь первую "протухшую" партию для товара, и сразу же переходишь к следующему товару. А если у одного товара несколько "протухших" партий?
    Ответ написан
    Комментировать
  • Появилась проблема при запуске приложения Python в Termux?

    Vindicar
    @Vindicar
    RTFM!
    Если нужен красивый визуал в командной строке под никсами, может, стоит использовать модуль curses?
    Ответ написан
  • Где и как в Pycharm можно проанализировать код на вызовы или использование не объявленных методов\полей?

    Vindicar
    @Vindicar
    RTFM!
    Ide подсвечивает такие опечатки, но реально на них можно напороться только при исполнение кода

    Про линтеры выше написали, а я добавлю: PyCharm позволяет просмотреть список всех наденных проблем (errors, warnings) в проекте. Если он не пустой - значит, надо чинить.
    Ответ написан
  • Бот на python/telebot не приветствует новых пользователей, что не так?

    Vindicar
    @Vindicar
    RTFM!
    А почему ты вызываешь bot.polling()? Это для асинхронных ботов, а твой синхронный. Используй infinity_polling().
    Вообще, читай официальные примеры, вроде вот этого, и сравнивай со своим кодом.
    Ответ написан
  • Как исключить ввод пустой строки?

    Vindicar
    @Vindicar
    RTFM!
    Тебе нужно повторять ввод аргументов, пока пользователь не введёт их правильно? Ну вот и повод почитать про циклы.
    while True:  # бесконечный цикл
      x_str = input('Введите x:')  # вводим число
      try:  # внутри блока try мы будем отлавливать ошибки - исключения
        x = float(x_str)  # пробуем преобразовать строку в число
      except ValueError:  # в случае чего float() выкинет исключение ValueError
        print(x_str, 'это не число. Попробуйте ещё раз.')  # сообщаем пользователю
      else:  # если ошибки не было
        break  #  то прерываем цикл, у нас в x лежит нужное значение
      # мы делаем break только в ветке else - значит, пока отрабатывает ветка except, цикл продолжится

    В принципе, если тебе нужна только одна попытка, ты можешь вытащить блок try-except из цикла, убрать ветку else, и заменить ветку except на что-то типа:
    except ValueError:
      print(x_str, 'это не число. У вас есть ещё одна попытка')  # сообщаем пользователю
      x_str = input('Введите x: ')
      x = float(x_str)

    Тогда получится обойтись без циклов.

    Но без try-except обойтись сложнее. Да, ты можешь предварительно проверить строку на соответствие шаблону десятичного дробного числа, но честно? try-except куда проще, так что стоит его освоить. Главное - не "глотай" ошибки, реагируй на них.
    Ответ написан
    Комментировать
  • Как подсветить функцию как устаревшую?

    Vindicar
    @Vindicar
    RTFM!
    66a00bc093f98796053667.png
    У меня всё работает. Смотри настройки IDE, в частности, File->Settings->Editor->Inspections, ищешь "deprecate".
    Ответ написан
  • Питон,нейросеть, алгоритм выхода из лабиринта?

    Vindicar
    @Vindicar
    RTFM!
    Общая идея задачи. как я её понял: дрон с воздуха снимает лабиринт, затем алгоритм сшивает кадры в одно изображение, извлекает форму лабиринта и строит путь по нему. Так? Если так, это должно было быть в вопросе.

    У тебя три чётко видимых шага.
    1. Сборка карты лабиринта по видео с дрона. Я бы для начала использовал поиск локальных особенностей (SIFT, углы по Харрису, или что-то столь же простое).
    Вопрос 1
    Что использовать - сильно зависит от условий съёмки. Чем более контрастные элементы присутствуют на изображении, тем лучше.

    Они обычно не позволяют работать в реальном времени, но это для задачи и не требуется. Затем, зная позиции особенностей на каждом изображении, сравниваем их с "соседними" изображениями.
    Вопрос 2
    Тут будет большой вопрос, касающийся определения соседних изображений. Простой ответ: считаем соседними N последних кадров, но тут могут быть проблемы с длинной цепочкой кадров. Например, если дрон сделает круг, не факт, что на карте этот круг сомкнётся. Может потребоваться какая-то оценка позиции снимка.

    По итогам сравнения мы получим проективное преобразование, описывающее, как надо повернуть/наклонить/растянуть/etc один кадр, чтобы совпали общие с соседним кадром части. Подвергнув кадр такому преобразованию, мы сможем все кадры привести к одной плоскости.
    Вопрос 3
    Скорее всего, возможные преобразования будут на нескольких кадрах, так что придётся их как-то ранжировать, и "усреднять" несколько наилучших преобразований с ближайшими соседями

    Тогда постепенно построите одно изображение-карту, а уж дальше его можно будет обрабатывать обычными средствами. Там, пороговое преобразование или что для выделения "стен", определение границ лабиринта, и дальше уже тупой flood-fill для поиска пути.
    Ответ написан
    1 комментарий
  • Почему выводит ошибку Traceback (most recent call last) при сохранении json словаря в переменную?

    Vindicar
    @Vindicar
    RTFM!
    Читаем и осознаём текст ошибки. UnicodeDecodeError - ошибка декодирования строки байт в юникод-строку, т.е. косяк с кодировками. В списке traceback ищем последний элемент, который ссылается на твой код.
    File "C:\Users\Maks\PycharmProjects\pythonProject7\main.py", line 57, in check_news_update
        news_dict = json.load(file)

    Т.е. косяк при чтении json из файла. При вызове load() кодировку нельзя указать, так что проверяем, откуда приходит файл.
    with open('news_dict.json') as file:  # где encoding? где режим открытия?
            news_dict = json.load(file)

    Вот и ошибка - не указана кодировка (и я бы явно указал режим открытия 'rt'). Причём при сохранении не забыл ведь указать...
    with open("news_dict.json", "w",encoding="utf-8") as file:  # а тут всё правильно
                json.dump(news_dict, file, indent=4, ensure_ascii=False)
    Ответ написан
    Комментировать
  • Как получать информацию о подарках в прямых эфирах TikTok?

    Vindicar
    @Vindicar
    RTFM!
    Вот трудно вбить в гугл "API tiktok gift"?
    Вторая же ссылка ведёт на гитхаб подходящей под описание библиотеки. А там и примеры есть...
    Вторая ссылка, Карл!
    Ответ написан
    1 комментарий
  • Не могу решить задание ЕГЭ. Почему мой код не работает?

    Vindicar
    @Vindicar
    RTFM!
    Тебе выше правильно намекнули.
    Во-первых, нужно увидеть простую вещь: дописывание нуля в конец двоичного представления - это то же, что умножение на 2. Назовём это операция А. А дописывание единицы - то же, что умножение на 2 и прибавление 1. Назовём это операция Б. Таким образом, можно избавиться от двоичной системы в задании.

    Второе: операция А будет давать в результате только чётные числа, независимо от аргумента. Аналогично, операция Б будет давать только нечётные числа, независимо от аргумента.

    Третье: результат как операции А, так и операции Б всегда больше аргумента. Более того, больший аргумент даёт больший результат. Это легко показать.
    Пусть у нас есть число x. Результат операции A меньше, чем операции Б над тем же числом, поэтому мы должны проверить только один сценарий: когда к x применяется операция А, а к x - 1 - операция Б. Значит, x станет 2 * x, а x - 1 станет 2 * (x - 1) + 1 = 2 * x - 1. Т.е. для x - 1 результат меньше. Т.е. даже в таком, самом спорном случае, меньшее число даёт меньший результат

    Четвёртое: задачу можно решать задом наперёд. У нас есть целевые числа - значит, мы должны применить к ним алгоритм наоборот, чтобы получить исходные. Это и даст нам диапазон для поиска исходных чисел.

    Дальше уже проще. Смотрим на начало целевого диапазона, число 876 544. Оно чётное, значит, оно могло быть создано только операцией А. Значит, чтобы получить исходное число, нужно поделить его на 2. 876 544 / 2 = 438 272. Это число тоже чётное, оно тоже могло быть создано только операцией А. 438 272/ 2 = 219 136. Оно тоже чётное. Значит, третий раз применяем операцию А наоборот: 219 136 / 2 = 109 568. Исходя из пункта 3, можем сказать, что числа, меньше 109 568, не могут дать результат, больший или равный 876 544.

    Аналогично анализируем верхнюю границу целевого диапазона.
    1 234 567 899 = 2 * 617 283 949 + 1 (операция Б)
    617 283 949 = 2 * 308 641 974 + 1 (операция Б)
    308 641 974 = 2 * 154 320 987 (операция А)
    Значит, исходное число должно быть не более 154 320 987. Вот тебе и диапазон для поиска.
    А поскольку большие аргументы дают большие результаты, то это означает, что ни одно число меньше 154 320 987 не даст результат, который выйдет за верхнюю границу диапазона (1 234 567 899). Т.е. искомые числа - это весь диапазон от 109 568 до 154 320 987. Посчитать, сколько чисел в диапазоне, тривиально.
    Ответ написан
    Комментировать