2014 dxdy logo

Научный форум dxdy

Математика, Физика, Computer Science, Machine Learning, LaTeX, Механика и Техника, Химия,
Биология и Медицина, Экономика и Финансовая Математика, Гуманитарные науки




Начать новую тему Ответить на тему На страницу Пред.  1, 2, 3, 4  След.
 
 Re: Блокировки
Сообщение16.06.2017, 13:58 


04/05/13
125
Кажется, я немного облажался с сортировкой.. :oops: сейчас исправлю

 Профиль  
                  
 
 Re: Блокировки
Сообщение16.06.2017, 13:59 
Заслуженный участник
Аватара пользователя


01/09/13
4706
inky в сообщении #1226101 писал(а):
Исправил :-)

Ну два условия в if, а не 4!....

 Профиль  
                  
 
 Re: Блокировки
Сообщение16.06.2017, 14:03 
Заслуженный участник


04/03/09
916
Geen в сообщении #1226112 писал(а):
Необязательно. Нужно только если возникает пересечение с добавляемой.
Согласен. В худшем случае, у нас всем равно будут в структуре сидеть все блокировки, и тогда удаление старых - это лишняя операция (но худший случай для неудаления старых и худший случай для удаления старых - это разные худшие случаи). А вот что будет выгоднее в среднем - пока не знаю. С одной стороны, если не удалять просроченные, это минус одна операция, с другой стороны, если удалять просроченные, у нас будет меньше элементов в структуре, что определенно плюс.

 Профиль  
                  
 
 Re: Блокировки
Сообщение16.06.2017, 14:05 
Заслуженный участник
Аватара пользователя


01/09/13
4706
12d3 в сообщении #1226118 писал(а):
у нас будет меньше элементов в структуре, что определенно плюс.

Если поиск логарифмичен это не очень критично...

 Профиль  
                  
 
 Re: Блокировки
Сообщение16.06.2017, 14:14 
Заслуженный участник


04/03/09
916
Geen в сообщении #1226121 писал(а):
Если поиск логарифмичен это не очень критично...

Хех, я опять накосячил. Ведь у нас сортировка не по времени а по границам ресурсов. Так что поиск старых в любом случае линейный, что сводит все плюсы сета на нет. Значит, не нужно специально искать старые.

 Профиль  
                  
 
 Re: Блокировки
Сообщение16.06.2017, 14:43 


04/05/13
125
Geen в сообщении #1226117 писал(а):
Ну два условия в if, а не 4!....

:oops:
Используется синтаксис C++
if (!(it->end < start) &&
    !(it->start > end)) {
    cout << "No";
    return 1;
}
 


-- 16.06.2017, 16:54 --

Не могу понять, где я теперь накосячил....неверный ответ на 7-ом тесте. Видимо, какие-то активные блокировки остаются в хвосте. Функция insert вроде как добавляет элемент перед итератором.
код: [ скачать ] [ спрятать ]
Используется синтаксис C++
#include <iostream>
#include <list>

using namespace std;

struct block {
    int start, end, time;
};

int block_me(list<block> *blocked, int time, int start, int end, int duration) {
    list<block>::const_iterator spot = blocked->cbegin();
   
    for (auto it = blocked->cbegin(); it != blocked->cend(); ++it) {
        if (it->time <= time) {
            break;
        }
        if (it->time > spot->time) {
            spot = it;
            ++spot;
        }
       
        if (!(it->end < start) &&
            !(it->start > end)) {
            cout << "No";
            return 1;
        }
    }
   
    block t = {
        .start = start,
        .end = end,
        .time = time + duration
    };
    blocked->insert(spot, t);
   
    cout << "Yes";
    return 0;
}

int main() {
    int n, m;
    cin >> n >> m;
    list<block> blocked;
    for (int i = 0; i < m; ++i) {
        int s, l, r, d;
        cin >> s >> l >> r >> d;
        block_me(&blocked, s, l, r, d);
        cout << endl;
    }
   
    return 0;
}
 


-- 16.06.2017, 17:13 --

Все, понял где облажался. И исправил.
код: [ скачать ] [ спрятать ]
Используется синтаксис C++
#include <iostream>
#include <list>

using namespace std;

struct block {
    int start, end, time;
};

