swap.ml 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. (******************************************************************************)
  2. (* Copyright © Joly Clément, 2015 *)
  3. (* *)
  4. (* leowzukw@vmail.me *)
  5. (* *)
  6. (* Ce logiciel est un programme informatique servant à exécuter *)
  7. (* automatiquement des programmes à l'ouverture du terminal. *)
  8. (* *)
  9. (* Ce logiciel est régi par la licence CeCILL soumise au droit français et *)
  10. (* respectant les principes de diffusion des logiciels libres. Vous pouvez *)
  11. (* utiliser, modifier et/ou redistribuer ce programme sous les conditions *)
  12. (* de la licence CeCILL telle que diffusée par le CEA, le CNRS et l'INRIA *)
  13. (* sur le site "http://www.cecill.info". *)
  14. (* *)
  15. (* En contrepartie de l'accessibilité au code source et des droits de copie, *)
  16. (* de modification et de redistribution accordés par cette licence, il n'est *)
  17. (* offert aux utilisateurs qu'une garantie limitée. Pour les mêmes raisons, *)
  18. (* seule une responsabilité restreinte pèse sur l'auteur du programme, le *)
  19. (* titulaire des droits patrimoniaux et les concédants successifs. *)
  20. (* *)
  21. (* A cet égard l'attention de l'utilisateur est attirée sur les risques *)
  22. (* associés au chargement, à l'utilisation, à la modification et/ou au *)
  23. (* développement et à la reproduction du logiciel par l'utilisateur étant *)
  24. (* donné sa spécificité de logiciel libre, qui peut le rendre complexe à *)
  25. (* manipuler et qui le réserve donc à des développeurs et des professionnels *)
  26. (* avertis possédant des connaissances informatiques approfondies. Les *)
  27. (* utilisateurs sont donc invités à charger et tester l'adéquation du *)
  28. (* logiciel à leurs besoins dans des conditions permettant d'assurer la *)
  29. (* sécurité de leurs systèmes et ou de leurs données et, plus généralement, *)
  30. (* à l'utiliser et l'exploiter dans les mêmes conditions de sécurité. *)
  31. (* *)
  32. (* Le fait que vous puissiez accéder à cet en-tête signifie que vous avez *)
  33. (* pris connaissance de la licence CeCILL, et que vous en avez accepté les *)
  34. (* termes. *)
  35. (******************************************************************************)
  36. open Core.Std
  37. open Core_bench.Std
  38. (* File to test who is the faster to swap two elements *)
  39. (* TODO:
  40. * Use core_bench for trustable result
  41. * Improve list algo ?
  42. * Verify that output of all functions are the same *)
  43. (* Arguments:
  44. * li: a list of elements
  45. * a,b: positions
  46. * a < b*)
  47. let data = List.init 5 ~f:(fun i -> i)
  48. let big_data = List.init 1_000_000 ~f:(fun i -> i)
  49. (* With an array *)
  50. let array li a b =
  51. let ar = List.to_array li in
  52. (* Store a value *)
  53. let v = ar.(a) in
  54. ar.(a) <- ar.(b);
  55. ar.(b) <- v;
  56. Array.to_list ar
  57. ;;
  58. (* Manipulating list *)
  59. let list li a b =
  60. (* We need a < b *)
  61. let ( a, b ) : ( int * int ) =
  62. if a < b
  63. then ( a, b )
  64. else ( b, a )
  65. in
  66. let b' = List.nth_exn li b in (* Ephemeral value, to swap *)
  67. let a' = ref b' in (* Avoid walk around the list twice *)
  68. List.mapi li
  69. ~f:(fun i element ->
  70. if i = a
  71. then begin
  72. a' := element;
  73. b'
  74. end
  75. else if i = b
  76. then !a'
  77. else
  78. element
  79. )
  80. ;;
  81. (* Another idee *)
  82. let once li a b =
  83. let rec worker li head middle tail ~ca ~cb ~i =
  84. match ( ca, cb ) with
  85. | ( Some ca ), ( Some cb ) -> head @ [cb] @ middle @ [ca] @ tail
  86. | _ -> li |> (function
  87. | hd :: tl ->
  88. if i < a
  89. then worker tl (head @ [hd]) middle tail ~ca ~cb ~i:(succ i)
  90. else if i = a
  91. then worker tl head middle tail ~ca:(Some hd) ~cb ~i:(succ i)
  92. else if i < b
  93. then worker tl head (middle @ [hd]) tail ~ca ~cb ~i:(succ i)
  94. else if i = b
  95. then worker ~ca ~cb:(Some hd) tl ~i:(succ i) head middle tail
  96. else if i > b
  97. then worker [] head middle li ~ca ~cb ~i:(-1)
  98. else assert false
  99. | [] -> []) (* both a and b are set, should be taken before *)
  100. in
  101. let ( a, b ) = if a < b then (a,b) else (b,a) in
  102. worker li [] [] [] ~ca:None ~cb:None ~i:0
  103. ;;
  104. (* Element to swap *)
  105. let a' = 2;;
  106. let b' = 4;;
  107. let a'' = 0;;
  108. let b'' = 10_000 - 1;;
  109. (* Visual test *)
  110. let print_li li =
  111. printf "[ ";
  112. List.iter li ~f:(fun e -> printf "%i; " e);
  113. printf "]\n"
  114. ;;
  115. print_li (array data a' b');;
  116. print_li (list data a' b');;
  117. print_li (once data a' b');;
  118. let tests = [
  119. "Array - small list", (fun () -> array data a' b');
  120. "Array - big list", (fun () -> array big_data a'' b'');
  121. "List - small list", (fun () -> list data a' b');
  122. "List - big list", (fun () -> list big_data a'' b'');
  123. "Once - small list", (fun () -> once data a' b');
  124. "Once - big once", (fun () -> once big_data a'' b'');
  125. ]
  126. let () =
  127. List.map tests ~f:(fun (name,test) -> Bench.Test.create ~name test)
  128. |> Bench.make_command
  129. |> Command.run