ה-STL היא ספריה של C++ המכילה classes מאד שימושיים. היא מחולקת ל-4 חלקים עיקריים:
1. Containers
2. Iterators
3. Algorithms
4. functors
אני מתכנן לכתוב מספר פוסטים בנושאים הנ"ל.
בפוסט הזה נתחיל עם הסבר כללי על Containers, ובהמשך נתייחס ונתין דוגמא ל-vector שהוא אחד ה-classes השימושיים מסוג container.
בכל אופן, מומלץ להכיר את נושא ה-templates - הנה קישור לפוסט שדן ב-templates.
אז מי הם ה-containers? זהו class שמאחסן קבוצה של אובייקטים - בדומה למערך. ה-containers בנויים כ-templates, כך ש-container יכול להכיל כל סוג של data type.
ה-containers מחולקים לשתי קבוצות משנה: sequence containers ו-associative containers.
הסבר על שני סוגי ה-containers:
sequence containers - כאן האובייקטים שב-container מאופיינים ע"י מיקומם ברשימה.
דוגמאות: vector. הוקטור מסודר בדיוק כמו מערך: רשימה של איברים, לכל אחד מספר סידורי, שמבטא את מיקומו בקבוצה.
דוגמא נוספת: list. גם כאן מדובר בקבוצה מסודרת. המיקום של כל איבר נקבע בזמן הכנסתו. הסדר יכול כמובן להשתנות ע"י פעולות שיעשו לאחר הכנסת האלמנט. למשל: הכנסה של אלמנטים נוספים, מיון, היפוך סדר וכו.
associative containers - כאן סדר האובייקטים ברשימה נקבע ע"י פעולת מיון הקשורה לתוכן של האובייקטים. לכל אובייקט כאן שני ערכים המאפיינים אותו: value type - שהוא התוכן של האובייקט. key_type - זהו המפתח על פיו ממוינת הרשימה. ישנם containers בהם ה-key type וה-value_type הם אותה ישות.
הנה רשימה של ה-containers הסטנדרטים:
אני מתכנן לכתוב מספר פוסטים בנושאים הנ"ל.
בפוסט הזה נתחיל עם הסבר כללי על Containers, ובהמשך נתייחס ונתין דוגמא ל-vector שהוא אחד ה-classes השימושיים מסוג container.
בכל אופן, מומלץ להכיר את נושא ה-templates - הנה קישור לפוסט שדן ב-templates.
אז מי הם ה-containers? זהו class שמאחסן קבוצה של אובייקטים - בדומה למערך. ה-containers בנויים כ-templates, כך ש-container יכול להכיל כל סוג של data type.
ה-containers מחולקים לשתי קבוצות משנה: sequence containers ו-associative containers.
הסבר על שני סוגי ה-containers:
sequence containers - כאן האובייקטים שב-container מאופיינים ע"י מיקומם ברשימה.
דוגמאות: vector. הוקטור מסודר בדיוק כמו מערך: רשימה של איברים, לכל אחד מספר סידורי, שמבטא את מיקומו בקבוצה.
דוגמא נוספת: list. גם כאן מדובר בקבוצה מסודרת. המיקום של כל איבר נקבע בזמן הכנסתו. הסדר יכול כמובן להשתנות ע"י פעולות שיעשו לאחר הכנסת האלמנט. למשל: הכנסה של אלמנטים נוספים, מיון, היפוך סדר וכו.
associative containers - כאן סדר האובייקטים ברשימה נקבע ע"י פעולת מיון הקשורה לתוכן של האובייקטים. לכל אובייקט כאן שני ערכים המאפיינים אותו: value type - שהוא התוכן של האובייקט. key_type - זהו המפתח על פיו ממוינת הרשימה. ישנם containers בהם ה-key type וה-value_type הם אותה ישות.
הנה רשימה של ה-containers הסטנדרטים:
Sequence Containers:
vector
list
dequeu
:Associative Containers
map
multimap
set
multiset
bitset
בנוסף יש containers נוספים שאינם שייכים לרשימה הסטנדרטית:
hash_map
hash_multimap
hash_set
hash_multiset
ה-containers הם כאמור classes והם כוללים מתודות המשמשות לפעולתם, למשל להכנסת אלמנטים ל-container, למחיקה, לסידור מחדש וכו,. חלק גדול מהמתודות נתמכות ע"י רב ה-containers, ויש כמובן מתדודות יחודיות לcontainers מסוימים או לקבוצה של containers. בהמשך נצלול לתוך חלק מה-containers ונבין את פעולתם. במידה שלא נכסה כאן את כולם, יהיה די קל להכנס אליהם אחרי הכרות עם ה-containers שמציג כאן.
אז ניגש כאן להכרות עם אחד ה-containers הנפוצים - vector.
כולנו מכירים את המערך (array) בשפת c.
אפשר להגדיר מערך בצורה סטטית, כמו בדוגמא הבאה:
נתחיל עם רשימת המתודות של vector. לאחר הרשימה נציג דוגמאות לשימוש במתודות, ונפרט את הפעולה של כל אחת מהן.
רשימת המתודות של vector לפי סדר אלפאבתי:
כולנו מכירים את המערך (array) בשפת c.
אפשר להגדיר מערך בצורה סטטית, כמו בדוגמא הבאה:
int a[10];
החיסרון הבולט: גודל המערך נקבע בזמן הקומפילציה. הוא כמובן לא ניתן להגדלה באופן דינמי.
אפשר גם להקצות את המערך באופן דינאמי. הנה דוגמא:
int array_size = 100;
int *a = new int[array_size];
כאן יש לדאוג להחזרת הזכרון בסיום (delete), ועדיין לא ניתן להגדיל\להקטין את גודל הזיכרון המוקצה.
נראה בהמשך שה-vector container מאפשר הקצאה דינאמית, והגדלת\הקטנת הקצאת הזיכרון אפשרית בכל שלב. אלו רק חלק מהיכולות של vector.
כמו כל ה-container, ה-vector הוא template class. ההגדרה שלו נראית כך:
template <class T, class Allocator = allocator<T>>
class vector{
..
..
..
}
ה-template של vector כולל שני ארגומנטים:
T: זהו ה-type של האלמנטים של ה-vector.
Allocator: ה-type של האובייקטים שמוגדרים עבור ה-storage allocation. זהו פרמטר אופציונלי, וברירת המחדל היא T.
נתחיל עם רשימת המתודות של vector. לאחר הרשימה נציג דוגמאות לשימוש במתודות, ונפרט את הפעולה של כל אחת מהן.
רשימת המתודות של vector לפי סדר אלפאבתי:
assign();
at();
back();
begin();
capacity();
clear();
data();
empty();
end();
erase();
front();
insert();
max_size();
operator =();
operator []();
pop_back();
push_back();
rbegin();
rend();
reserve();
resize();
size();
swap();
והאופרטורים:
==
!=
<
>
<=
>=
לפני שנציג דוגמא שמפעילה את המתודות הנ"ל, נציג אובייקט נוסף- iterator.
ה-iterator הוא התחליף לשימוש ב-pointer ומשמש לצורך גישה לאלמנטים של ה-containers ושל ה-vector בפרט.
(בסוף הדוגמא כמה מילים על סיבוכיות.)
(בסוף הדוגמא כמה מילים על סיבוכיות.)
הנה דוגמא ל-iterator:
- vector<int> vec;
- vector<int>::iterator iter;
- iter = vec.begin();
- iter++;
- *iter = 5;
הסבר קטע הקוד הנ"ל:
שורה 1: יצירת וקטור מסוג int.
שורה 2: יצירת iterator מסוג int.
שורה 3: המתודה begin מחזירה iterator מצביע על תחילת הוקטור.
שורה 4: הגדלת ה-iterator. כעת יצביע על האלמנט השני בוקטור.
שורה 5: כתיבת הערך 5 לאלמנט השני בוקטור.
כעת, עם מושג כלשהו על iterators, נראה דוגמא המשתמשת במתודות של vector.
לפנינו תוכנית שמשתמשת בכל המתודות של vector. התוכנית אורגנה כך שסדר ההופעה של המתודות הוא לפי הסדר האלפביתי (פרט ל begin ו-end שהסתננו קודם, כי התוכנית דרשה זאת...).
ליד כל מתודה בהופעתה הראשונה מצורף הסבר על פעולתה- ראה רקע צהוב. מעבר לכך, ה-comments בתוכנית מספקים הסבר מפורט. כדי להמנע מסרבול לא הוספתי פרוט בעברית.
אפשר כמובן להעתיק את התוכנית ולהריץ אותה (כרגיל).
- /*
- * vector.cpp
- *
- * Created on: Mar 19, 2012
- * Author: ronen
- */
- #include <vector> // Must include the vector header file
- #include <iostream>
- using namespace std;
- int main(){
- vector<int> vec; // here vec is created with size 0
- /* assign elements to the vector with size and value*/
- vec.assign(20,7); // here vec size is set to 20, objects value are all 7
- vector<int>::iterator iter1; // create an iterator
- vector<int>::iterator iter2; // create an iterator
- vector<int> new_vec; // create another int vector
- /** begin()
- * Returns a read/write iterator that points to the first
- * element in the %vector. Iteration is done in ordinary
- * element order.
- */
- iter1 = vec.begin() + 4; // iter1 points to the 5th element of vec.
- /** end()
- * Returns a read/write iterator that points one past the last
- * element in the %vector. Iteration is done in ordinary
- * element order.
- */
- iter2 = vec.end(); // iter2 points to the place after the last element of vec
- /* assign - iterator first ,iterator last */
- new_vec.assign(iter1, iter2);
- /* assign iterators anonymously: */
- new_vec.assign(vec.begin(), vec.end());// now new_vec[] will have 20 elements, contents copied from vec
- /**at(n)
- * @brief Provides access to the data contained in the vector.
- * @param n -The index of the element for which data should be
- * accessed.
- * @return Read/write reference to data.
- * @throw std::out_of_range If @a n is an invalid index.
- *
- * This function provides for safer data access. The parameter
- * is first checked that it is in the range of the vector. The
- * function throws out_of_range if the check fails.
- */
- int a = vec.at(6); // a=vec[6] which is now 7.
- cout << " check result of vec.at. vec.at(6)= " << vec.at(6) << " vec[6] = " << vec[6]<<endl;
- /** back()
- * Returns a read/write reference to the data at the last
- * element of the %vector.
- */
- a = vec.back(); // a will get value of last element of vec which is vec[19] equals 7.
- /** capacity()
- * Returns the total number of elements that the %vector can
- * hold before needing to allocate more memory.
- */
- a = vec.capacity(); // here the capacity is according to the size set by assign. it equals vec.size(). This may change if we use vec.reserve.
- /** clear()
- * Erases all the elements. Note that this function only erases the
- * elements, and that if the elements themselves are pointers, the
- * pointed-to memory is not touched in any way. Managing the pointer is
- * the user's responsibility.
- */
- vec.clear();
- /** empty()
- * Returns true if the %vector is empty. (Thus begin() would
- * equal end().)
- */
- if(!vec.empty())
- cout << "vec is not empty" << endl;
- else
- cout << " vec is empty"<<endl;
- vec.assign(20,7); // // After the vec.clear. vec is surely empty!. Let's assign it again with 20 elements
- /** end()
- * Returns a read/write iterator that points one past the last
- * element in the %vector. Iteration is done in ordinary
- * element order.
- */
- iter1 = vec.end(); // note: iter1 point to the entry beyond the vector's last element.
- /* erase vector at iterator position */
- vec.erase(vec.begin()+5); // the 6th element of the vector is erased. The vector is now shrinked to 19 elements.
- /* erase range between iterators positions */
- vec.erase(vec.begin(), vec.begin() + 10);// 1st to 10th elements are erased (range is always [ist, last) i.e. last element not included.
- // vector has now only 9 elements
- cout << "size of vector: " << vec.size() << endl;
- /** front
- * Returns a read/write reference to the data at the first
- * element of the %vector.
- */
- a = vec.front(); // a=vec[0];
- vec.front() = 4;// same as saying: vec[0] = 4;
- /* insert: insert new elements to the vector */
- vec.insert(vec.begin() + 3, new_vec.begin(), new_vec.begin()+5); // here 6 elements form new_vec are inserted after vec's 4th element.
- vec.insert(vec.begin(),20);// 20 is inserted after the first element
- vec.insert(vec.begin(),10, 5);// 10 elements with value of 5 are inserted after first element
- /** max_size() Returns the size() of the largest possible %vector. */
- vec.max_size();// have no idea which is the max possible size. let's print it
- cout << " vec.max size = " << vec.max_size()<< endl;
- //vec.operator =();
- vec = new_vec; // vec is assigned with new_vec's value
- //vec.operator []();
- a = vec[4]; // can read and write directly to the nth element.
- /* pop_back() @brief Removes last element.
- *
- * This is a typical stack operation. It shrinks the %vector by one.
- *
- * Note that no data is returned, and if the last element's
- * data is needed, it should be retrieved before pop_back() is
- * called.
- */
- vec.pop_back();// last element is popped. Vector shrinks by 1. Can't get the popped value from this method.
- /** push_back(n)
- * @brief Add data to the end of the %vector.
- * @param x Data to be added.
- *
- * This is a typical stack operation. The function creates an
- * element at the end of the %vector and assigns the given data
- * to it. Due to the nature of a %vector this operation can be
- * done in constant time if the %vector has preallocated space
- * available.
- */
- vec.push_back(11); // 11 is added to the end of vec. vec.size() is incremented by 1
- vec.push_back(12); // 12 is added to the end of vec. vec.size() is incremented by 1
- vec.push_back(13); // 13 is added to the end of vec. vec.size() is incremented by 1
- vec.push_back(14); // 14 is added to the end of vec. vec.size() is incremented by 1
- /** rbegin()
- * Returns a read/write reverse iterator that points to the
- * last element in the %vector. Iteration is done in reverse
- * element order.
- */
- /* rbegin() returns a reverse iterator, so let's create one:
- *
- */
- vector<int>::reverse_iterator riter1;
- vector<int>::reverse_iterator riter2;
- riter1 = vec.rbegin(); // now riter1 points to vec's last element. Note:Remember that vec.end() points one place after that.
- /** rend()
- * Returns a read/write reverse iterator that points to one
- * before the first element in the %vector. Iteration is done
- * in reverse element order.
- */
- riter2 = vec.rend(); // riter2 points on the place before vec's first element.
- // let's print vec in reverse order:
- vector<int>::reverse_iterator riter;
- cout << " vec in reverse order: " << endl;
- for ( riter=riter1 ; riter <riter2; ++riter )
- cout << *riter << ", ";
- // as a reference: here is the non reverse order:
- cout << "\nsame vec in regular order: " << endl;
- iter1 = vec.begin();
- iter2 = vec.end();
- vector<int>::iterator iter;
- for ( iter=iter1 ; iter <iter2; ++iter )
- cout << *iter << ", ";
- /* reserve(n) Set the allocated storage of the vector. It doesn't change the size of the vector, but the reserved capacity can avoid the expensive memory
- * relocation when expending the vector's size
- */
- vec.reserve(2); // here it will do nothing because capacity is more than 2
- vec.reserve(200); // the vector is preserved with a total of 200 elements. size is not changed.
- cout << " 200 places were resreved without affecting size:\n";
- cout << "vec.capacity() = " << vec.capacity() << ", vec.size() = " << vec.size() << endl;
- /** resize(n)
- * @brief Resizes the %vector to the specified number of elements.
- * @param new_size Number of elements the %vector should contain.
- * @param x Data with which new elements should be populated.
- *
- * This function will %resize the %vector to the specified
- * number of elements. If the number is smaller than the
- * %vector's current size the %vector is truncated, otherwise
- * the %vector is extended and new elements are populated with
- * given data.
- */
- vec.resize(50); // here size of vector is changed to 50 elements. No relocation is needed, as the vector is reserved with 200 elements.
- vec.resize(40, 3); // size is changed to 40 entries with value = 3
- /** Returns the number of elements in the %vector. */
- vec.size(); // size is 40 now
- cout << "\ncurrent vec size is " << vec.size();
- /** swap(vector)
- * @brief Swaps data with another %vector.
- * @param x A %vector of the same element and allocator types.
- *
- * This exchanges the elements between two vectors in constant time.
- * (Three pointers, so it should be quite fast.)
- * Note that the global std::swap() function is specialized such that
- * std::swap(v1,v2) will feed to this function.
- */
- // let print vectors before and after swap:
- cout << "\nprint vec before swap" << endl;
- for ( iter=vec.begin() ; iter < vec.end(); ++iter )
- cout << *iter << ", ";
- cout << "\nprint new_vec before swap" << endl;
- for ( iter=new_vec.begin() ; iter < new_vec.end(); ++iter )
- cout << *iter << ", ";
- vec.swap(new_vec); // here vec and new_vec are swapped.
- cout << "\nprint vec after swap" << endl;
- for ( iter=vec.begin() ; iter < vec.end(); ++iter )
- cout << *iter << ", ";
- cout << "\nprint new_vec after swap" << endl;
- for ( iter=new_vec.begin() ; iter < new_vec.end(); ++iter )
- cout << *iter << ", ";
- }
עוד על constructors של vector. הנה הצורות הקימות:
יצירת וקטור ריק.
- vector<T>vect;
יצירת וקטור עם n אלמנטים
- vector<T>vect(size_type n);
יצירת וקטור עם n אלמנטים שערכם t
- vector<T>vect(size_type n, const T &t);
יצירת וקטור שגודלו ותוכנו בהתאם לתחום המוגדר ע"י שני ה-iterators:
- vector<T>vect(begin_iterator, end_iterator);
כל 4 סוגי ה-constructors הנ"ל משולבים בקטע הקוד הבא.
מודגשים בצהוב - ה-constructors.
מודגשות בירוק - פקודות שליחה ל-output.
מתחת לסוף התוכנית - העתק הפלט שודפס על המסך.
- /*
- * vector_const.cpp
- *
- * Created on: Mar 20, 2012
- * Author: ronen halevy
- */
- #include <vector>
- #include <iostream>
- using namespace std;
- int main(void){
- vector <string>vect1;
- vector<string> vect2(5);
- string str = "start";
- vector<string> vect3(5, str);
- vector<string>vect_source(10, "asd");
- vector<string>vect4(vect_source.begin(), vect_source.end());
- vector<string>::iterator iter; // create an iterator
- cout << "\nvector <string>vect1; " << endl;
- for ( iter=vect1.begin() ; iter < vect1.end(); ++iter )
- cout << *iter << ", ";
- cout << "\nvector<string> vect2(5); " << endl;
- for ( iter=vect2.begin() ; iter < vect2.end(); ++iter )
- cout << *iter << ", ";
- cout << "\n vector<string> vect3(5, str);" << endl;
- for ( iter=vect3.begin() ; iter < vect3.end(); ++iter )
- cout << *iter << ", ";
- cout << "\nvector<string>vect4(vect_source.begin(), vect_source.end()); " << endl;
- for ( iter=vect4.begin() ; iter < vect4.end(); ++iter )
- cout << *iter << ", ";
- cout << endl;
- }
הפלט לשהודפס על המסך:
- vector <string>vect1;
- vector<string> vect2(5);
- , , , , ,
- vector<string> vect3(5, str);
- start, start, start, start, start,
- vector<string>vect4(vect_source.begin(), vect_source.end());
- asd, asd, asd, asd, asd, asd, asd, asd, asd, asd,
הסבר הפלט (מספרי השורות לא שייכים לפלט המקורי):
הפלט מציג את תוכנם 4 הוקטורים שנוצרו ע"י 4 ה-constructors.
שורות 1.3.5.7 - הן רק כותרות.
שורות 2,4,6,8 - הדפסה של 4 הוקטורים שאותחלו ע"י 4 ה-constructors:
שורות 1.3.5.7 - הן רק כותרות.
שורות 2,4,6,8 - הדפסה של 4 הוקטורים שאותחלו ע"י 4 ה-constructors:
שורה 2: ריקה כצפוי.
שורה 4: 5 אלמנטים לא מאותחלים בערך - הם blanks.
שורה 6: 5 אלמנטים עם הערך "start"
שורה 8: 10 אלמנטים עם הערך "asd"
כמה מילים על סיבוכיות:
יפה מאוד !!
השבמחקשימושי מאוד !!
יש מצב שאתה מסיר איך לעשות השמה : נניח יש לי מערך א" של פוינטרים (הוגדר ע"י וקטור) ואני מגדיר עוד מערך של פוינטרים ב"
השבמחקאיך אני עושה השמה ביניהם.