int block_me(list<block> *blocked, int time, int start, int end, int duration) {
    list<block>::const_iterator spot = blocked->cbegin();
   
    for (auto it = blocked->cbegin(); it != blocked->cend(); ++it) {
        if (it->time <= time) {
            break;
        }
        if (it->time > time + duration) {
            spot = it;
            ++spot;
        }
       
        if (!(it->end < start) &&
            !(it->start > end)) {
            cout << "No";
            return 1;
        }
    }
   
    block t = {
        .start = start,
        .end = end,
        .time = time + duration
    };
    blocked->insert(spot, t);
   
    cout << "Yes";
    return 0;
}

int main() {
    int n, m;
    cin >> n >> m;
    list<block> blocked;
    for (int i = 0; i < m; ++i) {
        int s, l, r, d;
        cin >> s >> l >> r >> d;
        block_me(&blocked, s, l, r, d);
        cout << endl;
    }
   
    return 0;
}
 


-- 16.06.2017, 17:37 --

Снова застрял..не проходит ограничение по времени на 17-ом тесте :cry:

 Профиль  
                  
 
 Re: Блокировки
Сообщение16.06.2017, 15:40 
Заслуженный участник


06/07/11
5627
кран.набрать.грамота
На каких языках там можно писать? Java или паскаль в списке доступных есть? Тоже что ли попробую :mrgreen:

 Профиль  
                  
 
 Re: Блокировки
Сообщение16.06.2017, 15:45 


04/05/13
125
rockclimber
Можно. Вот ссылка. Задача E.

 Профиль  
                  
 
 Re: Блокировки
Сообщение16.06.2017, 16:11 
Заслуженный участник


06/07/11
5627
кран.набрать.грамота
inky
"У вас нет прав просматривать это соревнование"
Надо предварительно другие задания пройти?

 Профиль  
                  
 
 Re: Блокировки
Сообщение16.06.2017, 16:53 


04/05/13
125
rockclimber
Это были задачки на стажировку, регистрация на которую уже закрыта. :roll:

-- 16.06.2017, 18:54 --

Я не смог решить эту задачу с блокировками и теперь она не дает мне покоя :-)

-- 16.06.2017, 19:07 --

Geen, 12d3 я попробовал реализовать через set. Но, похоже, что-то не учел, программа выдает неверный ответ на 9-ом тесте..
Почитал документацию к set, переписал оператор "<" так, чтобы для неподходящей блокировки
Код:
!comp(a, b) && !comp(b, a) == true
.
код: [ скачать ] [ спрятать ]
Используется синтаксис C++
#include <iostream>
#include <set>

using namespace std;

class Block {
    int start, end, time, duration, expires;
   
public:
    Block(int time, int start, int end, int duration) : start(start), end(end), time(time), duration(duration), expires(time + duration) {}
   
    Block() {}
   
    Block& operator=(const Block& r) {
        this->time = r.time;
        this->start = r.start;
        this->end = r.end;
        this->duration = r.duration;
        this->expires = r.expires;
        return *this;
    }
   
    bool operator<(const Block& r) const {
        Block first, second;
        if (this->time < r.time) {
            first = *this;
            second = r;
        } else {
            first = r;
            second = *this;
        }
       
        if (first.expires > second.time &&
            !(first.end < second.start) &&
            !(first.start > second.end)) {
            return false;
        }
       
        return first.time < second.time;
    }
};

int main() {
    int n, m;
    cin >> n >> m;
    set<Block> blocked;
    for (int i = 0; i < m; ++i) {
        int s, l, r, d;
        cin >> s >> l >> r >> d;
        Block t(s, l, r, d);
        auto res = blocked.insert(t);
        if (res.second) {
            cout << "Yes";
        } else {
            cout << "No";
        }
        cout << endl;
    }

    return 0;
}
 

 Профиль  
                  
 
 Re: Блокировки
Сообщение16.06.2017, 17:51 
Заслуженный участник


04/03/09
916
inky
Так можно сортировать просто по левой границе занимаемых ресурсов, т.е. по возрастанию start. И будем следить, чтобы у блокировок, которые находятся в множестве, промежутки $[start;end]$ не пересекались. Делаем примерно так:
На этапе проверки на пересечение с имеющимимся блокировками, ищем наименьший элемент, у которого start больше вставляемого, а также наибольший элемент, у которого start меньше вставляемого. Если эти товарищи пересекаются (как прямоугольники), то не вставляем. А если не пересекаются прямоугольники, но пересекаются "в проекции" на ось ресурсов, то удаляем старый(е) и вставляем новый. А если даже в проекции не пересекаются, то просто вставляем.
UPD. Конечно же, проверять на пересечение надо не только найденные в множестве наибольший и наименьший, но и все между ними.

 Профиль  
                  
 
 Re: Блокировки
