Qt-де контейнерлер элементтерін қайталауға арналған өзінің foreach кілт сөзі бар. Бұл кілт сөз C++ 11 стандартынан бұрын енгізілген және макрос болып табылады. C++11 стандартында қазіргі уақытта foreach сияқты функцияларды орындайтын ауқымға негізделген for циклдары бар.
Бірақ екі жағдайда да нюанстар бар. Оны анықтап көрейік.
Қандай нысандарға қолдануға болады
foreach және аралымға негізделген итераторы бар контейнерлерде жұмыс істейді. Бұл жағдайда ешқандай айырмашылықтар жоқ. Оларды тек итераторы бар нысандарға қолдануға болады.
Қолданба, бағдарлама коды
Бағдарлама кодын қарастырайық. Бізде контейнер бар делік
QList<QString> strings;
Біз осы контейнердің элементтерімен бірдеңе жасағымыз келеді, содан кейін foreach жазбасы келесідей болады
foreach (QString& str, strings) { // ToDo something }
және ауқымға негізделген жазба үшін келесідей болады
for (QString& str : strings) { // ToDo something }
Препроцессор
foreach кілт сөзі макрос болып табылады, ол сайып келгенде осы жазбаға баламалы болады
for (QList<QString>::iterator it = strings.begin(); it != strings.end(); ++it) { QString &str = *it; // ваш код }
Бірақ бұл макрос болғандықтан, біз барлық осы макростарды жұмыс кодына кеңейтетін препроцессор арқылы файлдарды өңдеуге арналған жинақтау уақытын аламыз.
Жұмыс жылдамдығы
foreach ауқымға негізделген for циклдарына қарағанда баяуырақ, себебі ол Qt құжаттамасында айтылғандай, контейнерді алдын ала көшіреді. Көшіруден келесі нюанс шығады, егер сіз foreach кезінде контейнермен бірдеңе жасасаңыз, ол циклдегі элементтерге әсер етпейді. Сонымен қатар, жасырын көшіру контейнер элементтерін өзгертпесеңіз де жүзеге асырылады.
Бұл код әрекеті көшіру қымбат тұратын STL контейнерлері үшін өте қымбат болуы мүмкін. Сондықтан, тек осы себепті C++ 11 стандартындағы диапазонға негізделген циклдарды қолданған дұрыс.
Диапазонға негізделген кемшіліктер
Qt құжаттамасын оқысаңыз, олар әлі де Qt контейнерлері үшін foreach пайдалануды ұсынатынын, ал қалғандарының барлығына (STL, Boost, т.б.) ауқымға негізделген for циклдерін пайдалануды ұсынатынын таба аласыз. Бұл үшін ауқымға негізделген циклдің жұмысы кезінде ортақ деректері бар контейнер осы ортақ ортақ деректерден күшпен ажыратылуы (ажырату) мүмкін және әрекеттерді орындағаннан кейін оның бастапқы күйіне қайтарылмайтындығы себеп болады. оның ішінде деректердің бүлінуі мүмкін.
Жанама ішінара класстардың бір класы - ажырату кезінде терең көшірмені пайдаланатын QPen.
void QPen::setStyle(Qt::PenStyle style) { detach(); // detach from common data d->style = style; // set the style member } void QPen::detach() { if (d->ref != 1) { ... // perform a deep copy } }
Қорытынды
Мен өз тәжірибемде Qt контейнерлері үшін диапазонға негізделген циклдарды пайдаланған кезде мұндай проблемаларға тап болған жоқпын, менің ойымша, бұл QPen нысандары немесе басқа Qt кластары бар контейнерлер үшін мұндай циклдарды жиі қолдануға тура келмегендіктен болуы мүмкін. ортақ деректерден бөлуді талап етеді.
Шын мәнінде, жақында STL контейнерлерін пайдалану маған Qt контейнерлеріне қарағанда ыңғайлырақ болып көрінеді, себебі контейнердегі нақты объектілерді іздеу (std:;find, std::find_if алгоритмдері және т.б.), шарты бар әдеттегі циклдің орнына.
Мен сонымен қатар MACROS орнына тілдік құрылымдарды қолданған дұрыс деген көзқарасты ұстанамын. Тек контейнерлердің және басқа Qt сыныптарының ерекшеліктеріне байланысты проблемалар болуы мүмкін екенін ескеру қажет.
Сондықтан foreach және диапазонға негізделген үшін пайдалану туралы соңғы шешім сіздің қалауыңызда.
Since Qt 5.7 the foreach macro is deprecated, Qt encourages you to use the C++11 for instead.
http://doc.qt.io/qt-5/qtglobal.html#foreach
(more details about the difference here : https://www.kdab.com/goodbye-q_foreach/ )