8-(927)-977-80-70 web-i-seo@yandex.ru
Режим работы: 10-00 до 20-00 МСК

Вы нашли нас по запросу -"Сравнение методов объединения двух отсортированных списков в Python Прокопьевск" - это лучшая рекомендация для подрядчика SEO продвижения в городе Прокопьевск или по России!

Сравнение методов объединения двух отсортированных списков в Python

Пусть у нас есть два списка (для простоты из целых чисел), каждый из которых отсортирован. Хотим объединить их в один список, который тоже должен быть отсортирован. Эта задача наверняка всем знакома, используется, например, при сортировке слиянием.

 

 

Способов реализации (особенно на python) достаточно много. Давайте разберем некоторые из них и сравним затрачиваемое время на разных входных данных.

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

 

Входные данные не меняются

Пусть есть два списка list1 и list2.

Начнем с самого простого алгоритма: обозначим метки за i и j и будем брать меньший из list1[i]list2[j] и увеличивать его метку на единицу, пока одна из меток не выйдет за границу списка.

При первом сравнении мы выберем минимальный элемент из двух минимальных в своем списке и подвинемся на следующий элемент, поэтому наименьший элемент из двух списков будет стоять на нулевом месте результирующего. Дальше несложно по индукции доказать, что далее слияние пройдет верно.

Перейдем к коду:

def simple_merge(list1, list2):
    i, j = 0, 0
    res = []
    while i < len(list1) and j < len(list2):
        if list1[i] < list2[j]:
            res.append(list1[i])
            i += 1
        else:
            res.append(list2[j])
            j += 1
    res += list1[i:]
    res += list2[j:] 
    # один из list1[i:] и list2[j:] будет уже пустой, поэтому добавится только нужный остаток
    return res

 

Заметим, что в данном коде используется только перемещение вперед по списку. Поэтому будет достаточно работать с итераторами. Перепишем алгоритм с помощью итераторов.

 

Еще изменим обработку концов списков, так как теперь мы не умеем копировать сразу до конца. Будем обрабатывать элементы до того, когда оба итератора дойдут до конца, при этом, если один уже оказался в конце, будем просто брать из второго.

 

def iter_merge(list1, list2):
    result, it1, it2 = [], iter(list1), iter(list2)
    el1 = next(it1, None)
    el2 = next(it2, None)
    while el1 is not None or el2 is not None:
        if el1 is None or (el2 is not None and el2 < el1):
            result.append(el2)
            el2 = next(it2, None)
        else:
            result.append(el1)
            el1 = next(it1, None)
    return result

 

В этой реализации можно вместо добавления по одному элементу (result.append()) собрать генератор, а потом из него получить список. Для этого напишем отдельную функцию, которая будет строить генератор, а основная функция сделает из него список.

 

def gen_merge_inner(it1, it2):
    el1 = next(it1, None)
    el2 = next(it2, None)
    while el1 is not None or el2 is not None:
        if el1 is None or (el2 is not None and el2 < el1):
            yield el2
            el2 = next(it2, None)
        else:
            yield el1
            el1 = next(it1, None)

def gen_merge(list1, list2):
    return list(gen_merge_inner(iter(list1), iter(list2))) # из генератора получаем список

 

Встроенные реализации

Рассмотрим еще несколько способов слияния через встроенные в python функции.

  • merge из heapq. Как говорит документация, эта функция делает именно то, что мы хотим, и больше: объединяет несколько итерируемых объекта, можно задать ключ, можно сортировать в обратном порядке.
    Тогда нам нужно просто импортировать и использовать:

    from heapq import merge
    
    def heapq_merge(list1, list2):
        return list(merge(list1, list2)) # тоже возвращает генератор
  • Counter из collectionsCounter умеет считать количество вхождений каждого из элементов, выдавать их в тех количествах, в которых они входят, и еще несколько полезных вещей, которые сейчас не нужны (например, несколько самых часто встречающихся элементов).
    Воспользуемся gen_merge_inner для слияния элементов Counter(list1) и Counter(list2):

    def counter_merge(list1, list2):
        return list(gen_merge_inner(Counter(list1).elements(), Counter(list2).elements()))
  • И, наконец, просто сортировка. Объединяем и сортируем заново. Тут есть два варианта реализация через sort() и sorted(). Сразу сравним их:
