Array.h 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774
  1. #ifndef Array_H
  2. #define Array_H
  3. #include "Errors.h"
  4. #include <stdexcept>
  5. #include "Text.h"
  6. #include "ReferenceCounter.h"
  7. namespace Framework
  8. {
  9. template< class TYP >
  10. //! Ein Eintrag in einer Linked List
  11. struct ArrayEintrag
  12. {
  13. TYP var;
  14. bool set = false;
  15. ArrayEintrag< TYP > *next = 0;
  16. //! Setzt den Eintrag auf die Werte des anderen Eintrages
  17. ArrayEintrag &operator=( ArrayEintrag &r )
  18. {
  19. var = r.var;
  20. set = r.set;
  21. next = r.next;
  22. return *this;
  23. }
  24. //! Gibt den aktuell gespeicherten Wert zurück
  25. operator TYP()
  26. {
  27. if( !set )
  28. {
  29. Text err = "Index out of Range Exception File: ";
  30. err += __FILE__;
  31. err += " Line: ";
  32. err += __LINE__;
  33. throw std::out_of_range( (char *)err );
  34. }
  35. return var;
  36. }
  37. //! inkrementiert durch die Linked List durch
  38. ArrayEintrag< TYP > &operator++() //! prefix
  39. {
  40. if( !next )
  41. {
  42. ArrayEintrag<TYP> tmp;
  43. tmp.set = 0;
  44. tmp.next = 0;
  45. *this = tmp;
  46. return *this;
  47. }
  48. *this = *next;
  49. return *next;
  50. }
  51. //! inkrementiert durch die Linked List durch
  52. ArrayEintrag< TYP > &operator++( int ) //! postfix
  53. {
  54. if( !next )
  55. {
  56. ArrayEintrag<TYP> tmp;
  57. tmp.set = 0;
  58. tmp.next = 0;
  59. *this = tmp;
  60. return *this;
  61. }
  62. *this = *next;
  63. return *next;
  64. }
  65. #ifdef WIN32
  66. #pragma warning( once : 26495 )
  67. };
  68. #else
  69. };
  70. #endif
  71. template< class TYP >
  72. class Iterator
  73. {
  74. private:
  75. ArrayEintrag< TYP > *current;
  76. public:
  77. Iterator( ArrayEintrag< TYP > *start )
  78. {
  79. current = start;
  80. while( current && !current->set )
  81. {
  82. current = current->next;
  83. }
  84. }
  85. Iterator( const Iterator &it )
  86. {
  87. current = it.current;
  88. }
  89. Iterator< TYP > &operator=( Iterator< TYP > r )
  90. {
  91. current = r.current;
  92. return *this;
  93. }
  94. bool hasNext()
  95. {
  96. ArrayEintrag< TYP > *next = current->next;
  97. while( next && !next->set )
  98. {
  99. next = next->next;
  100. }
  101. return next != 0;
  102. }
  103. Iterator< TYP > next()
  104. {
  105. if( !current )
  106. {
  107. Text err = "Index out of Range Exception File: ";
  108. err += __FILE__;
  109. err += " Line: ";
  110. err += __LINE__;
  111. throw std::out_of_range( (char *)err );
  112. }
  113. return Iterator( current->next );
  114. }
  115. operator bool()
  116. {
  117. return current != 0;
  118. }
  119. operator TYP()
  120. {
  121. if( !current || !current->set )
  122. {
  123. Text err = "Index out of Range Exception File: ";
  124. err += __FILE__;
  125. err += " Line: ";
  126. err += __LINE__;
  127. throw std::out_of_range( (char *)err );
  128. }
  129. return current->var;
  130. }
  131. Iterator< TYP > &operator++() //! prefix
  132. {
  133. do
  134. {
  135. if( current )
  136. current = current->next;
  137. } while( current && !current->set );
  138. return *this;
  139. }
  140. Iterator< TYP > operator++( int ) //! postfix
  141. {
  142. Iterator< TYP > temp( *this );
  143. do
  144. {
  145. if( current )
  146. current = current->next;
  147. } while( current && !current->set );
  148. return temp;
  149. }
  150. TYP operator->()
  151. {
  152. if( !current || !current->set )
  153. {
  154. Text err = "Index out of Range Exception File: ";
  155. err += __FILE__;
  156. err += " Line: ";
  157. err += __LINE__;
  158. throw std::out_of_range( (char *)err );
  159. }
  160. return current->var;
  161. }
  162. TYP val()
  163. {
  164. if( !current || !current->set )
  165. {
  166. Text err = "Index out of Range Exception File: ";
  167. err += __FILE__;
  168. err += " Line: ";
  169. err += __LINE__;
  170. throw std::out_of_range( (char *)err );
  171. }
  172. return current->var;
  173. }
  174. void set( TYP val )
  175. {
  176. if( current )
  177. {
  178. current->var = val;
  179. current->set = true;
  180. }
  181. else
  182. {
  183. Text err = "Index out of Range Exception File: ";
  184. err += __FILE__;
  185. err += " Line: ";
  186. err += __LINE__;
  187. throw std::out_of_range( (char *)err );
  188. }
  189. }
  190. };
  191. #define _ val()
  192. template< class TYP >
  193. //! Eine Linked List von Klassen, die kein Reference Counting berteiben
  194. class Array : public virtual ReferenceCounter
  195. {
  196. private:
  197. ArrayEintrag< TYP > *entries;
  198. ArrayEintrag< TYP > *end;
  199. int count;
  200. public:
  201. //! Erstellt eine neue Linked List
  202. Array() noexcept
  203. : ReferenceCounter()
  204. {
  205. entries = new ArrayEintrag< TYP >();
  206. entries->set = 0;
  207. entries->next = 0;
  208. end = entries;
  209. count = 0;
  210. }
  211. //! Kopiert eine Linked list
  212. //!
  213. Array( const Array &arr )
  214. : Array()
  215. {
  216. int anz = arr.getEintragAnzahl();
  217. for( int i = 0; i < anz; i++ )
  218. add( arr.get( i ) );
  219. }
  220. //! Leert und löscht die Linked List
  221. ~Array()
  222. {
  223. leeren();
  224. delete entries;
  225. }
  226. //! Hängt ein Element ans Ende der Liste an
  227. //! \param t Das neue Element
  228. void add( TYP t )
  229. {
  230. count++;
  231. if( !last->set )
  232. {
  233. last->var = t;
  234. last->set = 1;
  235. return;
  236. }
  237. last->next = new ArrayEintrag< TYP >();
  238. last = last->next;
  239. last->set = 0;
  240. last->next = 0;
  241. }
  242. //! Fügt ein Element bei einer bestimmten Position in die Liste ein
  243. //! \param t das neue Element
  244. //! \param i Die Position, wo das Element eingefügt wird (danach der Index des neuen Elementes)
  245. void add( TYP t, int i )
  246. {
  247. if( i < 0 || i > count )
  248. throwOutOfRange( __FILE__, __LINE__, i, count );
  249. if( i == count )
  250. {
  251. add( t );
  252. return;
  253. }
  254. count++;
  255. ArrayEintrag< TYP > *e = entries;
  256. for( int a = 0; a < i; ++a )
  257. e = e->next;
  258. ArrayEintrag< TYP > *ne = new ArrayEintrag< TYP >();
  259. ne->var = e->var;
  260. ne->set = e->set;
  261. ne->next = e->next;
  262. e->next = ne;
  263. e->var = t;
  264. e->set = 1;
  265. }
  266. //! Setzt den Wert des i-ten Eintrags
  267. //! \param t der Neue Wert
  268. //! \param i Der Index des Eintrages der gesetzt werden soll
  269. void set( TYP t, int i )
  270. {
  271. if( i < || i >= count )
  272. throwOutOfRange( __FILE__, __LINE__, i, count );
  273. ArrayEintrag< TYP > *e = entries;
  274. for( int a = 0; a < i; ++a )
  275. e = e->next;
  276. e->var = t;
  277. e->set = 1;
  278. }
  279. //! Verändert die Position des i-ten Elementes in der Liste
  280. //! \param i Der Index des Elementes, welches verschoben werden soll
  281. //! \param p Die Zielposition des Elementes (danach der neue Index des Elementes)
  282. void setPosition( int i, int p )
  283. {
  284. if( i == p )
  285. return;
  286. if( i < 0 || p < 0 || i >= count || p >= count )
  287. throwOutOfRange( __FILE__, __LINE__, i, count );
  288. ArrayEintrag< TYP > *e = entries;
  289. ArrayEintrag< TYP > *ve = 0;
  290. for( int a = 0; a < i; ++a )
  291. {
  292. ve = e;
  293. e = e->next;
  294. }
  295. ArrayEintrag< TYP > *e2 = entries == e ? e->next : entries;
  296. ArrayEintrag< TYP > *ve2 = 0;
  297. for( int a = 0; a < p; ++a )
  298. {
  299. ve2 = e2;
  300. if( e2->next == e )
  301. e2 = e->next;
  302. else
  303. e2 = e2->next;
  304. }
  305. if( e == last )
  306. last = e2;
  307. else if( e2 == last )
  308. last = e;
  309. if( !ve2 )
  310. entries = e;
  311. else
  312. ve2->next = e;
  313. if( ve )
  314. ve->next = e->next;
  315. else
  316. entries = e->next;
  317. e->next = e2;
  318. }
  319. //! Löscht ein Bestimmtes Element
  320. //! \param i Der Index des Elementes das gelöscht werden soll
  321. void remove( int i )
  322. {
  323. if( i < 0 || i >= count )
  324. throwOutOfRange( __FILE__, __LINE__, i, count );
  325. ArrayEintrag< TYP > *e = entries;
  326. for( int a = 0; a < i; ++a )
  327. e = e->next;
  328. if( e->next )
  329. {
  330. e->var = e->next->var;
  331. e->set = e->next->set;
  332. }
  333. else
  334. e->set = 0;
  335. ArrayEintrag< TYP > *del = e->next;
  336. if( e->next )
  337. e->next = e->next->next;
  338. else
  339. e->next = 0;
  340. if( del )
  341. {
  342. del->set = 0;
  343. del->next = 0;
  344. if( last == del )
  345. last = e;
  346. delete del;
  347. }
  348. count--;
  349. }
  350. //! Löscht ein Bestimmtes Element
  351. //! \param i Der Index des Elementes das gelöscht werden soll
  352. void removeValue( TYP value )
  353. {
  354. ArrayEintrag< TYP > *e = entries;
  355. while( e->var != value )
  356. {
  357. if( !e->next )
  358. return;
  359. e = e->next;
  360. }
  361. if( !e )
  362. return;
  363. if( e->next )
  364. {
  365. e->var = e->next->var;
  366. e->set = e->next->set;
  367. }
  368. else
  369. e->set = 0;
  370. ArrayEintrag< TYP > *del = e->next;
  371. if( e->next )
  372. e->next = e->next->next;
  373. else
  374. e->next = 0;
  375. if( del )
  376. {
  377. del->set = 0;
  378. del->next = 0;
  379. if( last == del )
  380. last = e;
  381. delete del;
  382. }
  383. count--;
  384. }
  385. //! Vertauscht zwei Elemente in der Liste
  386. //! \param vi Der Index des ersten Elementes
  387. //! \param ni Der Index des zweiten Elementes
  388. void tausch( int vi, int ni )
  389. {
  390. TYP tmp = get( ni );
  391. set( get( vi ), ni );
  392. set( tmp, vi );
  393. }
  394. //! Löscht alle Elemente der Liste
  395. void leeren()
  396. {
  397. ArrayEintrag< TYP > *e2 = 0;
  398. for( ArrayEintrag< TYP > *e = entries; e; e = e->next )
  399. {
  400. delete e2;
  401. e2 = e;
  402. }
  403. delete e2;
  404. entries = new ArrayEintrag< TYP >();
  405. entries->set = 0;
  406. entries->next = 0;
  407. last = entries;
  408. count = 0;
  409. }
  410. //! Gibt einen Iterator zurück.
  411. //! Mit ++ kann durch die Liste iteriert werden
  412. Iterator< TYP > getIterator() const
  413. {
  414. return Iterator< TYP >( entries );
  415. }
  416. //! Gibt zurück, wie viele Elemente in der Liste sind
  417. int getEintragAnzahl() const
  418. {
  419. return count;
  420. }
  421. //! Gibt den Wert des i-ten Elementes zurück
  422. //! \param i Der index des gesuchten Elementes
  423. //! throws:
  424. //! \param std:out_of_range wenn i < 0 oder i >= getEintragAnzahl()
  425. TYP get( int i ) const
  426. {
  427. if( i < 0 || i >= count )
  428. throwOutOfRange( __FILE__, __LINE__, i, count );
  429. ArrayEintrag< TYP > *e = entries;
  430. for( int a = 0; a < i && e; ++a )
  431. e = e->next;
  432. return e->var;
  433. }
  434. //! Überprüft, ob ein Element in der Liste enthalten ist
  435. //! \param i Der Index des gesuchten Elementes
  436. //! \return (true), wenn der Index vorhanden ist. (false) sonnst
  437. bool hat( int i ) const
  438. {
  439. return i >= 0 && i < count;
  440. }
  441. //! Gibt den Index eines Wertes zurück
  442. //! \param t Der Wert, nach dem gesucht werden soll
  443. int getWertIndex( TYP t ) const
  444. {
  445. int ret = 0;
  446. for( ArrayEintrag< TYP > *e = entries; e; e = e->next )
  447. {
  448. if( e->set && e->var == t )
  449. return ret;
  450. ++ret;
  451. }
  452. return -1;
  453. }
  454. Array &operator=( const Array &arr )
  455. {
  456. leeren();
  457. int anz = arr.getEintragAnzahl();
  458. for( int i = 0; i < anz; i++ )
  459. add( arr.get( i ) );
  460. return *this;
  461. }
  462. };
  463. template< class TYP >
  464. //! Eine Linked List von Zeigern auf Zeichnunge, die Reference Counting berteiben
  465. class RCArray : public virtual ReferenceCounter
  466. {
  467. private:
  468. ArrayEintrag< TYP * > *entries;
  469. ArrayEintrag< TYP * > *last;
  470. int count;
  471. public:
  472. //! Erstellt eine neue Linked List
  473. RCArray() noexcept
  474. : ReferenceCounter()
  475. {
  476. entries = new ArrayEintrag< TYP * >();
  477. entries->set = 0;
  478. entries->next = 0;
  479. last = entries;
  480. count = 0;
  481. }
  482. //! Kopiert eine Linked list
  483. RCArray( const RCArray &arr )
  484. : RCArray()
  485. {
  486. int anz = arr.getEintragAnzahl();
  487. for( int i = 0; i < anz; i++ )
  488. add( arr.get( i ) );
  489. ref = 1;
  490. }
  491. //! Leert und löscht die Linked List
  492. ~RCArray()
  493. {
  494. leeren();
  495. delete entries;
  496. }
  497. //! Hängt ein Element ans Ende der Liste an
  498. //! \param t Das neue Element
  499. void add( TYP *t )
  500. {
  501. count++;
  502. if( !last->set )
  503. {
  504. last->var = t;
  505. last->set = 1;
  506. return;
  507. }
  508. last->next = new ArrayEintrag< TYP * >();
  509. last = last->next;
  510. last->var = t;
  511. last->set = 1;
  512. }
  513. //! Fügt ein Element bei einer bestimmten Position in die Liste ein
  514. //! \param t das neue Element
  515. //! \param i Die Position, wo das Element eingefügt wird (danach der Index des neuen Elementes)
  516. void add( TYP *t, int i )
  517. {
  518. if( i < 0 || i > count )
  519. throwOutOfRange( __FILE__, __LINE__, i, count );
  520. if( i == count )
  521. {
  522. add( t );
  523. return;
  524. }
  525. ArrayEintrag< TYP * > *e = entries;
  526. for( int a = 0; a < i; ++a )
  527. e = e->next;
  528. ArrayEintrag< TYP * > *ne = new ArrayEintrag< TYP * >();
  529. ne->var = e->var;
  530. ne->set = e->set;
  531. ne->next = e->next;
  532. e->next = ne;
  533. e->var = t;
  534. e->set = 1;
  535. if( last->next )
  536. last = last->next;
  537. count++;
  538. }
  539. //! Setzt den Wert des i-ten Eintrags
  540. //! \param t der Neue Wert
  541. //! \param i Der Index des Eintrages der gesetzt werden soll
  542. void set( TYP *t, int i )
  543. {
  544. if( i < 0 || i >= count )
  545. throwOutOfRange( __FILE__, __LINE__, i, count );
  546. ArrayEintrag< TYP * > *e = entries;
  547. for( int a = 0; a < i; ++a )
  548. e = e->next;
  549. if( e->set && e->var )
  550. e->var->release();
  551. e->var = t;
  552. e->set = 1;
  553. }
  554. //! Verändert die Position des i-ten Elementes in der Liste
  555. //! \param i Der Index des Elementes, welches verschoben werden soll
  556. //! \param p Die Zielposition des Elementes (danach der neue Index des Elementes)
  557. void setPosition( int i, int p )
  558. {
  559. if( i == p )
  560. return;
  561. if( i < 0 || p < 0 || i >= count || p >= count )
  562. throwOutOfRange( __FILE__, __LINE__, i, count );
  563. ArrayEintrag< TYP * > *ve = 0;
  564. ArrayEintrag< TYP * > *e = entries;
  565. for( int a = 0; a < i; ++a )
  566. {
  567. ve = e;
  568. e = e->next;
  569. }
  570. ArrayEintrag< TYP * > *e2 = entries == e ? e->next : entries;
  571. ArrayEintrag< TYP * > *ve2 = 0;
  572. for( int a = 0; a < p; ++a )
  573. {
  574. ve2 = e2;
  575. if( e2->next == e )
  576. e2 = e->next;
  577. else
  578. e2 = e2->next;
  579. }
  580. if( last == e )
  581. last = e2;
  582. else if( last == e2 )
  583. last = e;
  584. if( !ve2 )
  585. entries = e;
  586. else
  587. ve2->next = e;
  588. if( ve )
  589. ve->next = e->next;
  590. else
  591. entries = e->next;
  592. e->next = e2;
  593. }
  594. //! Löscht ein Bestimmtes Element
  595. //! \param i Der Index des Elementes das gelöscht werden soll
  596. void remove( int i )
  597. {
  598. if( i < 0 || i >= count )
  599. throwOutOfRange( __FILE__, __LINE__, i, count );
  600. ArrayEintrag< TYP * > *e = entries;
  601. for( int a = 0; a < i; ++a )
  602. e = e->next;
  603. if( e->next )
  604. {
  605. if( e->set && e->var )
  606. e->var->release();
  607. e->var = e->next->var;
  608. e->set = e->next->set;
  609. }
  610. else
  611. {
  612. if( e->set && e->var )
  613. e->var->release();
  614. e->set = 0;
  615. }
  616. ArrayEintrag< TYP * > *del = e->next;
  617. if( e->next )
  618. e->next = e->next->next;
  619. else
  620. e->next = 0;
  621. if( del )
  622. {
  623. del->set = 0;
  624. del->next = 0;
  625. if( last == del )
  626. last = e;
  627. delete del;
  628. }
  629. count--;
  630. }
  631. //! Vertauscht zwei Elemente in der Liste
  632. //! \param vi Der Index des ersten Elementes
  633. //! \param ni Der Index des zweiten Elementes
  634. void tausch( int vi, int ni )
  635. {
  636. if( vi < 0 || ni < 0 )
  637. return;
  638. TYP *tmp = get( ni );
  639. set( get( vi ), ni );
  640. set( tmp, vi );
  641. }
  642. //! Löscht alle Elemente der Liste
  643. void leeren()
  644. {
  645. ArrayEintrag< TYP * > *e2 = 0;
  646. for( ArrayEintrag< TYP * > *e = entries; e; e = e->next )
  647. {
  648. if( e2 && e2->var && e2->set )
  649. e2->var->release();
  650. delete e2;
  651. e2 = e;
  652. }
  653. if( e2 && e2->var && e2->set )
  654. e2->var->release();
  655. delete e2;
  656. entries = new ArrayEintrag< TYP * >();
  657. entries->set = 0;
  658. entries->next = 0;
  659. last = entries;
  660. count = 0;
  661. }
  662. //! Gibt einen Iterator zurück.
  663. //! Mit ++ kann durch die Liste iteriert werden
  664. Iterator< TYP * > getIterator() const
  665. {
  666. return Iterator< TYP * >( entries );
  667. }
  668. //! Gibt zurück, wie viele Elemente in der Liste sind
  669. int getEintragAnzahl() const
  670. {
  671. return count;
  672. }
  673. int getLastIndex() const
  674. {
  675. return count - 1;
  676. }
  677. //! Gibt den Wert des i-ten Elementes zurück mit erhöhtem Reference Counter
  678. //! \param i Der index des gesuchten Elementes, (0) wenn der Index nicht existiert
  679. TYP *get( int i ) const
  680. {
  681. if( i < 0 || i >= count )
  682. throwOutOfRange( __FILE__, __LINE__, i, count );
  683. ArrayEintrag< TYP * > *e = entries;
  684. for( int a = 0; a < i && e; ++a )
  685. e = e->next;
  686. if( e && e->set && e->var )
  687. return dynamic_cast<TYP *>(e->var->getThis());
  688. return (TYP *)0;
  689. }
  690. //! Gibt den Wert des i-ten Elementes zurück ohne erhöhten Reference Counter
  691. //! \param i Der index des gesuchten Elementes, (0) wenn der Index nicht existiert
  692. TYP *z( int i ) const //! gibt den index - ten T zurück
  693. {
  694. if( i < 0 || i >= count )
  695. throwOutOfRange( __FILE__, __LINE__, i, count );
  696. ArrayEintrag< TYP * > *e = entries;
  697. for( int a = 0; a < i && e; ++a )
  698. e = e->next;
  699. if( e && e->set && e->var )
  700. return (TYP *)e->var;
  701. return (TYP *)0;
  702. }
  703. //! Überprüft, ob ein Element in der Liste enthalten ist
  704. //! \param i Der Index des gesuchten Elementes
  705. //! \return (true), wenn der Index vorhanden ist. (false) sonnst
  706. bool hat( int i ) const
  707. {
  708. return i >= 0 && i < count;
  709. }
  710. RCArray &operator=( const RCArray &arr )
  711. {
  712. leeren();
  713. int anz = arr.getEintragAnzahl();
  714. for( int i = 0; i < anz; i++ )
  715. add( arr.get( i ) );
  716. return *this;
  717. }
  718. };
  719. }
  720. #endif