Форум программистов, компьютерный форум, киберфорум
С++ для начинающих
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
 
Рейтинг 4.91/22: Рейтинг темы: голосов - 22, средняя оценка - 4.91
0 / 0 / 0
Регистрация: 15.06.2012
Сообщений: 21

В неубывающей последовательности целых чисел найти количество пар чисел с заданной разностью "D"

15.06.2012, 12:31. Показов 4948. Ответов 30
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Помогите пожалуйста оптимизировать алгоритм, тут приведен простой перебор и на большом тесте программа работает очень долго. По заданию время работы программы должно не превышать 2 секунд.
Заранее большое спасибо!
C++ (Qt)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
int _tmain(int argc, _TCHAR* argv[])
{
    typedef vector<double> MyVector;
    MyVector vec;
    FILE *pFile1 = fopen("input.txt", "r");
    FILE *pFile2 = fopen("output.txt", "w");
    int ch;
    if (!pFile1)
    {
        printf("File opening error\n");
        return 1;
    }
    else
    {
        int n, d, x, support, diff;
        int counter = 0;
        fscanf(pFile1, "%d", &n);
        fscanf(pFile1, "%d", &d);
        for (int i = 1; i <= n; i++)
        {
            fscanf(pFile1, "%d", &x);
            vec.push_back(x);
        }
        for (int i = 1; i < vec.size(); i++)
        {   
            support = i;
            while (support >= 0)
            {
                diff = vec[i] - vec[support];
                if (diff == d)
                {
                    counter++;
                }
                support = support - 1;
            }
        }
        fprintf(pFile2, "%d", counter);
    }
    fclose(pFile1);
    fclose(pFile2);
    return 0;
}
0
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
15.06.2012, 12:31
Ответы с готовыми решениями:

В заданной последовательности целых чисел найти количество чисел кратных заданному
Напишите программу, которая в последовательности целых чисел определяет количество чисел, кратных 5 или 7. Программа получает на вход целые...

Определить в заданной последовательности целых чисел количество чисел Фибоначчи
Выполнить задания, если задана последовательность целых чисел длиной n. Определить в заданной последовательности целых чисел количество...

Определить в заданной последовательности целых чисел количество чисел Фибоначчи
Определить в заданной последовательности целых чисел количество чисел Фибоначчи.

30
Форумчанин
Эксперт CЭксперт С++
 Аватар для MrGluck
8216 / 5047 / 1437
Регистрация: 29.11.2010
Сообщений: 13,453
15.06.2012, 17:10
Смотри как можно:
1. Отсортировать всю последовательность
2. Сравниваем число со следующим:
2.1 Если текущее число + D меньше, то сравниваем со след. числом с последов.
2.2 Если равно, то счетчик++, сравниваем со след. числом.
2.3 Если число + D больше, то берем след. число.
1
0 / 0 / 0
Регистрация: 15.06.2012
Сообщений: 21
15.06.2012, 17:46  [ТС]
Цитата Сообщение от MrGluck Посмотреть сообщение
Смотри как можно:
1. Отсортировать всю последовательность
2. Сравниваем число со следующим:
2.1 Если текущее число + D меньше, то сравниваем со след. числом с последов.
2.2 Если равно, то счетчик++, сравниваем со след. числом.
2.3 Если число + D больше, то берем след. число.
А можешь реализовать этот алгоритм? кстати последовательность уже отсортирована, вот полное задание: Задана неубывающая последовательность целых чисел. Найти количество пар чисел с заданной разностью D.
Ввод из файла INPUT.TXT. В первой строке задаются через пробел два целых числа: количество членов последовательности N и разность D (2 ≤ N ≤ 106, 1 ≤ D ≤ 106). Во второй строке вводятся через пробел N натуральных чисел последовательности A1 ≤ A2 ≤ ...≤ AN (Ai ≤ 109).
Вывод в файл OUTPUT.TXT. Вывести число пар (Ai, Aj) таких, что Ai – A j = D.
Ограничения. Время работы на одном тесте до 2 с. Объем используемой памяти до 1 Мгб.
Пример
Ввод
10 7
3 5 12 18 25 40 47 47 48 49
Вывод
4
0
Эксперт С++
 Аватар для valeriikozlov
