:- use_module( library( lists) ) .
% ----------------------------
% задание 1: шелл (седжвик)
% ----------------------------
% shs(+l, -s)
shs( L, S) :-
length( L, N) ,
sg( N, Gs) ,
sp( Gs, L, S) .
% sg(+n, -gs)
sg( N, Gs) :-
( N =< 1 - > Gs = [ 1 ]
; sgk( N, 0 , [ ] , Rs) ,
reverse( Rs, Gs)
) .
sgk( N, K, A, Rs) :-
hk( K, H) ,
hk( K1, H1) ,
( 3 * H1 > N - >
Rs = [ H| A]
; sgk( N, K1, [ H| A] , Rs)
) .
% hk(+k,-h)
hk( K, H) :-
) .
sp( [ ] , L, L) .
sp( [ G| Gs] , L, S) :-
gp( G, L, L1) ,
sp( Gs, L1, S) .
% gp(+g,+l,-s)
gp( G, L, S) :-
en( L, 0 , Ps) ,
findall ( Ns
, ( between
( 0 , G1
, R
) , cls
( Ps
, G
, R
, Ns
) ) , Bs
) , append( Bs, All) ,
keysort( All, Ss) ,
vs( Ss, S) .
cls( Ps, G, R, Ns) :-
iv( Cs, Vs1, Is1) ,
ins( Vs1, Vs2) ,
pi( Is1, Vs2, Ns) .
pr
( G
, R
, I
- _
) :- 0 is ( I
- R
) mod G
.
en( [ ] , _, [ ] ) .
en( [ X| Xs] , I, [ I- X| Ps] ) :-
en( Xs, I1, Ps) .
iv( [ ] , [ ] , [ ] ) .
iv
( [ I
- V
| T
] , [ V
| Vs
] , [ I
| Is ] ) :- iv
( T
, Vs
, Is ) .
pi( [ ] , [ ] , [ ] ) .
pi
( [ I
| Is ] , [ V
| Vs
] , [ I
- V
| T
] ) :- pi
( Is , Vs
, T
) .
vs( [ ] , [ ] ) .
vs( [ _- V| T] , [ V| Vs] ) :- vs( T, Vs) .
% ins(+l,-s)
ins( [ ] , [ ] ) .
ins( [ X| Xs] , S) :- ins( Xs, S1) , is1( X, S1, S) .
is1( X, [ ] , [ X] ) .
is1( X, [ Y| Ys] , [ X, Y| Ys] ) :- X =< Y, ! .
is1( X, [ Y| Ys] , [ Y| Zs] ) :- is1( X, Ys, Zs) .
% Оценка сложности сортировки Шелла (Sedgewick's sequence): O(N^(4/3)) в среднем, до O(N^(3/2)) в худшем случае
shellsort_complexity( N, Complexity) :-
Complexity
is N
** ( 4 / 3 ) . % Средняя сложность
% Предикат для измерения времени выполнения
time_sort( Predicate, InputList, SortedList, Time) :-
statistics( runtime, [ T1, _] ) ,
call ( Predicate
, InputList
, SortedList
) , statistics( runtime, [ T2, _] ) ,
% Пример использования сортировки Шелла
ex_shs :-
L = [ 3 , 1 , 4 , 1 , 5 , 9 , 2 , 6 , 5 , 3 , 5 ] ,
time_sort( shs, L, S, Time) ,
length( L, N) ,
shellsort_complexity( N, Complexity) ,
format( 'Сортировка Шелла:~n' ) ,
format( ' Исходный список: ~w~n' , [ L] ) ,
format( ' Отсортированный список: ~w~n' , [ S] ) ,
format( ' Время выполнения: ~w мс~n' , [ Time] ) ,
format( ' Оценка сложности (O(N^(4/3))): ~w для N = ~w~n' , [ Complexity, N] ) .
% ----------------------------
% задание 2: бинарные включения
% ----------------------------
% bis(+l,-s)
bis( L, S) :- bis1( L, [ ] , S) .
bis1( [ ] , A, A) .
bis1( [ X| Xs] , A, S) :-
bin_ins( X, A, A1) ,
bis1( Xs, A1, S) .
% bin_ins(+x,+l,-r)
bin_ins( X, L, R) :-
length( L, N) ,
bin_pos( X, L, 0 , N, P) ,
sp2( P, L, A, B) ,
append( A, [ X| B] , R) .
% bin_pos(+x,+l,+lo,+hi,-p)
bin_pos( X, L, Lo, Hi, P) :-
( Lo >= Hi - >
P = Lo
nth0( M, L, V) ,
( X =< V - >
bin_pos( X, L, Lo, M, P)
bin_pos( X, L, M1, Hi, P)
)
) .
% sp2(+k,+l,-a,-b)
sp2( 0 , L, [ ] , L) :- ! .
sp2( K, [ X| Xs] , [ X| A] , B) :-
K > 0 ,
sp2( K1, Xs, A, B) .
% Оценка сложности бинарной вставки: O(N^2) в худшем случае, O(N log N) в среднем (благодаря бинарному поиску)
binary_insertion_complexity( N, Complexity) :-
Complexity
is N
* N
. % Или N * log(N) для среднего случая
% Пример использования бинарных включений
ex_bis :-
L = [ 3 , 1 , 4 , 1 , 5 , 9 , 2 , 6 , 5 , 3 , 5 ] ,
time_sort( bis, L, S, Time) ,
length( L, N) ,
binary_insertion_complexity( N, Complexity) ,
format( 'Бинарные включения:~n' ) ,
format( ' Исходный список: ~w~n' , [ L] ) ,
format( ' Отсортированный список: ~w~n' , [ S] ) ,
format( ' Время выполнения: ~w мс~n' , [ Time] ) ,
format( ' Оценка сложности (O(N^2)): ~w для N = ~w~n' , [ Complexity, N] ) .
% Сравнение эффективности
compare_sorts( List) :-
time_sort( shs, List, _, ShellSortTime) ,
time_sort( bis, List, _, BinaryInsertionTime) ,
length( List, N) ,
shellsort_complexity( N, ShellsortComplexity) ,
binary_insertion_complexity( N, BinaryInsertionComplexity) ,
format( 'Сравнение сортировок:~n' ) ,
format( ' Список для сортировки: ~w~n' , [ List] ) ,
format( ' Длина списка (N): ~w~n' , [ N] ) ,
format( ' Сортировка Шелла (Sedgewick):~n' ) ,
format( ' Время выполнения: ~w мс~n' , [ ShellSortTime] ) ,
format( ' Оценка сложности (O(N^(4/3))): ~w~n' , [ ShellsortComplexity] ) ,
format( ' Сортировка бинарными включениями:~n' ) ,
format( ' Время выполнения: ~w мс~n' , [ BinaryInsertionTime] ) ,
format( ' Оценка сложности (O(N^2)): ~w~n' , [ BinaryInsertionComplexity] ) ,
( ShellSortTime < BinaryInsertionTime - >
format( ' Вывод: Сортировка Шелла быстрее для данного списка.~n' )
; format( ' Вывод: Сортировка бинарными включениями быстрее для данного списка.~n' )
) .
% ----------------------------
% задание 3: "слово из домино" (игра)
% ----------------------------
% igra(-w)
igra( w( I, G, R, A) ) :-
ds( Ts) ,
gr( Ts, 7 , 42 , Ti, T1) , ch( Ti, I) ,
gr( T1, 7 , 42 , Tg, T2) , ch( Tg, G) ,
gr( T2, 7 , 42 , Tr, T3) , ch( Tr, R) ,
T3 = Ta, sm( Ta, 42 ) , ch( Ta, A) .
% ds(-ts)
ds
( Ts
) :- findall ( A
- B
, ( between
( 0 , 6 , A
) , between
( A
, 6 , B
) ) , Ts
) .
% sm(+ts,-s)
sm( [ ] , 0 ) .
sm( [ A- B| T] , S) :-
sm( T, S1) ,
% gr(+ts,+k,+s,-g,-r)
gr( Ts, 0 , 0 , [ ] , Ts) :- ! .
gr( Ts, K, S, [ X| G] , R) :-
K > 0 , S >= 0 ,
select( X, Ts, T1) ,
X = A- B,
gr( T1, K1, S1, G, R) .
% ch(+ts,-c)
ch( Ts, C) :-
select( T, Ts, R) ,
or( T, A- B) ,
ch1( R, B, [ A- B] , C) .
ch1( [ ] , _, A, C) :- reverse( A, C) .
ch1( Ts, X, A, C) :-
select( T, Ts, R) ,
or1( T, X- Y) ,
ch1( R, Y, [ X- Y| A] , C) .
or( A- B, A- B) .
or( A- B, B- A) .
or1( A- B, X- Y) :- ( X= A, Y= B ; X= B, Y= A) .
% Пример использования игры в домино
ex_igra :-
igra( W) ,
format( 'Игра "Слово из домино":~n' ) ,
format( ' Решение: ~w~n' , [ W] ) .
% ----------------------------
% задание 4 (вариант 2): вставка подсписка с i-го элемента
% ----------------------------
% ins_sub(+sub,+l,+i,-r)
ins_sub( Sub, L, I, R) :-
I >= 1 ,
sp2( K, L, P, S) ,
append( P, Sub, T) ,
append( T, S, R) .
% Пример использования вставки подсписка
ex_ins_sub :-
Sub = [ 7 , 8 , 9 ] ,
L = [ 1 , 2 , 3 , 4 , 5 , 6 ] ,
I = 3 ,
ins_sub( Sub, L, I, R) ,
format( 'Вставка подсписка:~n' ) ,
format( ' Исходный список: ~w~n' , [ L] ) ,
format( ' Подсписок: ~w~n' , [ Sub] ) ,
format( ' Индекс вставки: ~w~n' , [ I] ) ,
format( ' Результат: ~w~n' , [ R] ) .
% Запуск всех примеров
run_examples :-
ex_shs,
ex_bis,
List = [ 5 , 2 , 8 , 1 , 9 , 4 , 7 , 3 , 6 , 0 , 5 , 2 , 8 , 1 , 9 , 4 , 7 , 3 , 6 , 0 ] , % Пример списка для сравнения
compare_sorts( List) ,
ex_igra,
ex_ins_sub.
?- run_examples.
:- use_module(library(lists)).

% ----------------------------
% задание 1: шелл (седжвик)
% ----------------------------

% shs(+l, -s)
shs(L, S) :-
    length(L, N),
    sg(N, Gs),
    sp(Gs, L, S).

% sg(+n, -gs)
sg(N, Gs) :-
    ( N =< 1 -> Gs = [1]
    ; sgk(N, 0, [], Rs),
      reverse(Rs, Gs)
    ).

sgk(N, K, A, Rs) :-
    hk(K, H),
    K1 is K + 1,
    hk(K1, H1),
    ( 3*H1 > N ->
        Rs = [H|A]
    ;   sgk(N, K1, [H|A], Rs)
    ).

% hk(+k,-h)
hk(K, H) :-
    ( 0 is K mod 2 ->
        P2k  is 1 << K,
        K2   is K // 2,
        P2k2 is 1 << K2,
        H is 9*P2k - 9*P2k2 + 1
    ;   P2k  is 1 << K,
        K2   is (K + 1) // 2,
        P2k2 is 1 << K2,
        H is 8*P2k - 6*P2k2 + 1
    ).

sp([], L, L).
sp([G|Gs], L, S) :-
    gp(G, L, L1),
    sp(Gs, L1, S).

% gp(+g,+l,-s)
gp(G, L, S) :-
    en(L, 0, Ps),
    G1 is G - 1,
    findall(Ns, (between(0, G1, R), cls(Ps, G, R, Ns)), Bs),
    append(Bs, All),
    keysort(All, Ss),
    vs(Ss, S).

cls(Ps, G, R, Ns) :-
    include(pr(G, R), Ps, Cs),
    iv(Cs, Vs1, Is1),
    ins(Vs1, Vs2),
    pi(Is1, Vs2, Ns).

pr(G, R, I-_) :- 0 is (I - R) mod G.

en([], _, []).
en([X|Xs], I, [I-X|Ps]) :-
    I1 is I + 1,
    en(Xs, I1, Ps).

iv([], [], []).
iv([I-V|T], [V|Vs], [I|Is]) :- iv(T, Vs, Is).

pi([], [], []).
pi([I|Is], [V|Vs], [I-V|T]) :- pi(Is, Vs, T).

vs([], []).
vs([_-V|T], [V|Vs]) :- vs(T, Vs).

% ins(+l,-s)
ins([], []).
ins([X|Xs], S) :- ins(Xs, S1), is1(X, S1, S).

is1(X, [], [X]).
is1(X, [Y|Ys], [X,Y|Ys]) :- X =< Y, !.
is1(X, [Y|Ys], [Y|Zs]) :- is1(X, Ys, Zs).

% Оценка сложности сортировки Шелла (Sedgewick's sequence): O(N^(4/3)) в среднем, до O(N^(3/2))  в худшем случае
shellsort_complexity(N, Complexity) :-
    Complexity is N**(4/3).  % Средняя сложность

% Предикат для измерения времени выполнения
time_sort(Predicate, InputList, SortedList, Time) :-
    statistics(runtime, [T1, _]),
    call(Predicate, InputList, SortedList),
    statistics(runtime, [T2, _]),
    Time is T2 - T1.


% Пример использования сортировки Шелла
ex_shs :-
    L = [3, 1, 4, 1, 5, 9, 2, 6, 5, 3, 5],
    time_sort(shs, L, S, Time),
    length(L, N),
    shellsort_complexity(N, Complexity),
    format('Сортировка Шелла:~n'),
    format('  Исходный список: ~w~n', [L]),
    format('  Отсортированный список: ~w~n', [S]),
   format('  Время выполнения: ~w мс~n', [Time]),
    format('  Оценка сложности (O(N^(4/3))): ~w для N = ~w~n', [Complexity, N]).


% ----------------------------
% задание 2: бинарные включения
% ----------------------------

% bis(+l,-s)
bis(L, S) :- bis1(L, [], S).

bis1([], A, A).
bis1([X|Xs], A, S) :-
    bin_ins(X, A, A1),
    bis1(Xs, A1, S).

% bin_ins(+x,+l,-r)
bin_ins(X, L, R) :-
    length(L, N),
    bin_pos(X, L, 0, N, P),
    sp2(P, L, A, B),
    append(A, [X|B], R).

% bin_pos(+x,+l,+lo,+hi,-p)
bin_pos(X, L, Lo, Hi, P) :-
    ( Lo >= Hi ->
        P = Lo
    ; M is (Lo + Hi) // 2,
      nth0(M, L, V),
      ( X =< V ->
          bin_pos(X, L, Lo, M, P)
      ; M1 is M + 1,
        bin_pos(X, L, M1, Hi, P)
      )
    ).

% sp2(+k,+l,-a,-b)
sp2(0, L, [], L) :- !.
sp2(K, [X|Xs], [X|A], B) :-
    K > 0,
    K1 is K - 1,
    sp2(K1, Xs, A, B).

% Оценка сложности бинарной вставки: O(N^2) в худшем случае, O(N log N) в среднем (благодаря бинарному поиску)
binary_insertion_complexity(N, Complexity) :-
    Complexity is N*N.  % Или N * log(N) для среднего случая

% Пример использования бинарных включений
ex_bis :-
    L = [3, 1, 4, 1, 5, 9, 2, 6, 5, 3, 5],
    time_sort(bis, L, S, Time),
    length(L, N),
    binary_insertion_complexity(N, Complexity),
    format('Бинарные включения:~n'),
    format('  Исходный список: ~w~n', [L]),
    format('  Отсортированный список: ~w~n', [S]),
        format('  Время выполнения: ~w мс~n', [Time]),
    format('  Оценка сложности (O(N^2)): ~w для N = ~w~n', [Complexity, N]).

% Сравнение эффективности
compare_sorts(List) :-
    time_sort(shs, List, _, ShellSortTime),
    time_sort(bis, List, _, BinaryInsertionTime),
    length(List, N),
    shellsort_complexity(N, ShellsortComplexity),
    binary_insertion_complexity(N, BinaryInsertionComplexity),

    format('Сравнение сортировок:~n'),
    format('  Список для сортировки: ~w~n', [List]),
    format('  Длина списка (N): ~w~n', [N]),

    format('  Сортировка Шелла (Sedgewick):~n'),
    format('    Время выполнения: ~w мс~n', [ShellSortTime]),
    format('    Оценка сложности (O(N^(4/3))): ~w~n', [ShellsortComplexity]),

    format('  Сортировка бинарными включениями:~n'),
    format('    Время выполнения: ~w мс~n', [BinaryInsertionTime]),
    format('    Оценка сложности (O(N^2)): ~w~n', [BinaryInsertionComplexity]),

    (ShellSortTime < BinaryInsertionTime ->
        format('  Вывод: Сортировка Шелла быстрее для данного списка.~n')
    ;   format('  Вывод: Сортировка бинарными включениями быстрее для данного списка.~n')
    ).



% ----------------------------
% задание 3: "слово из домино" (игра)
% ----------------------------

% igra(-w)
igra(w(I, G, R, A)) :-
    ds(Ts),
    gr(Ts, 7, 42, Ti, T1), ch(Ti, I),
    gr(T1, 7, 42, Tg, T2), ch(Tg, G),
    gr(T2, 7, 42, Tr, T3), ch(Tr, R),
    T3 = Ta, sm(Ta, 42), ch(Ta, A).

% ds(-ts)
ds(Ts) :- findall(A-B, (between(0,6,A), between(A,6,B)), Ts).

% sm(+ts,-s)
sm([], 0).
sm([A-B|T], S) :-
    sm(T, S1),
    S is S1 + A + B.

% gr(+ts,+k,+s,-g,-r)
gr(Ts, 0, 0, [], Ts) :- !.
gr(Ts, K, S, [X|G], R) :-
    K > 0, S >= 0,
    select(X, Ts, T1),
    X = A-B,
    S1 is S - (A + B),
    K1 is K - 1,
    gr(T1, K1, S1, G, R).

% ch(+ts,-c)
ch(Ts, C) :-
    select(T, Ts, R),
    or(T, A-B),
    ch1(R, B, [A-B], C).

ch1([], _, A, C) :- reverse(A, C).
ch1(Ts, X, A, C) :-
    select(T, Ts, R),
    or1(T, X-Y),
    ch1(R, Y, [X-Y|A], C).

or(A-B, A-B).
or(A-B, B-A).

or1(A-B, X-Y) :- (X=A, Y=B ; X=B, Y=A).

% Пример использования игры в домино
ex_igra :-
    igra(W),
    format('Игра "Слово из домино":~n'),
    format('  Решение: ~w~n', [W]).

% ----------------------------
% задание 4 (вариант 2): вставка подсписка с i-го элемента
% ----------------------------

% ins_sub(+sub,+l,+i,-r)
ins_sub(Sub, L, I, R) :-
    I >= 1,
    K is I - 1,
    sp2(K, L, P, S),
    append(P, Sub, T),
    append(T, S, R).

% Пример использования вставки подсписка
ex_ins_sub :-
    Sub = [7, 8, 9],
    L = [1, 2, 3, 4, 5, 6],
    I = 3,
    ins_sub(Sub, L, I, R),
    format('Вставка подсписка:~n'),
    format('  Исходный список: ~w~n', [L]),
    format('  Подсписок: ~w~n', [Sub]),
    format('  Индекс вставки: ~w~n', [I]),
    format('  Результат: ~w~n', [R]).

% Запуск всех примеров
run_examples :-
    ex_shs,
    ex_bis,
    List = [5, 2, 8, 1, 9, 4, 7, 3, 6, 0, 5, 2, 8, 1, 9, 4, 7, 3, 6, 0], % Пример списка для сравнения
    compare_sorts(List),
     ex_igra,
    ex_ins_sub.
    
    ?- run_examples.