-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsol09b.erl
executable file
·65 lines (57 loc) · 1.64 KB
/
sol09b.erl
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
#!/usr/bin/escript
main([]) ->
main([25, "sol09.data"]);
main([WindowT, Filename])
when is_list(WindowT) ->
Window = list_to_integer(WindowT),
main([Window, Filename]);
main([Window, Filename]) ->
{ok, Data} = file:read_file(Filename),
TextItems = string:lexemes(Data, "\r\n\t "),
Nums = [binary_to_integer(T) || T <- TextItems],
Inv = find_invalid(Nums, Window),
Range = find_sum_range(Nums, Inv),
io:format("~p~n", [Range]),
io:format("~B~n", [lists:min(Range)+lists:max(Range)]).
find_sum([], _, _Tgt) ->
err;
find_sum(_, [], _Tgt) ->
err;
find_sum([X|_], [Y|_], Tgt)
when X + Y =:= Tgt ->
ok;
find_sum([X|XR], [Y|_]=YL, Tgt)
when X + Y < Tgt ->
find_sum(XR, YL, Tgt);
find_sum([X|_]=XL, [Y|YR], Tgt)
when X + Y > Tgt ->
find_sum(XL, YR, Tgt).
find_invalid(L, Window)
when length(L) =< Window ->
-1;
find_invalid([_|Rest] = L, Window) ->
Preamble = lists:sublist(L, Window),
[Target|_] = lists:nthtail(Window, L),
{A, B} = split_list(Preamble, Target div 2),
case find_sum(A, B, Target) of
err ->
Target;
ok ->
find_invalid(Rest, Window)
end.
split_list(List, Mid) ->
{A, B} = lists:partition(fun (X) -> X =< Mid end, List),
{lists:sort(A),
lists:reverse(lists:sort(B))}.
find_sum_range(Nums, Target) ->
iterate_sum(Nums, Target, queue:new(), 0).
iterate_sum(_Nums, Target, Queue, Target) ->
queue:to_list(Queue);
iterate_sum(Nums, Target, Queue, Total)
when Total > Target ->
{{value, N}, NewQueue} = queue:out(Queue),
iterate_sum(Nums, Target, NewQueue, Total-N);
iterate_sum([N|Nums], Target, Queue, Total)
when Total < Target ->
NewQueue = queue:in(N, Queue),
iterate_sum(Nums, Target, NewQueue, Total+N).