4728 / 2549 / 757
Регистрация: 18.08.2009
Сообщений: 4,568
16.06.2012, 05:23
MilitaNt, Вам поможет двоичный поиск.
C++ (Qt)
1
2
3
4
5
6
for (int i = 1; i < vec.size(); i++)
{
    t=vec[i]+d;
    // и вот с этого места ищите в диапазоне от i+1 до vec.size()-1 элемент со значением t
    // как только его нашли, то влево и вправо просматриваете и считаете общее количество элементов со значением t, увеличивая при этом значение  counter
}
Если значение d может быть отрицательным, то сразу после считывания этого значения умножаете его на -1 (результат все равно посчитает правильно).
Если значение d может быть равным 0, то можно сделать так:
C++ (Qt)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
for (int i = 1; i < vec.size(); i++)
{
    j=i+1;
    while(j<vec.size() &&  vec[i]==vec[j])
    {
        j++;
    }
    if(j>i+1)
    {
        t=j;
        while(t>i+1)
        {
            counter+=t-i+1;
            t--;
        }
        i=j-1;
    }
}
1
0 / 0 / 0
Регистрация: 15.06.2012
Сообщений: 21
16.06.2012, 09:55  [ТС]
valeriikozlov, большое спасибо!
а не могли бы вы этот алгоритм полностью реализовать, d в диапазоне от 1 до 10^6
заранее спасибо!)
0
 Аватар для dima koz
23 / 17 / 7
Регистрация: 05.06.2012
Сообщений: 72
Записей в блоге: 5
16.06.2012, 10:02
stdafx.h:

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
// stdafx.h: включаемый файл для стандартных системных включаемых файлов
// или включаемых файлов для конкретного проекта, которые часто используются, но
// не часто изменяются
//
 
#pragma once
 
#include "targetver.h"
 
#include <stdio.h>
#include <tchar.h>
 
#include <time.h>
#include <stdlib.h>
#include <iostream>
#include <conio.h>
 
#include <windows.h>
sequence.cpp:

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
// sequence.cpp: определяет точку входа для консольного приложения.
//
//Ввод из файла INPUT.TXT.
//В первой строке задаются через пробел два целых числа: количество членов последовательности N и разность D (2 <= N <= 106, 1 <= D <= 106).
//Во второй строке вводятся через пробел N натуральных чисел последовательности A1 <= A2 <= ...<= AN (Ai <= 109).
 
// Вывод в файл OUTPUT.TXT. Вывести число пар (Ai, Aj) таких, что Ai – A j = D.
// Ограничения. Время работы на одном тесте до 2 с. Объем используемой памяти до 1 Мгб.
// Пример
// Ввод
// 10 7
// 3 5 12 18 25 40 47 47 48 49
// Вывод
// 4
 
#include "stdafx.h"
 
using namespace std;
 
void recordingOriginalFile(); //заполнение/запись исходного файла
int getRandomInt(unsigned short int,unsigned short int); //получение рандомного числа типа Int
 
 
int _tmain(int argc, _TCHAR* argv[])
{
    //при  первом запуске программы, для программного создания файла для задания
    //при его наличии можно не использовать
    //написал, т.к. влом записывать файл вручную
    //recordingOriginalFile();
 
    //чтение input.txt
    //заполнение переменных заданными значениями
    int n,d;
 
    int FirstTime = GetTickCount();
    
    FILE *pFile1;
    fopen_s(&pFile1,"d:\\test\\input.txt", "r");
    fscanf_s(pFile1, "%d", &n);
    fscanf_s(pFile1, "%d", &d);
    int *arr = new int [n];
    for (int i = 0; i < n; i++)
    {
            fscanf_s(pFile1, "%d", &arr[i]);
    }
    fclose(pFile1);
 
    //поиск пар
    int count = 0; // счетчик пар
    for (int i = 0; i < n; i++)
    {
 
        for (int j = 0; j<n ; j++)
        {
            if (arr[i] - arr[j] == d) 
                count ++;
 
        }
    }
    count /=2 ;
    int SecondTime = GetTickCount();
    cout << "count ="<< count <<endl;
    cout << "runtime this programm = "<< SecondTime - FirstTime << " ms (1 s = 1000 ms)"<<endl;
    _getch();
    return 0;
}
 
