vectors.cpp 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316
  1. ///////////////////////////////////////////////////////////////////////////////
  2. // Name: tests/vectors/vectors.cpp
  3. // Purpose: wxVector<T> unit test
  4. // Author: Vaclav Slavik
  5. // Created: 2007-07-07
  6. // Copyright: (c) 2007 Vaclav Slavik
  7. ///////////////////////////////////////////////////////////////////////////////
  8. // ----------------------------------------------------------------------------
  9. // headers
  10. // ----------------------------------------------------------------------------
  11. #include "testprec.h"
  12. #ifdef __BORLANDC__
  13. #pragma hdrstop
  14. #endif
  15. #ifndef WX_PRECOMP
  16. #include "wx/wx.h"
  17. #endif // WX_PRECOMP
  18. #include "wx/vector.h"
  19. // ----------------------------------------------------------------------------
  20. // simple class capable of detecting leaks of its objects
  21. // ----------------------------------------------------------------------------
  22. class CountedObject
  23. {
  24. public:
  25. CountedObject(int n = 0) : m_n(n) { ms_count++; }
  26. CountedObject(const CountedObject& co) : m_n(co.m_n) { ms_count++; }
  27. ~CountedObject() { ms_count--; }
  28. int GetValue() const { return m_n; }
  29. static int GetCount() { return ms_count; }
  30. private:
  31. static int ms_count;
  32. int m_n;
  33. };
  34. int CountedObject::ms_count = 0;
  35. // ----------------------------------------------------------------------------
  36. // simple class capable of checking its "this" pointer validity
  37. // ----------------------------------------------------------------------------
  38. class SelfPointingObject
  39. {
  40. public:
  41. SelfPointingObject() { m_self = this; }
  42. SelfPointingObject(const SelfPointingObject&) { m_self = this; }
  43. ~SelfPointingObject() { CPPUNIT_ASSERT( this == m_self ); }
  44. // the assignment operator should not modify our "this" pointer so
  45. // implement it just to prevent the default version from doing it
  46. SelfPointingObject& operator=(const SelfPointingObject&) { return *this; }
  47. private:
  48. SelfPointingObject *m_self;
  49. };
  50. // ----------------------------------------------------------------------------
  51. // test class
  52. // ----------------------------------------------------------------------------
  53. class VectorsTestCase : public CppUnit::TestCase
  54. {
  55. public:
  56. VectorsTestCase() {}
  57. private:
  58. CPPUNIT_TEST_SUITE( VectorsTestCase );
  59. CPPUNIT_TEST( PushPopTest );
  60. CPPUNIT_TEST( Insert );
  61. CPPUNIT_TEST( Erase );
  62. CPPUNIT_TEST( Iterators );
  63. CPPUNIT_TEST( Objects );
  64. CPPUNIT_TEST( NonPODs );
  65. CPPUNIT_TEST( Resize );
  66. CPPUNIT_TEST( Swap );
  67. CPPUNIT_TEST( Sort );
  68. CPPUNIT_TEST_SUITE_END();
  69. void PushPopTest();
  70. void Insert();
  71. void Erase();
  72. void Iterators();
  73. void Objects();
  74. void NonPODs();
  75. void Resize();
  76. void Swap();
  77. void Sort();
  78. DECLARE_NO_COPY_CLASS(VectorsTestCase)
  79. };
  80. // register in the unnamed registry so that these tests are run by default
  81. CPPUNIT_TEST_SUITE_REGISTRATION( VectorsTestCase );
  82. // also include in its own registry so that these tests can be run alone
  83. CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( VectorsTestCase, "VectorsTestCase" );
  84. void VectorsTestCase::PushPopTest()
  85. {
  86. wxVector<int> v;
  87. CPPUNIT_ASSERT( v.size() == 0 );
  88. v.push_back(1);
  89. CPPUNIT_ASSERT( v.size() == 1 );
  90. v.push_back(2);
  91. CPPUNIT_ASSERT( v.size() == 2 );
  92. v.push_back(42);
  93. CPPUNIT_ASSERT( v.size() == 3 );
  94. CPPUNIT_ASSERT( v[0] == 1 );
  95. CPPUNIT_ASSERT( v[1] == 2 );
  96. CPPUNIT_ASSERT( v[2] == 42 );
  97. v.pop_back();
  98. CPPUNIT_ASSERT( v.size() == 2 );
  99. CPPUNIT_ASSERT( v[0] == 1 );
  100. CPPUNIT_ASSERT( v[1] == 2 );
  101. v.pop_back();
  102. CPPUNIT_ASSERT( v.size() == 1 );
  103. CPPUNIT_ASSERT( v[0] == 1 );
  104. v.pop_back();
  105. CPPUNIT_ASSERT( v.size() == 0 );
  106. CPPUNIT_ASSERT( v.empty() );
  107. wxVector<char> vEmpty;
  108. }
  109. void VectorsTestCase::Insert()
  110. {
  111. wxVector<char> v;
  112. v.insert(v.end(), 'a');
  113. CPPUNIT_ASSERT( v.size() == 1 );
  114. CPPUNIT_ASSERT( v[0] == 'a' );
  115. v.insert(v.end(), 'b');
  116. CPPUNIT_ASSERT( v.size() == 2 );
  117. CPPUNIT_ASSERT( v[0] == 'a' );
  118. CPPUNIT_ASSERT( v[1] == 'b' );
  119. v.insert(v.begin(), '0');
  120. CPPUNIT_ASSERT( v.size() == 3 );
  121. CPPUNIT_ASSERT( v[0] == '0' );
  122. CPPUNIT_ASSERT( v[1] == 'a' );
  123. CPPUNIT_ASSERT( v[2] == 'b' );
  124. v.insert(v.begin() + 2, 'X');
  125. CPPUNIT_ASSERT( v.size() == 4 );
  126. CPPUNIT_ASSERT( v[0] == '0' );
  127. CPPUNIT_ASSERT( v[1] == 'a' );
  128. CPPUNIT_ASSERT( v[2] == 'X' );
  129. CPPUNIT_ASSERT( v[3] == 'b' );
  130. }
  131. void VectorsTestCase::Erase()
  132. {
  133. wxVector<int> v;
  134. v.push_back(1);
  135. v.push_back(2);
  136. v.push_back(3);
  137. v.push_back(4);
  138. CPPUNIT_ASSERT( v.size() == 4 );
  139. v.erase(v.begin(), v.end()-1);
  140. CPPUNIT_ASSERT( v.size() == 1 );
  141. CPPUNIT_ASSERT( v[0] == 4 );
  142. v.clear();
  143. v.push_back(1);
  144. v.push_back(2);
  145. v.push_back(3);
  146. v.push_back(4);
  147. CPPUNIT_ASSERT( v.size() == 4 );
  148. v.erase(v.begin());
  149. CPPUNIT_ASSERT( v.size() == 3 );
  150. CPPUNIT_ASSERT( v[0] == 2 );
  151. CPPUNIT_ASSERT( v[1] == 3 );
  152. CPPUNIT_ASSERT( v[2] == 4 );
  153. }
  154. void VectorsTestCase::Iterators()
  155. {
  156. wxVector<int> v;
  157. v.push_back(1);
  158. v.push_back(2);
  159. v.push_back(3);
  160. v.push_back(4);
  161. int value = 1;
  162. for ( wxVector<int>::iterator i = v.begin(); i != v.end(); ++i, ++value )
  163. {
  164. CPPUNIT_ASSERT_EQUAL( value, *i );
  165. }
  166. }
  167. void VectorsTestCase::Objects()
  168. {
  169. wxVector<CountedObject> v;
  170. v.push_back(CountedObject(1));
  171. v.push_back(CountedObject(2));
  172. v.push_back(CountedObject(3));
  173. v.erase(v.begin());
  174. CPPUNIT_ASSERT_EQUAL( 2, v.size() );
  175. CPPUNIT_ASSERT_EQUAL( 2, CountedObject::GetCount() );
  176. v.clear();
  177. CPPUNIT_ASSERT_EQUAL( 0, CountedObject::GetCount() );
  178. }
  179. void VectorsTestCase::NonPODs()
  180. {
  181. wxVector<SelfPointingObject> v;
  182. v.push_back(SelfPointingObject());
  183. v.push_back(SelfPointingObject());
  184. v.push_back(SelfPointingObject());
  185. v.erase(v.begin());
  186. v.clear();
  187. // try the same with wxString, which is not POD, but is implemented in
  188. // a movable way (this won't assert, but would crash or show some memory
  189. // problems under Valgrind if wxString couldn't be safely moved with
  190. // memmove()):
  191. wxVector<wxString> vs;
  192. vs.push_back("one");
  193. vs.push_back("two");
  194. vs.push_back("three");
  195. vs.erase(vs.begin());
  196. vs.clear();
  197. }
  198. void VectorsTestCase::Resize()
  199. {
  200. wxVector<CountedObject> v;
  201. v.resize(3);
  202. CPPUNIT_ASSERT_EQUAL( 3, v.size() );
  203. CPPUNIT_ASSERT_EQUAL( 3, CountedObject::GetCount() );
  204. CPPUNIT_ASSERT_EQUAL( 0, v[0].GetValue() );
  205. CPPUNIT_ASSERT_EQUAL( 0, v[1].GetValue() );
  206. CPPUNIT_ASSERT_EQUAL( 0, v[2].GetValue() );
  207. v.resize(1);
  208. CPPUNIT_ASSERT_EQUAL( 1, v.size() );
  209. CPPUNIT_ASSERT_EQUAL( 1, CountedObject::GetCount() );
  210. v.resize(4, CountedObject(17));
  211. CPPUNIT_ASSERT_EQUAL( 4, v.size() );
  212. CPPUNIT_ASSERT_EQUAL( 4, CountedObject::GetCount() );
  213. CPPUNIT_ASSERT_EQUAL( 0, v[0].GetValue() );
  214. CPPUNIT_ASSERT_EQUAL( 17, v[1].GetValue() );
  215. CPPUNIT_ASSERT_EQUAL( 17, v[2].GetValue() );
  216. CPPUNIT_ASSERT_EQUAL( 17, v[3].GetValue() );
  217. }
  218. void VectorsTestCase::Swap()
  219. {
  220. wxVector<int> v1, v2;
  221. v1.push_back(17);
  222. v1.swap(v2);
  223. CPPUNIT_ASSERT( v1.empty() );
  224. CPPUNIT_ASSERT_EQUAL( 1, v2.size() );
  225. CPPUNIT_ASSERT_EQUAL( 17, v2[0] );
  226. v1.push_back(9);
  227. v2.swap(v1);
  228. CPPUNIT_ASSERT_EQUAL( 1, v1.size() );
  229. CPPUNIT_ASSERT_EQUAL( 17, v1[0] );
  230. CPPUNIT_ASSERT_EQUAL( 1, v2.size() );
  231. CPPUNIT_ASSERT_EQUAL( 9, v2[0] );
  232. v2.clear();
  233. v1.swap(v2);
  234. CPPUNIT_ASSERT( v1.empty() );
  235. }
  236. void VectorsTestCase::Sort()
  237. {
  238. size_t idx;
  239. wxVector<int> v;
  240. v.push_back(5);
  241. v.push_back(7);
  242. v.push_back(2);
  243. v.push_back(9);
  244. v.push_back(4);
  245. v.push_back(1);
  246. v.push_back(3);
  247. v.push_back(8);
  248. v.push_back(0);
  249. v.push_back(6);
  250. wxVectorSort(v);
  251. for (idx=1; idx<v.size(); idx++)
  252. {
  253. CPPUNIT_ASSERT( v[idx-1] <= v[idx] );
  254. }
  255. }