list1 = [i for i in range(1, 200000, 3)]
list2 = [i for i in range(2, 250000, 4)]
%timeit res1 = sorted(list1 + list2)
%timeit res2 = list1 + list2; res2.sort()
6.73 ms ± 64.9 µs per loop (mean ± std. dev. of 7 runs, 100 loops each)
4.43 ms ± 38.4 µs per loop (mean ± std. dev. of 7 runs, 100 loops each)

В результате:

    def sort_merge(list1, list2):
        return (list1 + list2).sort()

Если можно менять исходные списки

 

Предположим, что после слияния старые списки больше не нужны (как обычно и случается). Тогда можно написать еще один способ. Будем как и раньше сравнивать нулевые элементы списков и вызывать pop(0) у списка с меньшим, пока один из списков не закончится.

 

def pop_merge(list1, list2):
    result = []
    while list1 and list2:
        result.append((list1 if list1[0] < list2[0] else list2).pop(0))
    return result + list1 + list2

 

Получили простенькую функцию на 4 строчки, но использовать дальше исходные списки не получится. Можно их скопировать, потом работать с копиями, но это потребует много дополнительного времени. Здесь будут проблемы с тем, что удаление нулевого элемента очень дорогое. Поэтому еще одна модификация будет заключаться в том, что мы будем вместо удаления из начала списка использовать удаление из конца, но придется в конце развернуть списки.

 

def reverse_pop_merge(list1, list2):
    result = []
    while list1 and list2:
        result.append((list1 if list1[-1] > list2[-1] else list2).pop(-1))
    return (result + list1[-1::-1] + list2[-1::-1])[-1::-1]

 

Сравнение

 

Пора перейти к самому интересному.
Составим список функций, которые будем сравнивать:

 

  • simple_merge
  • iter_merge
  • gen_merge
  • heapq_merge
  • counter_merge
  • sort_merge
  • pop_merge
  • reverse_pop_merge

 

Будем измерять время работы с помощью модуля timeit. Код можно посмотреть здесь.

 

Разберем несколько ситуаций: оба списка примерно одинакового размера, один список большой, а второй маленький, количество вариантов элементов большое, количество вариантов маленькое. Кроме этого проведем просто общий случайный тест.

Тест первый

 

Проведем общий тест, размеры от $1$ до $10^5$, элементы от $1$ до $10^6$.

 

Отдельно сравним pop и reverse_pop:

 

 

pop_merge тратит колоссально больше времени в общем случае, как и ожидалось.

 

Не будем учитывать здесь огромный pop_merge, чтобы лучше видеть разницу между другими:

 

 

reverse_pop_merge показал себя относительно неплохо по сравнению с ручной реализацией и heapq_merge.

 

Методы на итераторах работают еще быстрее, при этом видно, что получилось выгоднее построить генератор, чем добавлять элементы в список.

 

Тест второй, сравнимые размеры

 

Размеры будут принадлежать отрезку $[50x, 50(x+1))$, а $x$ увеличиваем, начиная с $1$. Шаг $50$.

 

 

Как уже можно видеть pop_merge при небольшом размере списков еще ведет себя как heapq_merge, а дальше обгоняет всех.

 

Тест третий, один маленький, второй большой

 

Размер первого равен $x$, размер второго $10^4 + 100x$.

 

 

В самом начале (на очень маленьких списках) reverse_pop_merge обгоняет всех, кроме sort_merge, но на чуть больших все выходит на стандартные позиции.

 

Тест четвертый, много повторных

 

Размеры фиксированы, а количество элементов увеличивается на $5$, начиная с $1$.

 

 

Как видно, на достаточно малых количествах counter_merge оказывается быстрее reverse_pop_merge и heapq_merge, но потом он отстает.

 

Чемпионы