//функция записывает входной файл с  заданными параметрами по условиям задачи
//
void recordingOriginalFile()
{
    const int n = 106; //количество членов последовательности N
    const int d = 11;   //разность D
 
    int original [n], hold;
    //заполняем массив размера n рандомно числами от 2 до n
 
    srand((unsigned int)time(NULL));
    bool sorted;
    for (int i=0;i<n;i++)
    original[i] = getRandomInt(2,n);
    // сортируем 
    for (int j = 1; j < n; j++)
        {
        sorted = true;
        for (int k = 0; k < n - 1; k++)
            if (original[k] > original[k + 1])
            {
                hold = original[k];
                original[k] = original[k + 1];
                original[k + 1] = hold;
                sorted = false;
            }
        if (sorted) 
            break;
        }   
 
    //записываем
    FILE *pFileOriginalRecording;
 
    fopen_s(&pFileOriginalRecording,"d:\\test\\input.txt", "w");
    fprintf(pFileOriginalRecording,"%d ",n);
    fprintf(pFileOriginalRecording,"%d ",d);
    fprintf(pFileOriginalRecording,"%c",char(10));
 
    for (int i = 0; i < n; i++)
    {
        fprintf(pFileOriginalRecording,"%d ",original[i]);
    }
    fclose(pFileOriginalRecording);
}
 
int getRandomInt(unsigned short int initialValue,unsigned short int finalValue)
{
    
    return  (initialValue + rand() % (finalValue-initialValue+1));
}
время выполнения несколько мс при N = 106, занимает в памяти не больше 700 кб

запись счетчика пар, думаю, сможешь самостоятельно сделать, по образцу в recordingOriginalFile()
1
0 / 0 / 0
Регистрация: 15.06.2012
Сообщений: 21
16.06.2012, 10:20  [ТС]
dima koz, а для чего сортировка? последовательность по заданию вроде как неубывающая => отсортирована, или я не прав?
0
 Аватар для dima koz
23 / 17 / 7
Регистрация: 05.06.2012
Сообщений: 72
Записей в блоге: 5
16.06.2012, 10:26
Цитата Сообщение от MilitaNt Посмотреть сообщение
dima koz, а для чего сортировка? последовательность по заданию вроде как неубывающая => отсортирована, или я не прав?
прав, они не нужны.
эти две функции(генерируют исходный файл для задания - input.txt) после мэйна я сделал только для того,чтобы ручками самому не создавать файл, ведь ты его к заданию не прикладывал, а у меня его нет.

+ не забудь сделать запись счетчика в output.txt
1
0 / 0 / 0
Регистрация: 15.06.2012
Сообщений: 21
16.06.2012, 10:31  [ТС]
dima koz а можешь проверить как долго твоя программа будет работать на тесте с последовательностью в 100000 элементов?
0
 Аватар для dima koz
23 / 17 / 7
Регистрация: 05.06.2012
Сообщений: 72
Записей в блоге: 5
16.06.2012, 10:43
Цитата Сообщение от MilitaNt Посмотреть сообщение
dima koz а можешь проверить как долго твоя программа будет работать на тесте с последовательностью в 100000 элементов?
Если предоставишь файл input.txt с последовательностью в 100000 элементов, то почему нет
а так самому его долго генерировать, сортировка выбрана самая медленная и простая

но N = 100000 противоречит условию твоего же собственного задания :
Цитата Сообщение от MilitaNt Посмотреть сообщение
количество членов последовательности N и разность D (2 ≤ N ≤ 106, 1 ≤ D ≤ 106).

