set.tin 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637
  1. #nop vim: set filetype=tt:;
  2. /*
  3. 本文件属于 PaoTin++ 的一部分。
  4. PaoTin++ © 2020~2023 的所有版权均由担子炮(dzp <danzipao@gmail.com>) 享有并保留一切法律权利
  5. 你可以在遵照 GPLv3 协议的基础之上使用、修改及重新分发本程序。
  6. */
  7. #nop 本文件是 xtintin 的一部分,实现了一些集合处理函数;
  8. ///=== {
  9. ///// 集合处理函数:
  10. /////
  11. ///// 集合在 TinTin++ 中并无原生概念对应,PaoTin++ 中的集合用 #list 来实现,保证每个元素都不相同。
  12. ///// 也就是说,无论是对集合进行何种运算,重复元素在集合中只会出现一次。
  13. /////
  14. ///// 由于 PaoTin++ 中是用 #list 来存储 set,因此 #list 相关的命令和 xtintin API 都可以用于集合。
  15. //
  16. // #@ set.Create <元素> [...]
  17. // 利用提供的多个元素创建新集合。
  18. // };
  19. #func {set.Create} {
  20. #return {@set.Add{{};%0}};
  21. };
  22. ///=== {
  23. // #@ set.FromList <列表>
  24. // 将列表转换成集合。
  25. // };
  26. #func {set.FromList} {
  27. #local list {%0};
  28. #return {@set.Add{{};$list[%*]}};
  29. };
  30. ///=== {
  31. // #@ set.Size <集合>
  32. // 计算集合的元素数量。
  33. // };
  34. #func {set.Size} {
  35. #local set {%0};
  36. #return &set[];
  37. };
  38. ///=== {
  39. // #@ set.Contains <集合> <元素>
  40. // 判断元素是否在集合中。
  41. // };
  42. #func {set.Contains} {
  43. #info arguments save;
  44. #local set {$info[ARGUMENTS][1]};
  45. #local elem {$info[ARGUMENTS][2]};
  46. #if { @list.IndexOf{{$set};{$elem}} > 0 } {
  47. #return 1;
  48. };
  49. #else {
  50. #return 0;
  51. };
  52. };
  53. ///=== {
  54. // #@ set.Add <集合> <元素> [...]
  55. // 将一个或者多个元素添加到集合当中。
  56. // };
  57. #func {set.Add} {
  58. #info arguments save;
  59. #unvar info[ARGUMENTS][0];
  60. #local argv {$info[ARGUMENTS]};
  61. #if { &argv[] < 2 } {
  62. #return {$argv[1]};
  63. };
  64. #local set {$argv[1]};
  65. #local idx {0};
  66. #loop {2} {&argv[]} {idx} {
  67. #local elem {$argv[+$idx]};
  68. #if { !@set.Contains{{$set};{$elem}} } {
  69. #list set {add} {{$elem}};
  70. };
  71. };
  72. #return {$set};
  73. };
  74. ///=== {
  75. // #@ set.Remove <集合> <元素> [...]
  76. // 从集合中删除一个或者多个元素。
  77. // };
  78. #func {set.Remove} {
  79. #info arguments save;
  80. #unvar info[ARGUMENTS][0];
  81. #local argv {$info[ARGUMENTS]};
  82. #if { &argv[] < 2 } {
  83. #return {$argv[1]};
  84. };
  85. #local set {$argv[1]};
  86. #local idx {0};
  87. #loop {2} {&argv[]} {idx} {
  88. #local elem {$argv[+$idx]};
  89. #local idx @list.IndexOf{{$set};{$elem}};
  90. #if { $idx > 0 } {
  91. #list set {delete} {$idx};
  92. };
  93. };
  94. #return {$set};
  95. };
  96. ///=== {
  97. // #@ set.Equal <集合1> <集合2>
  98. // 判断集合1和集合2是否相等。
  99. // 📖 如果两个集合中包含的元素是相同的,那么就判定为相等。相等判定并不要求元素顺序相同。
  100. // };
  101. #func {set.Equal} {
  102. #local set1 {%1};
  103. #local set2 {%2};
  104. #if { @set.IsSubset{{$set1};{$set2}} && &set1[] == &set2[] } {
  105. #return 1;
  106. };
  107. #else {
  108. #return 0;
  109. };
  110. };
  111. ///=== {
  112. // #@ set.IsSubset <集合1> <集合2>
  113. // 判断集合2是否为集合1的子集。
  114. // 📖 如果集合1包含了集合2中的每个元素,那么集合2就是集合1的子集。
  115. // };
  116. #func {set.IsSubset} {
  117. #local set1 {%1};
  118. #local set2 {%2};
  119. #local idx {0};
  120. #foreach {*set2[]} {idx} {
  121. #if { !@set.Contains{{$set1};{$set2[$idx]}} } {
  122. #return 0;
  123. };
  124. };
  125. #return 1;
  126. };
  127. ///=== {
  128. // #@ set.IsProperSubset <集合1> <集合2>
  129. // 判断集合2是否为集合1的真子集。
  130. // 📖 如果集合1包含了集合中2的每个元素,而且集合1比集合2的元素还要多,
  131. // 那么集合2就是集合1的真子集。
  132. // };
  133. #func {set.IsProperSubset} {
  134. #local set1 {%1};
  135. #local set2 {%2};
  136. #if { @set.IsSubset{{$set1};{$set2}} && &set1[] > &set2[] } {
  137. #return 1;
  138. };
  139. #else {
  140. #return 0;
  141. };
  142. };
  143. ///=== {
  144. // #@ set.IsSuperset <集合1> <集合2>
  145. // 判断集合2是否为集合1的超集。
  146. // 📖 如果集合2包含了集合1中的每个元素,那么集合2就是集合1的超集。
  147. // };
  148. #func {set.IsSuperset} {
  149. #return {@set.IsSubset{{%2};{%1}}};
  150. };
  151. ///=== {
  152. // #@ set.IsProperSuperset <集合1> <集合2>
  153. // 判断集合2是否为集合1的真超集。
  154. // 📖 如果集合2包含了集合中1的每个元素,而且集合2比集合1的元素还要多,
  155. // 那么集合2就是集合1的真超集。
  156. // };
  157. #func {set.IsProperSuperset} {
  158. #return {@set.IsProperSubset{{%2};{%1}}};
  159. };
  160. ///=== {
  161. // #@ set.Union <集合1> <集合2> [...]
  162. // 求两个或者多个集合的并集。
  163. // 📖 并集是指把所有集合元素放在一块儿之后得到的新集合。
  164. // };
  165. #func {set.Union} {
  166. #info arguments save;
  167. #unvar info[ARGUMENTS][0];
  168. #local argv {$info[ARGUMENTS]};
  169. #if { &argv[] < 2 } {
  170. #return {$argv[1]};
  171. };
  172. #local set {$argv[1]};
  173. #local idx {0};
  174. #loop {2} {&argv[]} {idx} {
  175. #local other {$argv[+$idx]};
  176. #local set {@set.Add{{$set};$other[%*]}};
  177. };
  178. #return {$set};
  179. };
  180. ///=== {
  181. // #@ set.Intersection <集合1> <集合2> [...]
  182. // 求两个或者多个集合的交集。
  183. // 📖 交集是指由同时出现在所有集合中的元素组成的新集合。
  184. // };
  185. #func {set.Intersection} {
  186. #info arguments save;
  187. #unvar info[ARGUMENTS][0];
  188. #local argv {$info[ARGUMENTS]};
  189. #if { &argv[] < 2 } {
  190. #return {};
  191. };
  192. #local set {$argv[1]};
  193. #local idx {0};
  194. #loop {2} {&argv[]} {idx} {
  195. #if { &set[] == 0 } {
  196. #return {};
  197. };
  198. #local other {$argv[+$idx]};
  199. #loop {&set[]} {1} {idx} {
  200. #local elem {$set[$idx]};
  201. #if { !@set.Contains{{$other};{$elem}} } {
  202. #list set {delete} {$idx};
  203. };
  204. };
  205. };
  206. #return {$set};
  207. };
  208. ///=== {
  209. // #@ set.IsDisjoint <集合1> <集合2> [...]
  210. // 判断两个或者多个集合是否为不交集。
  211. // 📖 不交集是交集为空的意思。
  212. // };
  213. #func {set.IsDisjoint} {
  214. #local set {@set.Intersection{%0}};
  215. #if { &set[] == 0 } {
  216. #return 1;
  217. };
  218. #else {
  219. #return 0;
  220. };
  221. };
  222. ///=== {
  223. // #@ set.Product <集合1> <集合2> [...]
  224. // 计算两个或者多个集合的笛卡尔积。返回结果集合中的每个元素都是一个字符串列表,
  225. // 其值分别来自各个参数集合的元素。
  226. // 📖 两个集合的笛卡尔积是指由两个集合的元素组成的所有可能的有序对的集合。
  227. // 多个集合的积也可由此扩展。
  228. // };
  229. #func {set.Product} {
  230. #info arguments save;
  231. #unvar info[ARGUMENTS][0];
  232. #local argv {$info[ARGUMENTS]};
  233. #if { &argv[] < 2 } {
  234. #return {};
  235. };
  236. #local set {$argv[1]};
  237. #if { &set[] == 0 } {
  238. #return {};
  239. };
  240. #local idx {0};
  241. #loop {2} {&argv[]} {idx} {
  242. #local other {$argv[+$idx]};
  243. #if { &other[] == 0 } {
  244. #return {};
  245. };
  246. #local newSet {};
  247. #local left {};
  248. #local right {};
  249. #loop {1} {&set[]} {left} {
  250. #local left {$set[+$left]};
  251. #loop {1} {&other[]} {right} {
  252. #local right {$other[+$right]};
  253. #list newSet {add} {{$left;$right}};
  254. };
  255. };
  256. #local set {$newSet};
  257. };
  258. #return {$set};
  259. };
  260. ///=== {
  261. // #@ set.Diff <集合1> <集合2> [...]
  262. // 计算集合1与其它集合的相对差。
  263. // 📖 相对差是指从集合1中去掉所有在其它集合中出现过的元素之后剩下的集合。
  264. // };
  265. #func {set.Diff} {
  266. #info arguments save;
  267. #unvar info[ARGUMENTS][0];
  268. #local argv {$info[ARGUMENTS]};
  269. #if { &argv[] < 2 } {
  270. #return {$argv[1]};
  271. };
  272. #local set {$argv[1]};
  273. #local idx {0};
  274. #loop {2} {&argv[]} {idx} {
  275. #if { &set[] == 0 } {
  276. #return {};
  277. };
  278. #local other {$argv[+$idx]};
  279. #local set {@set.Remove{{$set};$other[%*]}};
  280. };
  281. #return {$set};
  282. };
  283. ///=== {
  284. // #@ set.SymmDiff <集合1> <集合2>
  285. // 计算集合1与集合2的对称差。
  286. // 📖 对称差是指并集与交集的相对差。
  287. // };
  288. #func {set.SymmDiff} {
  289. #return {@set.Diff{{@set.Union{{%1};{%2}}};{@set.Intersection{{%1};{%2}}}}};
  290. };
  291. ///=== {
  292. ///// 下面这一组 API 采用字符串列表来作为集合表达。功能与上面相同。只是数据类型不一样。
  293. ///// 为了和上面以示区分,前缀 set 改为 sset,意指 string set。
  294. //
  295. // #@ sset.Create <字符串列表>
  296. // 从字符串列表创建集合。
  297. // };
  298. #func {sset.Create} {
  299. #return {@sset.Add{{};%0}};
  300. };
  301. ///=== {
  302. // #@ sset.Size <集合>
  303. // 计算集合的元素数量。
  304. // };
  305. #func {sset.Size} {
  306. #return {@slist.Size{%0}};
  307. };
  308. ///=== {
  309. // #@ sset.Contains <集合> <元素>
  310. // 判断元素是否在集合中。
  311. // };
  312. #func {sset.Contains} {
  313. #info arguments save;
  314. #local set {$info[ARGUMENTS][1]};
  315. #local elem {$info[ARGUMENTS][2]};
  316. #if { @slist.IndexOf{{$set};{$elem}} > 0 } {
  317. #return 1;
  318. };
  319. #else {
  320. #return 0;
  321. };
  322. };
  323. ///=== {
  324. // #@ sset.Add <集合> <元素> [...]
  325. // 将一个或者多个元素添加到集合当中。
  326. // } ;
  327. #func {sset.Add} {
  328. #info arguments save;
  329. #unvar info[ARGUMENTS][0];
  330. #local argv {$info[ARGUMENTS]};
  331. #if { &argv[] < 2 } {
  332. #return {$argv[1]};
  333. };
  334. #local set {$argv[1]};
  335. #return {@slist.FromList{@set.Add{{@list.FromSlist{$set}};$argv[2..-1]}}};
  336. };
  337. ///=== {
  338. // #@ sset.Remove <集合> <元素> [...]
  339. // 从集合中删除一个或者多个元素。
  340. // };
  341. #func {sset.Remove} {
  342. #info arguments save;
  343. #unvar info[ARGUMENTS][0];
  344. #local argv {$info[ARGUMENTS]};
  345. #if { &argv[] < 2 } {
  346. #return {$argv[1]};
  347. };
  348. #local set {$argv[1]};
  349. #return {@slist.FromList{@set.Remove{{@list.FromSlist{$set}};$argv[2..-1]}}};
  350. };
  351. ///=== {
  352. // #@ sset.Equal <集合1> <集合2>
  353. // 判断集合1和集合2是否相等。
  354. // 📖 如果两个集合中包含的元素是相同的,那么就判定为相等。相等判定并不要求元素顺序相同。
  355. // };
  356. #func {sset.Equal} {
  357. #local set1 {%1};
  358. #local set2 {%2};
  359. #if { @sset.IsSubset{{$set1};{$set2}} && @slist.Size{$set1} == @slist.Size{$set2} } {
  360. #return 1;
  361. };
  362. #else {
  363. #return 0;
  364. };
  365. };
  366. ///=== {
  367. // #@ sset.IsSubset <集合1> <集合2>
  368. // 判断集合2是否为集合1的子集。
  369. // 📖 如果集合1包含了集合2中的每个元素,那么集合2就是集合1的子集。
  370. // };
  371. #func {sset.IsSubset} {
  372. #local set1 {%1};
  373. #local set2 {%2};
  374. #local elem {0};
  375. #foreach {$set2} {elem} {
  376. #if { !@sset.Contains{{$set1};{$elem}} } {
  377. #return 0;
  378. };
  379. };
  380. #return 1;
  381. };
  382. ///=== {
  383. // #@ sset.IsProperSubset <集合1> <集合2>
  384. // 判断集合2是否为集合1的真子集。
  385. // 📖 如果集合1包含了集合中2的每个元素,而且集合1比集合2的元素还要多,
  386. // 那么集合2就是集合1的真子集。
  387. // };
  388. #func {sset.IsProperSubset} {
  389. #local set1 {%1};
  390. #local set2 {%2};
  391. #if { @sset.IsSubset{{$set1};{$set2}} && @slist.Size{$set1} > @slist.Size{$set2} } {
  392. #return 1;
  393. };
  394. #else {
  395. #return 0;
  396. };
  397. };
  398. ///=== {
  399. // #@ sset.IsSuperset <集合1> <集合2>
  400. // 判断集合2是否为集合1的超集。
  401. // 📖 如果集合2包含了集合1中的每个元素,那么集合2就是集合1的超集。
  402. // };
  403. #func {sset.IsSuperset} {
  404. #return {@sset.IsSubset{{%2};{%1}}};
  405. };
  406. ///=== {
  407. // #@ sset.IsProperSuperset <集合1> <集合2>
  408. // 判断集合2是否为集合1的真超集。
  409. // 📖 如果集合2包含了集合中1的每个元素,而且集合2比集合1的元素还要多,
  410. // 那么集合2就是集合1的真超集。
  411. // };
  412. #func {sset.IsProperSuperset} {
  413. #return {@sset.IsProperSubset{{%2};{%1}}};
  414. };
  415. ///=== {
  416. // #@ sset.Union <集合1> <集合2> [...]
  417. // 求两个或者多个集合的并集。
  418. // 📖 并集是指把所有集合放在一块儿之后得到的新集合。
  419. // };
  420. #func {sset.Union} {
  421. #info arguments save;
  422. #unvar info[ARGUMENTS][0];
  423. #local argv {$info[ARGUMENTS]};
  424. #if { &argv[] < 2 } {
  425. #return {$argv[1]};
  426. };
  427. #local set {@list.FromSlist{$argv[1]}};
  428. #local idx {0};
  429. #loop {2} {&argv[]} {idx} {
  430. #local other {$argv[+$idx]};
  431. #local set {@set.Add{{$set};$other}};
  432. };
  433. #return {@slist.FromList{$set}};
  434. };
  435. ///=== {
  436. // #@ sset.Intersection <集合1> <集合2> [...]
  437. // 求两个或者多个集合的交集。
  438. // 📖 交集是指由同时出现在所有集合中的元素组成的新集合。
  439. // };
  440. #func {sset.Intersection} {
  441. #info arguments save;
  442. #unvar info[ARGUMENTS][0];
  443. #local argv {$info[ARGUMENTS]};
  444. #if { &argv[] < 2 } {
  445. #return {};
  446. };
  447. #local set {@list.FromSlist{$argv[1]}};
  448. #local idx {0};
  449. #loop {2} {&argv[]} {idx} {
  450. #if { &set[] == 0 } {
  451. #return {};
  452. };
  453. #local other {$argv[+$idx]};
  454. #loop {&set[]} {1} {idx} {
  455. #local elem {$set[$idx]};
  456. #if { !@sset.Contains{{$other};{$elem}} } {
  457. #list set {delete} {$idx};
  458. };
  459. };
  460. };
  461. #return {@slist.FromList{$set}};
  462. };
  463. ///=== {
  464. // #@ sset.IsDisjoint <集合1> <集合2> [...]
  465. // 判断两个或者多个集合是否为不交集。
  466. // 📖 不交集是交集为空的意思。
  467. // };
  468. #func {sset.IsDisjoint} {
  469. #local set {@sset.Intersection{%0}};
  470. #if { @slist.Size{$set} == 0 } {
  471. #return 1;
  472. };
  473. #else {
  474. #return 0;
  475. };
  476. };
  477. ///=== {
  478. // #@ sset.Product <集合1> <集合2> [...]
  479. // 计算两个或者多个集合的笛卡尔积。返回结果集合中的每个元素都是一个字符串,
  480. // 其值由各个参数集合的元素拼接而成。
  481. // 如果调用前设置了环境变量 sset-product-divider,则会以它为分隔符拼接。
  482. // <171>注意<299>,sset-product-divider 不能被设置为分号,这会导致灾难性的后果。
  483. // 📖 两个集合的笛卡尔积是指由两个集合的元素组成的所有可能的有序对的集合。
  484. // 多个集合的积也可由此扩展。
  485. // };
  486. #func {sset.Product} {
  487. #info arguments save;
  488. #unvar info[ARGUMENTS][0];
  489. #local argv {$info[ARGUMENTS]};
  490. #if { &argv[] < 2 } {
  491. #return {};
  492. };
  493. #local set {@list.FromSlist{$argv[1]}};
  494. #if { &set[] == 0 } {
  495. #return {};
  496. };
  497. #local divider {@defaultVar{$sset-product-divider}};
  498. #local idx {0};
  499. #loop {2} {&argv[]} {idx} {
  500. #local other {$argv[+$idx]};
  501. #if { "$other" == "" } {
  502. #return {};
  503. };
  504. #local newSet {};
  505. #local left {};
  506. #local right {};
  507. #loop {1} {&set[]} {left} {
  508. #local left {$set[+$left]};
  509. #foreach {$other} {right} {
  510. #list newSet {add} {{$left$divider$right}};
  511. };
  512. };
  513. #local set {$newSet};
  514. };
  515. #return {@slist.FromList{$set}};
  516. };
  517. ///=== {
  518. // #@ sset.Diff <集合1> <集合2> [...]
  519. // 计算集合1与其它集合的相对差。
  520. // 📖 相对差是指从集合1中去掉所有在其它集合中出现过的元素之后剩下的集合。
  521. // };
  522. #func {sset.Diff} {
  523. #info arguments save;
  524. #unvar info[ARGUMENTS][0];
  525. #local argv {$info[ARGUMENTS]};
  526. #if { &argv[] < 2 } {
  527. #return {$argv[1]};
  528. };
  529. #local set {@list.FromSlist{$argv[1]}};
  530. #local idx {0};
  531. #loop {2} {&argv[]} {idx} {
  532. #if { &set[] == 0 } {
  533. #return {};
  534. };
  535. #local other {$argv[+$idx]};
  536. #local set {@set.Remove{{$set};$other}};
  537. };
  538. #return {@slist.FromList{$set}};
  539. };
  540. ///=== {
  541. // #@ sset.SymmDiff <集合1> <集合2>
  542. // 计算集合1与集合2的对称差。
  543. // 📖 对称差是指并集与交集的相对差。
  544. // };
  545. #func {sset.SymmDiff} {
  546. #return {@sset.Diff{{@sset.Union{{%1};{%2}}};{@sset.Intersection{{%1};{%2}}}}};
  547. };