Is it possible to find the second maximum number from an array of integers by traversing the array only once?
As an example, I have a array of five integers from which I want to find second maximum number. Here is an attempt I gave in the interview:
#define MIN -1
int main()
{
int max=MIN,second_max=MIN;
int arr[6]={0,1,2,3,4,5};
for(int i=0;i<5;i++){
cout<<"::"<<arr[i];
}
for(int i=0;i<5;i++){
if(arr[i]>max){
second_max=max;
max=arr[i];
}
}
cout<<endl<<"Second Max:"<<second_max;
int i;
cin>>i;
return 0;
}
The interviewer, however, came up with the test case int arr[6]={5,4,3,2,1,0};
, which prevents it from going to the if
condition the second time.
I said to the interviewer that the only way would be to parse the array two times (two for
loops). Does anybody have a better solution?
codaddict
443k81 gold badges491 silver badges528 bronze badges
asked Mar 6, 2010 at 14:03
2
Your initialization of max
and second_max
to -1
is flawed. What if the array has values like {-2,-3,-4}
?
What you can do instead is to take the first 2 elements of the array (assuming the array has at least 2 elements), compare them, assign the smaller one to second_max
and the larger one to max
:
if(arr[0] > arr[1]) {
second_max = arr[1];
max = arr[0];
} else {
second_max = arr[0];
max = arr[1];
}
Then start comparing from the 3rd element and update max
and/or second_max
as needed:
for(int i = 2; i < arr_len; i++){
// use >= n not just > as max and second_max can hav same value. Ex:{1,2,3,3}
if(arr[i] >= max){
second_max=max;
max=arr[i];
}
else if(arr[i] > second_max){
second_max=arr[i];
}
}
answered Mar 6, 2010 at 14:15
codaddictcodaddict
443k81 gold badges491 silver badges528 bronze badges
4
The easiest solution would be to use std::nth_element
.
answered Mar 6, 2010 at 14:06
avakaravakar
31.8k9 gold badges65 silver badges102 bronze badges
9
You need a second test:
for(int i=0;i<5;i++){
if(arr[i]>max){
second_max=max;
max=arr[i];
}
else if (arr[i] > second_max && arr[i] != max){
second_max = arr[i];
}
}
answered Mar 6, 2010 at 14:08
Anders AbelAnders Abel
67.7k17 gold badges150 silver badges217 bronze badges
3
Your original code is okay, you just have to initialize the max and second_max variables. Use the first two elements in the array.
answered Mar 6, 2010 at 14:34
Hans PassantHans Passant
918k145 gold badges1680 silver badges2521 bronze badges
Here you are:
std::pair<int, int> GetTwoBiggestNumbers(const std::vector<int>& array)
{
std::pair<int, int> biggest;
biggest.first = std::max(array[0], array[1]); // Biggest of the first two.
biggest.second = std::min(array[0], array[1]); // Smallest of the first two.
// Continue with the third.
for(std::vector<int>::const_iterator it = array.begin() + 2;
it != array.end();
++it)
{
if(*it > biggest.first)
{
biggest.second = biggest.first;
biggest.first = *it;
}
else if(*it > biggest.second)
{
biggest.second = *it;
}
}
return biggest;
}
answered Mar 6, 2010 at 14:14
Johann GerellJohann Gerell
24.9k10 gold badges72 silver badges122 bronze badges
7
Quickselect is the way to go with this one. Pseudo code is available at that link so I shall just explain the overall algorithm:
QuickSelect for kth largest number:
Select a pivot element
Split array around pivot
If (k < new pivot index)
perform quickselect on left hand sub array
else if (k > new pivot index)
perform quickselect on right hand sub array (make sure to offset k by size of lefthand array + 1)
else
return pivot
This is quite obviously based on the good old quicksort algorithm.
Following this algorithm through, always selecting element zero as the pivot every time:
select 4th largest number:
1) array = {1, 3, 2, 7, 11, 0, -4}
partition with 1 as pivot
{0, -4, _1_, 3, 2, 7, 11}
4 > 2 (new pivot index) so...
2) Select 1st (4 - 3) largest number from right sub array
array = {3, 2, 7, 11}
partition with 3 as pivot
{2, _3_, 7, 11}
1 < 2 (new pivot index) so...
3) select 1st largest number from left sub array
array = {2}
4) Done, 4th largest number is 2
This will leave your array in an undefined order afterwards, it’s up to you if that’s a problem.
answered Mar 6, 2010 at 18:09
MartinMartin
12.4k12 gold badges64 silver badges128 bronze badges
Step 1. Decide on first two numbers.
Step 2. Loop through remaining numbers.
Step 3. Maintain latest maximum and second maximum.
Step 4. When updating second maximum, be aware that you are not making maximum and second maximum equal.
Tested for sorted input (ascending and descending), random input, input having duplicates, works fine.
#include <iostream>
#define MAX 50
int GetSecondMaximum(int* data, unsigned int size)
{
int max, secmax;
// Decide on first two numbers
if (data[0] > data[1])
{
max = data[0];
secmax = data[1];
}
else
{
secmax = data[0];
max = data[1];
}
// Loop through remaining numbers
for (unsigned int i = 2; i < size; ++i)
{
if (data[i] > max)
{
secmax = max;
max = data[i];
}
else if (data[i] > secmax && data[i] != max/*removes duplicate problem*/)
secmax = data[i];
}
return secmax;
}
int main()
{
int data[MAX];
// Fill with random integers
for (unsigned int i = 0; i < MAX; ++i)
{
data[i] = rand() % MAX;
std::cout << "[" << data[i] << "] "; // Display input
}
std::cout << std::endl << std::endl;
// Find second maximum
int nSecondMax = GetSecondMaximum(data, MAX);
// Display output
std::cout << "Second Maximum = " << nSecondMax << std::endl;
// Wait for user input
std::cin.get();
return 0;
}
answered Mar 6, 2010 at 18:39
Rajendra UppalRajendra Uppal
18.9k15 gold badges59 silver badges57 bronze badges
Other way to solve this problem, is to use comparisons among the elements. Like for example,
a[10] = {1,2,3,4,5,6,7,8,9,10}
Compare 1,2 and say max = 2 and second max = 1
Now compare 3 and 4 and compare the greatest of them with max.
if element > max
second max = max
element = max
else if element > second max
second max = element
The advantage with this is, you are eliminating two numbers in just two comparisons.
Let me know, if you have any problem understanding this.
answered Mar 7, 2010 at 6:19
BooleanBoolean
14.2k30 gold badges88 silver badges129 bronze badges
1
Check this solution.
max1 = a[0];
max2 = a[1];
for (i = 1; i < n; i++)
{
if (max1 < a[i])
{
max2 = max1;
max1 = a[i];
}
if (max2 == max1) max2 = a[i + 1];
if (max2 == a[n])
{
printf("All numbers are the same no second max.n");
return 0;
}
if (max2 < a[i] && max1 != a[i]) max2 = a[i];
}
Mario S
11.7k24 gold badges38 silver badges47 bronze badges
answered Oct 16, 2011 at 20:11
mittamitta
111 bronze badge
1
Here is something which may work ,
public static int secondLargest(int[] a){
int max=0;
int secondMax=0;
for(int i=0;i<a.length;i++){
if(a[i]<max){
if(a[i]>secondMax){
secondMax=a[i];
}
continue;
}
if(a[i]>max){
secondMax=max;
max=a[i];
}
}
return secondMax;
}
Kevin
53.3k15 gold badges99 silver badges130 bronze badges
answered Nov 26, 2011 at 18:03
The upper bound should have be n+log2n−2, but it bigger than O(n) in case of random selection algorithm, but in worst case it much smaller. The solution might be
-
build a tree like to find the MAX element with n – 1 comparisons
max(N)
/
max(N/2) max(N/2) -
remove the MAX and find the MAX again log2n – 1 comparison
PS. It uses additional memory, but it faster than random selection algorithm in worst case.
answered Apr 5, 2012 at 11:47
Can’t we just sort this in decreasing order and take the 2nd element from the sorted array?
answered Oct 5, 2012 at 0:16
jhonjhon
872 silver badges10 bronze badges
How about the following below.
make_heap is O(n) so this is efficient and this is 1-pass
We find the second max by taking advantage that it must be one of the heap children of the parent, which had the maximum.
#include <algorithm>
#include <iostream>
int main()
{
int arr[6]={0,1,2,3,4,5};
std::make_heap(arr, arr+6);
std::cout << "First Max: " << arr[0] << 'n';
std::cout << "Second Max: " << std::max(arr[1], arr[2]) << 'n';
return 0;
}
answered Jan 7, 2013 at 17:39
SJHoweSJHowe
7565 silver badges11 bronze badges
int max,secondMax;
max=secondMax=array[0];
for(int i=0;i<array.length;i++)
{ if(array[i]>max) { max=array[i]; }
if(array[i]>secondMax && array[i]<max) {
secondMax=array[i]; }
}
answered Apr 22, 2013 at 9:48
#include <iostream>
using namespace std;
int main() {
int max = 0;
int sec_Max = 0;
int array[] = {81,70,6,78,54,77,7,78};
int loopcount = sizeof(array)/sizeof(int);
for(int i = 0 ; i < loopcount ; ++i)
{
if(array[i]>max)
{
sec_Max = max;
max = array[i];
}
if(array[i] > sec_Max && array[i] < max)
{
sec_Max = array[i];
}
}
cout<<"Max:" << max << " Second Max: "<<sec_Max<<endl;
return 0;
}
coyotte508
9,0256 gold badges43 silver badges63 bronze badges
answered Jun 5, 2016 at 11:18
BunnyBunny
11 bronze badge
1
// Set the first two different numbers as the maximum and second maximum numbers
int max = array[0];
int i = 1;
//n is the amount of numbers
while (array[i] == max && i < n) i++;
int sec_max = array[i];
if( max < sec_max ) {
tmp = sec_max;
sec_max = max;
max = tmp;
}
//find the second maximum number
for( ; i < n; ++i ) {
if( array[i] > max ) {
sec_max = max;
max = array[i];
} else if( array[i] > sec_max && array[i] != max ) {
sec_max = array[i];
}
}
printf("The second maximum number is %dn", sec_max);
Flexo♦
86.9k22 gold badges190 silver badges272 bronze badges
answered Sep 24, 2011 at 12:13
tianyatianya
931 silver badge4 bronze badges
1
Можете просто примерно написать как думать, а то вообще идей нет
задан 6 окт 2019 в 6:26
2
first_max = int(input())
second_max = int(input())
if first_max < second_max:
first_max, second_max = second_max, first_max
element = int(input())
while element != 0:
if element > first_max:
second_max, first_max = first_max, element
elif element > second_max:
second_max = element
element = int(input())
print(second_max)
ответ дан 6 окт 2019 в 6:27
Андрей КрузликАндрей Крузлик
1,2633 золотых знака11 серебряных знаков17 бронзовых знаков
Пожалуй эффективнее всего будет воспользоваться функцией heapq.nlargest():
from heapq import nlargest
res = nlargest(2, items)[1]
ответ дан 6 окт 2019 в 7:14
Можно написать функцию:
def find_maxes(array, count):
# копируем список чтобы не изменить старую
copied_array = array.copy()
maximums = []
if count > len(copied_array):
raise ValueError('Количество не может превышать длину списка')
for _ in range(count):
max_val = max(copied_array) # получаем максимальное значение
copied_array.remove(copied_array) # удаляем его из списка
maximums.append(max_val) # добавляем в наш ожидаемый результат
return maximums
или же можно поступить хитро
def find_maxes(array, count):
if count > len(array):
raise ValueError('Количество не может превышать длину списка')
sorted_array = sorted(array) # отсортировать список
# Забрать последние элементы из спика так как они будут максимальными
return sorted_array[len(array)-count: len(array)]
ответ дан 6 окт 2019 в 6:44
E1mirE1mir
1,89811 серебряных знаков23 бронзовых знака
2
b=[3,5,6,7,7,7]
print(list(set(b))[-2])
функция set позволит создать множество отсортированных по возрастанию отличных друг от друга чисел, функция list позволит создать список и обратиться к предпоследнему (или -2) элементу.
<<6
ответ дан 29 окт 2020 в 21:37
FeToRFeToR
12 бронзовых знака
Skip to content
Задача «Второй максимум»
Условие
Последовательность состоит из различных натуральных чисел и завершается числом 0. Определите значение второго по величине элемента в этой последовательности. Гарантируется, что в последовательности есть хотя бы два элемента.
Решение задачи от разработчиков на Python:
Другая реализация задачи на Python:
Смотреть видео — Задача «Второй максимум» решение на Python
Делитесь с друзьями ссылкой на ответ и задавайте вопросы в комментариях! 👇
Related Posts
Какой алгоритм для поиска максимума в случайном массиве использовать? В статье собрано 5 эффективных must-have алгоритмов.
Если хочешь подтянуть свои знания, загляни на наш курс «Алгоритмы и структуры данных», на котором ты:
- углубишься в решение практических задач;
- узнаешь все про сложные алгоритмы, сортировки, сжатие данных и многое другое.
Ты также будешь на связи с преподавателем и другими студентами.
В итоге будешь браться за сложные проекты и повысишь чек за свою работу 🙂
Интересно, хочу попробовать
Самый быстрый и простейший алгоритм
Следует пояснить, что под “быстротой” алгоритма будет подразумеваться его асимптотическая сложность, а не физическое время выполнения.
В действительности, единственный способ точно найти самое большое число в случайном массиве будет перебор каждого числа в поисках максимума. Поэтому сложность такого алгоритма – O(N). Они называются линейными.
Простейшая реализация алгоритма на С:
#include <stdio.h> int main() { int array[100]; int maximum, size, index = 0; printf("Введите длину массива (не больше 100)n"); scanf("%d", &size); printf("Введите %d целых чисел массиваn", size); for (int i = 0; i < size; i++) scanf("%d", &array[i]); maximum = array[0]; // За изначальный максимум берем первый элемент массива for (int i = 1; i < size; i++) { if (array[i] > maximum) { maximum = array[i]; index = i; } } printf("Максимум находится в позиции %d и его значение %d.n", index, maximum); return 0; }
А как насчет распараллеливания?
После прочтения абзаца выше многие могут предложить поиск максимума с помощью параллельных вычислений, потому что он будет быстрее. И это так, если мы говорим о физическом времени. Но несмотря на то, сколько у вас есть процессов, вам всё равно придется просмотреть каждый элемент массива. Сложность алгоритма остается такой же, O(N).
Рассмотрим ситуацию на примере. Есть 200 карточек, на каждой из которых случайное число. Ваша задача найти карточку с самым большим числом. Допустим, друг согласился помочь, забрав половину карточек. Теперь каждый из вас ищет максимум из 100 карточек, вместо 200. Как только вы оба закончите, останется сравнить максимумы обеих половин. Время сократилось вдвое, но просмотреть пришлось все 200 карточек.
Сама по себе задача является так называемой чрезвычайно параллельной задачей, что ещё раз подтверждает возможность её решения именно распараллеливанием, без ущерба для конечного результата.
Задача о разборчивой невесте
Есть ещё один вариант нахождения максимума в случайном массиве.
Идем по первой половине всех значений в массиве. Запоминаем самое большое. Назовём его Х. Далее проходим по второй половине. Когда (и если) находим значение, которое больше Х, останавливаемся. Обозначим его Y. Какова вероятность того, что Y – максимум в массиве?
Чтобы это утверждение было истинным, Y должно находиться во второй половине массива. А так как значения в массиве случайны, шанс такого = 50%. Тогда если второй максимум присутствует в первой половине, это гарантирует, что найден правильный максимум.
Вероятность, что второй максимум в первой половине также = 50%. Посему вероятность, что и второй максимум в первой половине, и первый максимум во второй половине, равна 25%.
В итоге, такой алгоритм гарантирует точность 25% в нахождении максимума в массиве случайных чисел. Можно улучшить алгоритм, если искать второй максимум не в первой половине (1/2), а в 1/e (основание натурального логарифма) части массива. Этот способ впервые был предложен для решения задачи о разборчивой невесте или задачи секретаря (The Secretary Problem).
Аналоговый алгоритм
Спагетти-сортировка – прекрасный аналоговый алгоритм для решения задачи нахождения максимума в массиве.
Представим, что в массиве только N целых чисел. Возьмите N не приготовленных палочек спагетти, длина каждой сопоставляется с единственным значением в массиве.
Соберите спагетти в руку. Аккуратно, чтобы не сломать, поставьте горсть на ровную поверхность. В результате выше всех будет видна самая длинная (максимум) соломинка.
Асимптотическая сложность такого алгоритма – O(1). Но в цифровом мире он не применим.
Квантовый алгоритм
Алгоритм Гровера (или схема Гровера) используется в квантовых вычислениях для решения задач перебора. С его помощью сложность поиска максимума уменьшается до O(sqrt(N)) (большая О от корня N).
Данный способ решения может быть применен только на квантовом компьютере, что сильно умаляет его полезность в современном мире, но его нельзя не упомянуть.
Источник
Хотите дополнить? Ждём ваши предложения в комментариях 😉
Дан целочисленный массив… Второй максимум…
Задание:
Дан целочисленный массив из 30 элементов. Элементы массива могут принимать произвольные целые значения. Опишите на русском языке или на одном из языков программирования алгоритм, который находит и выводит второй максимум массива (элемент, который в отсортированном по невозрастанию массиве стоял бы вторым). Исходные данные объявлены так, как показано ниже. Запрещается использовать переменные, не описанные ниже, но разрешается не использовать часть из них.
Паскаль |
const N=30; |
Си |
#include<stdio.h> |
Естественный язык |
Объявляем массив A из 30 элементов. Объяв-ляем целочисленные переменные i, k, max, max2. В цикле от 1 до 30 вводим элементы массива A с 1-го по 30-й. … |
Решение:
Сложность в том, что нужно найти не максимальный элемент, а второй по величине. Можно, конечно, сначала найти максимум, а потом искать следующий за ним, но можно сделать это за один проход по массиву. Нам нужны две переменные, max (максимальный элемент) и max2 (второй максимум). Сначала выбираем максимальный из первых двух элементов и записываем его значение в max, а второй по величине записываем в max2:
if a[1] > a[2] then begin
max:=a[1]; max2:=a[2];
end
else begin
max:=a[2]; max2:=a[1];
end;
Затем в цикле перебираем все элементы, начиная с 3-го (первые два уже «задействованы»!) до последнего, 30-ого. Если очередной элемент a[i] больше, чем max, записываем значение max в max2 (предыдущий максимум становится вторым), а значение a[i] – в max. Иначе, если a[i] больше, чем max2, записываем значение a[i] в max2. После завершения цикла выводим значение переменной max2. Вот решение на Паскале
const N=30;
var a: array [1..N] of integer;
i, k, max, max2: integer;
begin
for i:=1 to N do readln(a[i]);
if a[1] > a[2] then begin
max:=a[1]; max2:=a[2]
end
else begin
max:=a[2]; max2:=a[1]
end;
for i:=3 to N do
if a[i] > max then begin
max2 := max;
max := a[i]
end
else if a[i] > max2 then max2 := a[i];
writeln(max2)
end.