проверил, при N=100001
ms - 12730
KБ - 1036
тестовый файл с последовательностью(input.txt) получился 554 КБ
1
0 / 0 / 0
Регистрация: 15.06.2012
Сообщений: 21
16.06.2012, 10:52  [ТС]
dima koz, ))) там было 10^6, ^ я обозначил степень)))) я уже протестировал) на такой последовательности работает 22 секунды...а нужно не более 2(((((
еще предложили двоичный поиск, но я не знаю как его реализовать
C++
1
2
3
4
5
6
for (int i = 1; i < vec.size(); i++)
{
    t=vec[i]+d;
    // и вот с этого места ищите в диапазоне от i+1 до vec.size()-1 элемент со значением t
    // как только его нашли, то влево и вправо просматриваете и считаете общее количество элементов со значением t, увеличивая при этом значение  counter
}
Добавлено через 3 минуты
dima koz, вот генератор теста)
C++
1
2
3
4
5
6
7
int _tmain(int argc, _TCHAR* argv[])
{
    FILE *pFile2 = fopen("output.txt", "w");
    for (int i = 1; i <= 100000; i++)
        fprintf(pFile2, "%d ", i);
    return 0;
}
0
 Аватар для dima koz
23 / 17 / 7
Регистрация: 05.06.2012
Сообщений: 72
Записей в блоге: 5
16.06.2012, 10:56
Цитата Сообщение от MilitaNt Посмотреть сообщение
dima koz, ))) там было 10^6, ^ я обозначил степень)))) я уже протестировал) на такой последовательности работает 22 секунды...а нужно не более 2(((((
еще предложили двоичный поиск, но я не знаю как его реализовать
при N = 10 ^6 нужен совершенно другой алгоритм поиска пар,чтобы уложиться за 2 секунды.
если найдешь такой алгоритм, возможно, помогу реализовать

Цитата Сообщение от MilitaNt Посмотреть сообщение
dima kozвот генератор теста)
у меня уже готовый файлик
1
0 / 0 / 0
Регистрация: 15.06.2012
Сообщений: 21
16.06.2012, 11:00  [ТС]
dima koz, так вот я и сам не знаю какой алгоритм нужен, ну вот я выше показал двоичный поиск, он ускорит работу?
C++
1
2
3
4
5
6
for (int i = 1; i < vec.size(); i++)
{
    t=vec[i]+d;
    // и вот с этого места ищите в диапазоне от i+1 до vec.size()-1 элемент со значением t
    // как только его нашли, то влево и вправо просматриваете и считаете общее количество элементов со значением t, увеличивая при этом значение  counter
}
Добавлено через 3 минуты
0
Эксперт С++
 Аватар для valeriikozlov
4728 / 2549 / 757
Регистрация: 18.08.2009
Сообщений: 4,568
16.06.2012, 11:20
Цитата Сообщение от MilitaNt Посмотреть сообщение
а не могли бы вы этот алгоритм полностью реализовать, d в диапазоне от 1 до 10^6
мог бы, но ответьте на несколько вопросов:
- какой нужен язык программы (Вы пишите на С++ в разделе Паскаля)?
- максимальное количество элементов массива?
- значения элементов массива в каком диапазоне?
1
0 / 0 / 0
Регистрация: 15.06.2012
Сообщений: 21
16.06.2012, 11:27  [ТС]
можно и на паскале и на с++, дело в том, что изначально я не видел что пишу на форуме паскаля)))
2 <= n <= 10^6 , 1 <= d <= 10^6
значения любые, но с условием что последовательность неубывающая (поэтому думаю сортировка уже не нужна)
вот простой генератор теста для данной задачи
C++
1
2
3
4
5
6
7
int _tmain(int argc, _TCHAR* argv[])
{
    FILE *pFile2 = fopen("output.txt", "w");
    for (int i = 1; i <= 100000; i++)
        fprintf(pFile2, "%d ", i);
    return 0;
}
0
0 / 0 / 0
Регистрация: 15.06.2012
Сообщений: 21
16.06.2012, 11:38  [ТС]
КонецСвета, да конечно)
0
Эксперт С++
 Аватар для valeriikozlov
4728 / 2549 / 757
Регистрация: 18.08.2009
Сообщений: 4,568
16.06.2012, 11:53
пробуйте:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
#include <iostream>
#include <vector>
using namespace std;
typedef vector<double> MyVector;
MyVector vec;
 
int f(int a, int b, int t)
{
    if(b-a<2)
    {
        if(vec[a]==t)
            return a;
        if(vec[b]==t)
            return b;
        return -1;
    }
    int tmp=(b+a)/2;
    if(vec[tmp]>t)
        return f(a,tmp,t);
    else
        return f(tmp,b,t);
}
 
int _tmain(int argc, _TCHAR* argv[])
{
 
    FILE *pFile1 = fopen("input.txt", "r");
    FILE *pFile2 = fopen("output.txt", "w");
    int ch;
    if (!pFile1)
    {
        printf("File opening error\n");
        return 1;
    }
    else
    {
        int n, d, x, t, ind, j;
        int counter = 0;
        fscanf(pFile1, "%d", &n);
        fscanf(pFile1, "%d", &d);
        for (int i = 0; i <n; i++)
        {
            fscanf(pFile1, "%d", &x);
            vec.push_back(x);
        }
        for (int i = 0; i < vec.size()-1; i++)
        {   
            t=vec[i]+d;
            if(t>vec[n-1])
                break;
            ind=f(i+1,n-1, t);
            if(ind!=-1)
            {
                counter++;
                j=ind+1;
                while(j<n && vec[j]==t)
                {
                    j++; counter++;
                }
                j=ind-1;
                while(vec[j]==t)
                {
                    j--; counter++;
                }
            }
            
        }
        fprintf(pFile2, "%d", counter);
    }
    fclose(pFile1);
    fclose(pFile2);
    return 0;
}
1
 Аватар для dima koz