Абсолютным победителем оказался sort_merge! Гораздо быстрее просто отсортировать список заново, чем использовать вроде бы линейные от длины списков функции.

На втором месте в подавляющем большинстве случаев идет gen_merge, за ним следует iter_merge.

Остальные методы используют еще больше времени, но некоторые в каких-то крайних случаях достигают результатов 2-3 мест.

Дата изменения


Индивидуальный Предприниматель Ознобин Р.А.
8-927-977-80-70
Адрес: г. Прокопьевск, ул. Строителей, строение 12

Полезная информация по теме - Сравнение методов объединения двух отсортированных списков в Python Прокопьевск

заказать создание сайта Прокопьевск

Откройте для своего дела новый рынок on-line торговли, увеличьте продажи и сделайте более доступными Ваши товары и сервис. У нас заказать создание сайта Прокопьевск  можно любым удобным способом, мы разработаем архитектуру портала, запрограммируем этот шаблон , качество которого определит внешний вид и расположение всех необходимых элементов, таких как, текст, изображения, видео, баннеры и реклама. Для нас создание Вашего портала это отлаженный процесс, мы предлагаем сайтподключ на выбранном домене, к примеру — com, ru и др. Разработаем Вам качественный сайт в короткие сроки. заказать создание сайта  Прокопьевск под ключ сэкономит ваши силы, время и финансы, при этом даст новый канал продаж и лицо Вашей компании во всемирной паутине. Чтобы купить сайт Прокопьевск по самым выгодным ценам необходимо определить задачу такого портала, его бюджет, и поставщика этой сервис. Вы можете ознакомиться с примерами наших работ, а так же нашими тарифными планами и обязательно обратите...

Интернет-магазин студии цветов Прокопьевск

Запустили Интернет магазин — Студия цветов «Подсолнух», занимается оформлением и доставкой букетов и цветочных композиций. Заказчик заказал on-line магазин с обязательной возможностью делать заказ и оставлять комментарии от посетителей. Сравнение методов объединения двух отсортированных списков в Python — получи СКИДКУ 10% Общие расходы составили: Базовая цена составила — 9 600 руб. ! наполнение материалами — от 300 до 1000 руб. за страницу  (в зависимости от наличия и объёмов таблиц, фото и текстов). визуальных эффектов — 5 000 руб.. доски объявлений — 20 000 руб. Итого: 34 600 рубле + наполнение + тех поддержка + время на организацию и ведение рекламных компаний + затраты на SEO раскрутка. Что бы заказать вебсайт у нас, вам надо лишь отправить заявку с данного сайта или связаться с нами любым из перечисленных в разделе Контакты...

стоимость создания сайта Прокопьевск

Определите выгодную для себя стоимость разработки сайта Прокопьевск которая будет соответствовать целям Вашего дела. Мы готовы предложить Вам от простой визитки компании до сложного корпоративного портала, с внутренними CRM системами и системами обмена данных. Готовы разработать продающий on-line-магазин под ключ или витрину Вашей компании. В стоимость разработки водит ряд действий и сервиса по обслуживанию портала в дальнейшем. Для начала мы подберем для Вас наименование или доменное имя портала. Вы выберете наиболее подходящий и понравившийся Вам домен и при создании стоимость сайтподключ будет включать в себя домен в зоне  com, ru и т.д. Создание сайта стоимость Прокопьевск будет включать в себя оплату домена, хостинга где разместиться портал, настройка и размещение платформы, правка платформы под требование и условия Вашего дела. Возможно добавить в перечень дизайнерское исполнение портала или оставить стоимость разработки сайта Прокопьевск под ключ где Вам будут предложены...

Веб-мастерская про услуги ремонта и строительства Прокопьевск