Сообщение16.06.2017, 18:09 
Заслуженный участник
Аватара пользователя


01/09/13
4706
inky в сообщении #1226208 писал(а):
попробовал реализовать через set.

Просто set не поможет (на множестве отрезков у нас не полное упорядочивание).
Возможно надо самому реализовать какое-нибудь достаточно сбалансированное дерево с эффективным блочным удалением...
Только всю память надо выделять один раз в начале.

 Профиль  
                  
 
 Re: Блокировки
Сообщение16.06.2017, 18:13 


04/05/13
125
12d3
Чуть позже попробую реализовать.

Geen
Я надеюсь, что не придется самому писать дерево :roll:

 Профиль  
                  
 
 Re: Блокировки
Сообщение16.06.2017, 18:37 
Заслуженный участник
Аватара пользователя


01/09/13
4706
Кстати, а сколько бит там int?

 Профиль  
                  
 
 Re: Блокировки
Сообщение16.06.2017, 18:48 


04/05/13
125
Geen
Не меньше 16 бит. На 64-разрядных обычно 32 бита.

-- 16.06.2017, 20:50 --

12d3
Реализовал, но определенно где-то закралась ошибка: runtime error на седьмом тесте.
код: [ скачать ] [ спрятать ]
Используется синтаксис C++
#include <iostream>
#include <set>

using namespace std;

class Block {
    int start, end, time, duration, expires;
   
public:
    Block(int time, int start, int end, int duration) : start(start), end(end), time(time), duration(duration), expires(time + duration) {}
   
    Block() {}
   
    Block& operator=(const Block& r) {
        this->time = r.time;
        this->start = r.start;
        this->end = r.end;
        this->duration = r.duration;
        this->expires = r.expires;
        return *this;
    }
   
    bool operator<(const Block& r) const {
        return this->start < r.start;
    }
   
    bool intersects(Block bl) const {
        if (!(this->end < bl.start) &&
            !(this->start > bl.end) &&
            ((this->time < bl.time && this->expires > bl.time) ||
             (bl.time < this->time && bl.expires > this->time))) {
            return true;
        }
        return false;
    }
   
    bool resources_intersect(Block bl) const {
        if (!(this->end < bl.start) &&
            !(this->start > bl.end)) {
            return true;
        }
        return false;
    }
};

bool insert(set<Block> *blocked, Block bl) {
    set<Block>::iterator up = blocked->upper_bound(bl), low;
    if (up != blocked->end() || !blocked->empty()) {
        low = prev(up);
    } else {
        low = up;
    }
   
    if ((low != blocked->end() && low->intersects(bl)) ||
        (up != blocked->end() && up->intersects(bl))) {
        return false;
    }
   
    if (low != blocked->end() && low->resources_intersect(bl)) {
        blocked->erase(low);
    }
   
    if (up != blocked->end() && up->resources_intersect(bl)) {
        blocked->erase(up);
    }
   
    blocked->insert(bl);
   
    return true;
}

int main() {
    int n, m;
    cin >> n >> m;
    set<Block> blocked;
    for (int i = 0; i < m; ++i) {
        int s, l, r, d;
        cin >> s >> l >> r >> d;
        Block t(s, l, r, d);
        if (insert(&blocked, t)) {
            cout << "Yes";
        } else {
            cout << "No";
        }
        cout << endl;
    }
   
    return 0;
}
 

 Профиль  
                  
Показать сообщения за:  Поле сортировки  
Начать новую тему Ответить на тему  [ Сообщений: 53 ]  На страницу Пред.  1, 2, 3, 4  След.

Модераторы: Karan, Toucan, PAV, maxal, Супермодераторы



Кто сейчас на конференции

Сейчас этот форум просматривают: Mikhail_K


Вы не можете начинать темы
Вы не можете отвечать на сообщения
Вы не можете редактировать свои сообщения
Вы не можете удалять свои сообщения
Вы не можете добавлять вложения

Найти:
Powered by phpBB © 2000, 2002, 2005, 2007 phpBB Group