23 / 17 / 7
Регистрация: 05.06.2012
Сообщений: 72
Записей в блоге: 5
16.06.2012, 11:57
vector не даст никакого убыстрения в данном случае, а алгоритм предложенный,если даже не хуже:
вместо цикла в цикле, делается два цикла в цикле
1
0 / 0 / 0
Регистрация: 15.06.2012
Сообщений: 21
16.06.2012, 11:58  [ТС]
valeriikozlov, ОГРОМНОЕ СПАСИБО!)
0
0 / 0 / 0
Регистрация: 15.06.2012
Сообщений: 21
16.06.2012, 12:22  [ТС]
dima koz, спасибо) уже сделали)
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
16.06.2012, 12:22
Помогаю со студенческими работами здесь

Определить в заданной последовательности целых чисел количество чисел Фибоначчи
Определить в заданной последовательности целых чисел количество чисел Фибоначчи.

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

В заданной конечной последовательности целых чисел найти количество элементов кратных 7 (MASM32)
В заданной конечной последовательности целых чисел найти количество элементов кратных 7 (MASM32)

В последовательности целых чисел найти количество чисел в которых нет 3 и 7 и наименьшее среди этих чисел
Разработать процедуру, которая в последовательности целых чисел находит количество чисел в которых нет 3-ки и 7-ки и наименьшее среди этих...

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


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
20
Ответ Создать тему
Новые блоги и статьи
PhpStorm 2025.3: WSL Terminal всегда стартует в ~
and_y87 14.12.2025
PhpStorm 2025. 3: WSL Terminal всегда стартует в ~ (home), игнорируя директорию проекта Симптом: После обновления до PhpStorm 2025. 3 встроенный терминал WSL открывается в домашней директории. . .
Access
VikBal 11.12.2025
Помогите пожалуйста !! Как объединить 2 одинаковые БД Access с разными данными.
Новый ноутбук
volvo 07.12.2025
Всем привет. По скидке в "черную пятницу" взял себе новый ноутбук Lenovo ThinkBook 16 G7 на Амазоне: Ryzen 5 7533HS 64 Gb DDR5 1Tb NVMe 16" Full HD Display Win11 Pro
Музыка, написанная Искусственным Интеллектом
volvo 04.12.2025
Всем привет. Некоторое время назад меня заинтересовало, что уже умеет ИИ в плане написания музыки для песен, и, собственно, исполнения этих самых песен. Стихов у нас много, уже вышли 4 книги, еще 3. . .
От async/await к виртуальным потокам в Python
IndentationError 23.11.2025
Армин Ронахер поставил под сомнение async/ await. Создатель Flask заявляет: цветные функции - провал, виртуальные потоки - решение. Не threading-динозавры, а новое поколение лёгких потоков. Откат?. . .
Поиск "дружественных имён" СОМ портов
Argus19 22.11.2025
Поиск "дружественных имён" СОМ портов На странице: https:/ / norseev. ru/ 2018/ 01/ 04/ comportlist_windows/ нашёл схожую тему. Там приведён код на С++, который показывает только имена СОМ портов, типа,. . .
Сколько Государство потратило денег на меня, обеспечивая инсулином.
Programma_Boinc 20.11.2025
Сколько Государство потратило денег на меня, обеспечивая инсулином. Вот решила сделать интересный приблизительный подсчет, сколько государство потратило на меня денег на покупку инсулинов. . . .
Ломающие изменения в C#.NStar Alpha
Etyuhibosecyu 20.11.2025
Уже можно не только тестировать, но и пользоваться C#. NStar - писать оконные приложения, содержащие надписи, кнопки, текстовые поля и даже изображения, например, моя игра "Три в ряд" написана на этом. . .
Мысли в слух
kumehtar 18.11.2025
Кстати, совсем недавно имел разговор на тему медитаций с людьми. И обнаружил, что они вообще не понимают что такое медитация и зачем она нужна. Самые базовые вещи. Для них это - когда просто люди. . .
Создание Single Page Application на фреймах
krapotkin 16.11.2025
Статья исключительно для начинающих. Подходы оригинальностью не блещут. В век Веб все очень привыкли к дизайну Single-Page-Application . Быстренько разберем подход "на фреймах". Мы делаем одну. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2025, CyberForum.ru