Создан новый сайт. Заказчик — частный предприниматель из Самары решил заказать вебсайт с перечнем усилий по ремонту и возможностью добавлять фото своих работ. Так же решили сразу заказать раскрутку по запросам о ремонте Самаре. В соответствии со стратегией и выбранными для оптимизации запросами, была составлена карта содержания и прописан мета-движок и сбалансированно содержание. Ведутся постоянные усилия по улучшению позиции в поисковых комплексах по целевым запросам клиента. Сравнение методов объединения двух отсортированных списков в Python — получи СКИДКУ 10% Базовая цена составила — 18 500 руб.. предварительного анализа целевых запросов и составления карты + 7000 руб.. Для того что бы заказать сайт у нас, вам надо лишь отправить заявку нам на почте или связаться с нами любым из перечисленных в разделе Контакты методов, мы свяжемся с Вами и поможем определится с...

стоимость разработки сайта Прокопьевск

стоимость разработки сайта Прокопьевск зависит от сложности исполнения портала, его индивидуальности и функционала, но мы готовы предложить Вам создание и раскрутка по выгодным ценам в зависимости от поставленной задачи. Если Вы задумались заказать создание портала, то надо знать, что стоимость разработки сайта Прокопьевск не всегда гарантирует качество исполнения портала. Основным критерием выбора разработчика должен быть его опыт в этой области. Узнайте сколько будет стоить создание портала под ключ, которая в первую очередь зависит от цели Вашего  дела. Если к примеру on-line визитка будет минимальной по цене, потому как обычно это требует меньше времени работ. Мы создаем качественные сайты визитки за короткий срок. Ознакомьтесь с нашими тарифами разработки и адаптации сайтов, у нас есть выгодные акции и скидки! Закажите у нас корпоративный сайт, цена разработки которого будет дешевле при комплексе работ по производству и оптимизации, при этом мы берем на себя сопровождение,...

8 основных структур данных в Python Прокопьевск

Структуры данных — это структуры кода для хранения и организации данных, которые упрощают изменение, навигацию и доступ к информации. Структуры данных определяют способ сбора данных, функциональные возможности, которые мы можем реализовать, и отношения между данными. Структуры данных используются практически во всех областях информатики и программирования, от операционных систем до интерфейсной разработки и машинного обучения. Структуры данных помогают: Управляйте большими наборами данных и используйте их. Быстрый поиск определённых данных в базе данных. Создавайте чёткие иерархические или реляционные связи между точками данных. Упростите и ускорьте обработку данных. Структуры данных являются жизненно важными строительными блоками для эффективного решения реальных проблем. Структуры данных — это проверенные и оптимизированные инструменты, которые дают вам удобную основу для организации ваших программ. В конце концов, вам не нужно переделывать колесо (или конструкцию) каждый раз,...

Раскрутка инстаграмма Прокопьевск

Если Вы хотите сделать свой аккаунт или аккаунт в Инстаграмме Вашего дела популярным, предлагаем выбрать Вам один из тарифов раскрутка инстаграм Прокопьевск, который поможет достигнуть Вашей цели. Основная доля пользователей подзапросов запроса раскрутка инстаграм Прокопьевск интересуется запросом раскрутка инстагра Прокопьевск, мы же хотим продолжить эту усилийу,в  действительно выгодный и главное правильный портал такой,  как раскрутка инстограмма, раскрутка канала Ютюб или раскрутка группы ВК. ...

продвижение сайта в топ Прокопьевск

Не зависимо от региона и масштаба Вашего дела, Вам необходимо раскрутка сайта в топ Прокопьевск поисковых систем. И.П. Ознобин Р.А. осуществит качественное оказание сервис в области адаптации информации Вашей продукции или усилий. Сравнение методов объединения двух отсортированных списков в Python — получи СКИДКУ 10% Раскрутка сайта в топ Прокопьевск включает в себя несколько обязательных шагов: Выделение и анализ целевой аудитории в web сети, оценка потенциала спроса и конкурентной стратегии оптимизации и раскрутки сайта Подбор ключевых запросов, по которым люди ищут Ваши товары и сервис. На основе анализа составляется семантическое ядро сайта, которое необходимо для правильной ориентации в выдаче поисковых систем. Происходит техническая перенастройка сайта под новую выработанную стратегию адаптации в топ Прокопьевск, отладка возможных ошибок, которые могут мешать...