SatSolverWrapper.ts 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395
  1. import { JavaObject, cast_java_lang_Object } from '../../../java/lang/JavaObject';
  2. import { JavaInteger, cast_java_lang_Integer } from '../../../java/lang/JavaInteger';
  3. import { SatSolverA, cast_de_nrw_schule_svws_core_kursblockung_satsolver_SatSolverA } from '../../../core/kursblockung/satsolver/SatSolverA';
  4. import { LinkedCollection, cast_de_nrw_schule_svws_core_adt_collection_LinkedCollection } from '../../../core/adt/collection/LinkedCollection';
  5. import { JavaIterator, cast_java_util_Iterator } from '../../../java/util/JavaIterator';
  6. import { System, cast_java_lang_System } from '../../../java/lang/System';
  7. export class SatSolverWrapper extends SatSolverA {
  8. private readonly _solver : SatSolverA;
  9. private readonly varFALSE : number;
  10. private readonly varTRUE : number;
  11. /**
  12. * Erstellt eine Ebene über dem {@link SatSolverA}, um verschiedene Bedingungen/Constraints als Klauseln zu
  13. * codieren.
  14. *
  15. * @param solver Der Solver, der intern verwendet wird.
  16. */
  17. public constructor(solver : SatSolverA) {
  18. super();
  19. this._solver = solver;
  20. this.varTRUE = this._solver.createNewVar();
  21. this.varFALSE = -this.varTRUE;
  22. this.c_1(this.varTRUE);
  23. }
  24. public createNewVar() : number {
  25. return this._solver.createNewVar();
  26. }
  27. public addClause(pVars : Array<number>) : void {
  28. this._solver.addClause(pVars);
  29. }
  30. public isVarTrue(pVar : number) : boolean {
  31. return this._solver.isVarTrue(pVar);
  32. }
  33. public solve(maxTimeMillis : number) : number {
  34. return this._solver.solve(maxTimeMillis);
  35. }
  36. public getVarCount() : number {
  37. return this._solver.getVarCount();
  38. }
  39. public getClauseCount() : number {
  40. return this._solver.getClauseCount();
  41. }
  42. /**
  43. * Liefert ein Array der Länge n mit neu erzeugten Variablennummern.
  44. *
  45. * @param n Die Länge des Arrays.
  46. *
  47. * @return Ein Array der Länge n mit neu erzeugten Variablennummern.
  48. */
  49. public createNewVars(n : number) : Array<number> {
  50. let temp : Array<number> | null = Array(n).fill(0);
  51. for (let i : number = 0; i < temp.length; i++){
  52. temp[i] = this._solver.createNewVar();
  53. }
  54. return temp;
  55. }
  56. /**
  57. * Liefert eine Variable, die zuvor auf FALSE forciert wurde.
  58. *
  59. * @return Eine Variable, die zuvor auf FALSE forciert wurde.
  60. */
  61. public getVarFALSE() : number {
  62. return this.varFALSE;
  63. }
  64. /**
  65. * Liefert eine Variable, die zuvor auf TRUE forciert wurde.
  66. *
  67. * @return Eine Variable, die zuvor auf TRUE forciert wurde.
  68. */
  69. public getVarTRUE() : number {
  70. return this.varTRUE;
  71. }
  72. /**
  73. * Fügt eine Klausel der Größe 1 hinzu. Forciert damit die übergebene Variable auf TRUE.
  74. *
  75. * @param x Die Variable wird auf TRUE gesetzt.
  76. */
  77. public c_1(x : number) : void {
  78. this._solver.addClause([x]);
  79. }
  80. /**
  81. * Fügt eine Klausel der Größe 2 hinzu. Forciert damit, dass mindestens eine der beiden Variablen TRUE ist,
  82. * letzlich @code{x + y >= 1}.
  83. *
  84. * @param x Die Variable x der Klausel (x OR y).
  85. * @param y Die Variable y der Klausel (x OR y).
  86. */
  87. public c_2(x : number, y : number) : void {
  88. this._solver.addClause([x, y]);
  89. }
  90. /**
  91. * Fügt eine Klausel der Größe 3 hinzu. Forciert damit, dass mindestens eine der drei Variablen TRUE ist,
  92. * letzlich @code{x + y + z >= 1}.
  93. *
  94. * @param x Die Variable x der Klausel (x OR y OR z).
  95. * @param y Die Variable y der Klausel (x OR y OR z).
  96. * @param z Die Variable z der Klausel (x OR y OR z).
  97. */
  98. public c_3(x : number, y : number, z : number) : void {
  99. this._solver.addClause([x, y, z]);
  100. }
  101. /**
  102. * Forciert, dass nicht beide Variablen TRUE sind, letzlich @code{x + y ≤ 1}.
  103. *
  104. * @param x Die Variable x der Klausel (-x OR -y).
  105. * @param y Die Variable y der Klausel (-x OR -y).
  106. */
  107. public c_not_both(x : number, y : number) : void {
  108. this.c_2(-x, -y);
  109. }
  110. /**
  111. * Liefert die Variable z für die {@code z = x AND y} gilt.
  112. *
  113. * @param x Variable der obigen Gleichung.
  114. * @param y Variable der obigen Gleichung.
  115. *
  116. * @return Die Variable z für die {@code z = x AND y} gilt.
  117. */
  118. public c_new_var_AND(x : number, y : number) : number {
  119. let c : number = this._solver.createNewVar();
  120. this.c_2(x, -c);
  121. this.c_2(y, -c);
  122. this.c_3(-x, -y, c);
  123. return c;
  124. }
  125. /**
  126. * Forciert, dass genau {@code amount} Variablen des Arrays den Wert TRUE haben.
  127. *
  128. * @param pArray Das Variablenarray.
  129. * @param amount Die Anzahl an TRUEs in der Variablenliste.
  130. */
  131. public c_exactly_GENERIC(pArray : Array<number>, amount : number) : void;
  132. /**
  133. * Forciert, dass genau {@code amount} Variablen der Variablenliste den Wert TRUE haben.
  134. *
  135. * @param list Die Variablenliste.
  136. * @param amount Die Anzahl an TRUEs in der Variablenliste.
  137. */
  138. public c_exactly_GENERIC(list : LinkedCollection<Number>, amount : number) : void;
  139. /**
  140. * Implementation for method overloads of 'c_exactly_GENERIC'
  141. */
  142. public c_exactly_GENERIC(__param0 : Array<number> | LinkedCollection<Number>, __param1 : number) : void {
  143. if (((typeof __param0 !== "undefined") && Array.isArray(__param0)) && ((typeof __param1 !== "undefined") && typeof __param1 === "number")) {
  144. let pArray : Array<number> = __param0;
  145. let amount : number = __param1 as number;
  146. this.c_exactly_GENERIC(SatSolverWrapper.toList(pArray), amount);
  147. } else if (((typeof __param0 !== "undefined") && ((__param0 instanceof JavaObject) && (__param0.isTranspiledInstanceOf('de.nrw.schule.svws.core.adt.collection.LinkedCollection'))) || (__param0 === null)) && ((typeof __param1 !== "undefined") && typeof __param1 === "number")) {
  148. let list : LinkedCollection<Number> = cast_de_nrw_schule_svws_core_adt_collection_LinkedCollection(__param0);
  149. let amount : number = __param1 as number;
  150. list = new LinkedCollection(list);
  151. if (amount > list.size()) {
  152. console.log(JSON.stringify("FEHLER: c_exactly_GENERIC --> amount > list.size()"));
  153. }
  154. if (amount === 0) {
  155. for (let x of list) {
  156. this.c_1(-x);
  157. }
  158. return;
  159. }
  160. if (amount === list.size()) {
  161. for (let x of list) {
  162. this.c_1(+x);
  163. }
  164. return;
  165. }
  166. if (amount === 1) {
  167. if (list.size() === 1) {
  168. this.c_1(list.getFirst().valueOf());
  169. return;
  170. }
  171. if (list.size() === 2) {
  172. this.c_unequal(list.getFirst().valueOf(), list.getLast().valueOf());
  173. return;
  174. }
  175. this.c_exactly_one(list);
  176. return;
  177. }
  178. this.c_exactly_NETWORK(list, amount);
  179. } else throw new Error('invalid method overload');
  180. }
  181. /**
  182. * Forciert, dass höchstens {@code maximum} Variablen der Variablenliste den Wert TRUE haben.
  183. *
  184. * @param list Die Variablenliste.
  185. * @param maximum Die maximale Anzahl an TRUEs in der Variablenliste.
  186. */
  187. public c_at_most_GENERIC(list : LinkedCollection<Number>, maximum : number) : void {
  188. list = new LinkedCollection(list);
  189. if (maximum >= list.size()) {
  190. return;
  191. }
  192. if (maximum === 0) {
  193. for (let x of list) {
  194. this.c_1(-x);
  195. }
  196. return;
  197. }
  198. if (maximum === 1) {
  199. this.c_at_most_one_tree(list);
  200. return;
  201. }
  202. this.c_at_most_NETWORK(list, maximum);
  203. }
  204. /**
  205. * Forciert, dass genau eine Variable der Liste TRUE ist. Falls die Liste leer ist, führt das zur direkten
  206. * Unlösbarkeit der Formel.
  207. *
  208. * @param list Genau eine der Variablen der Liste muss TRUE sein.
  209. */
  210. private c_exactly_one(list : LinkedCollection<Number>) : void {
  211. this.c_1(this.c_at_most_one_tree(list));
  212. }
  213. /**
  214. * Forciert, dass {@code z = x OR y} gilt.
  215. *
  216. * @param x Variable der obigen Gleichung.
  217. * @param y Variable der obigen Gleichung.
  218. * @param z Variable der obigen Gleichung.
  219. */
  220. private c_z_equals_x_or_y(x : number, y : number, z : number) : void {
  221. this.c_2(-x, z);
  222. this.c_2(-y, z);
  223. this.c_3(x, y, -z);
  224. }
  225. /**
  226. * Liefert die Variable z für die {@code z = x OR y} gilt.
  227. *
  228. * @param x Variable der obigen Gleichung.
  229. * @param y Variable der obigen Gleichung.
  230. *
  231. * @return Die Variable z für die {@code z = x OR y} gilt.
  232. */
  233. private c_new_var_OR(x : number, y : number) : number {
  234. let z : number = this._solver.createNewVar();
  235. this.c_2(-x, z);
  236. this.c_2(-y, z);
  237. this.c_3(x, y, -z);
  238. return z;
  239. }
  240. /**
  241. * Forciert, dass in der Liste maximal eine Variable TRUE ist. Die Ergebnisvariable ist eine OR-Verknüpfung
  242. * aller Variablen der Liste.
  243. *
  244. * @param list Forciert, dass maximal eine Variable der Liste TRUE ist.
  245. *
  246. * @return Die Ergebnisvariable ist eine OR-Verknüpfung aller Variablen der Liste.
  247. */
  248. private c_at_most_one_tree(list : LinkedCollection<Number>) : number {
  249. list = new LinkedCollection(list);
  250. if (list.isEmpty()) {
  251. list.add(this.varFALSE);
  252. }
  253. while (list.size() >= 2) {
  254. let a : number = list.removeFirst().valueOf();
  255. let b : number = list.removeFirst().valueOf();
  256. let c : number = this._solver.createNewVar();
  257. this.c_not_both(a, b);
  258. this.c_z_equals_x_or_y(a, b, c);
  259. list.add(c);
  260. }
  261. return list.removeFirst().valueOf();
  262. }
  263. private c_exactly_NETWORK(list : LinkedCollection<Number>, amount : number) : void {
  264. this.c_bitonic_sort(list);
  265. let i : number = 0;
  266. let iter : JavaIterator<Number> = list.iterator();
  267. while (iter.hasNext()) {
  268. let value : Number = iter.next();
  269. if (i < amount) {
  270. this.c_1(+value.valueOf());
  271. } else {
  272. this.c_1(-value.valueOf());
  273. }
  274. i++;
  275. }
  276. }
  277. private c_at_most_NETWORK(list : LinkedCollection<Number>, maximum : number) : void {
  278. this.c_bitonic_sort(list);
  279. let i : number = 0;
  280. let iter : JavaIterator<Number> = list.iterator();
  281. while (iter.hasNext()) {
  282. let value : Number = iter.next();
  283. if (i < maximum)
  284. i++; else
  285. this.c_1(-value.valueOf());
  286. }
  287. }
  288. private c_bitonic_sort(list : LinkedCollection<Number>) : void {
  289. this.c_fill_False_until_power_two(list);
  290. this.c_bitonic_sort_power_two(list);
  291. }
  292. private c_fill_False_until_power_two(list : LinkedCollection<Number>) : void {
  293. let size : number = 1;
  294. while (size < list.size()) {
  295. size *= 2;
  296. }
  297. while (list.size() < size) {
  298. list.addLast(this.varFALSE);
  299. }
  300. }
  301. private c_bitonic_sort_power_two(list : LinkedCollection<Number>) : void {
  302. for (let window : number = 2; window <= list.size(); window *= 2){
  303. this.c_bitonic_sort_spiral(list, window);
  304. for (let difference : number = Math.trunc(window / 2); difference >= 2; difference /= 2){
  305. this.c_bitonic_sort_difference(list, difference);
  306. }
  307. }
  308. }
  309. private c_bitonic_sort_spiral(list : LinkedCollection<Number>, size : number) : void {
  310. for (let i : number = 0; i < list.size(); i += size){
  311. for (let i1 : number = i, i2 : number = i + size - 1; i1 < i2; i1++, i2--){
  312. this.c_bitonic_comparator(list, i1, i2);
  313. }
  314. }
  315. }
  316. private c_bitonic_sort_difference(list : LinkedCollection<Number>, size : number) : void {
  317. let half : number = Math.trunc(size / 2);
  318. for (let i : number = 0; i < list.size(); i += size){
  319. for (let j : number = 0; j < half; j++){
  320. this.c_bitonic_comparator(list, i + j, i + j + half);
  321. }
  322. }
  323. }
  324. private c_bitonic_comparator(result : LinkedCollection<Number>, i1 : number, i2 : number) : void {
  325. if (i1 >= i2) {
  326. console.log(JSON.stringify("c_bitonic_comparator: " + i1 + "," + i2 + " <-- ERROR!!!"));
  327. }
  328. let a : number = result.get(i1).valueOf();
  329. let b : number = result.get(i2).valueOf();
  330. result.set(i1, this.c_new_var_OR(a, b));
  331. result.set(i2, this.c_new_var_AND(a, b));
  332. }
  333. private static toList(pArray : Array<number>) : LinkedCollection<Number> {
  334. let list : LinkedCollection<Number> = new LinkedCollection();
  335. for (let x of pArray) {
  336. list.addLast(x);
  337. }
  338. return list;
  339. }
  340. c_unequal(x : number, y : number) : void {
  341. this.c_equal(x, -y);
  342. }
  343. c_equal(x : number, y : number) : void {
  344. this.c_2(-x, +y);
  345. this.c_2(+x, -y);
  346. }
  347. isTranspiledInstanceOf(name : string): boolean {
  348. return ['de.nrw.schule.svws.core.kursblockung.satsolver.SatSolverA', 'de.nrw.schule.svws.core.kursblockung.satsolver.SatSolverWrapper'].includes(name);
  349. }
  350. }
  351. export function cast_de_nrw_schule_svws_core_kursblockung_satsolver_SatSolverWrapper(obj : unknown) : SatSolverWrapper {
  352. return obj as SatSolverWrapper;
  353. }