С.Б. Липпман, Ж. Лажойе - Язык программирования С++ Вводный курс (1114944), страница 28
Текст из файла (страница 28)
Если его значением является true, оценивается expr2,int min( int ia, int ib )если false, то expr3. Данный фрагмент кода:{ return ( ia < ib ) ? ia : ib; }int min(int ia, int ib) {if (ia < ib)return ia;elsereturn ib;эквивалентен}Приведенная ниже программа иллюстрирует использование условного оператора:С++ для начинающих#include <iostream>int main(){int i =cout <<<<<<10, j = 20, k = 30;"Большим из "i << " и " << j << " является "( i > j ? i : j ) << end1;cout << "Значение " << i<< ( i % 2 ? " нечетно." : " четно." )<< endl;/* условный оператор может быть вложенным,* но глубокая вложенность трудна для восприятия.* В данном примере max получает значение* максимальной из трех величин*/int max = ( (i > j)? (( i > k) ? i : k): ( j > k ) ? j : k);cout << "Большим из "<< i << ", " << j << " и " << k<< " является " << max << endl;}Результатом работы программы будет:Большим из 10 и 20 является 20Значение 10 четно.4.8.
Оператор sizeofОператор sizeof возвращает размер в байтах объекта или типа данных. Синтаксис егоsizeof ( type name );sizeof ( object );таков:sizeof object;Результат имеет специальный тип size_t, который определен как typedef взаголовочном файле cstddef. Вот пример использования обеих форм оператора sizeof:155С++ для начинающих#include <cstddef>int ia[] = { 0, 1, 2 };// sizeof возвращает размер всего массиваsize_t array_size = sizeof ia;// sizeof возвращает размер типа intsize_t element_size = array_size / sizeof( int );Применение sizeof к массиву дает количество байтов, занимаемых массивом, а неколичество его элементов и не размер в байтах каждого из них.
Так, например, всистемах, где int хранится в 4 байтах, значением array_size будет 12. Применениеint *pi = new int[ 3 ];sizeof к указателю дает размер самого указателя, а не объекта, на который он указывает:size_t pointer_size = sizeof ( pi );Здесь значением pointer_size будет память под указатель в байтах (4 в 32-битныхсистемах), а не массива ia.Вот пример программы, использующей оператор sizeof:156С++ для начинающих157#include <string>#include <iostream>#include <cstddef>int main() {size_t ia;правильноправильноia = sizeof( ia ); //ia = sizeof ia;//// ia = sizeof int; // ошибкаia = sizeof( int ); // правильноint *picout <<<<<<= new int[ 12 ];"pi: " << sizeof( pi )" *pi: " << sizeof( pi )endl;// sizeof строки не зависит от// ее реальной длиныstring stl( "foobar" );string st2( "a mighty oak" );string *ps = &stl;cout <<<<<<<<<<" st1: " << sizeof( st1 )" st2: " << sizeof( st2 )" ps: sizeof( ps )" *ps: " << sizeof( *ps )endl;coutcoutcoutcout"short :\t""shorf" :\t""short& :\t""short[3] :\t"<<<<<<<<<<<<<<<<sizeof(short)sizeof(short*)sizeof(short&)sizeof(short[3])<<<<<<<<endl;endl;endl;endl;}Результатом работы программы будет:pi: 4 *pi: 4st1: 12 st2: 12 ps: 4 *ps:12short : 2short* : 4short& : 2short[3] : 6Из данного примера видно, что применение sizeof к указателю позволяет узнать размерпамяти, необходимой для хранения адреса.
Если же аргументом sizeof является ссылка,мы получим размер связанного с ней объекта.// char_size == 1Гарантируется, что в любой реализации С++ размер типа char равен 1.size_t char_size = sizeof( char );С++ для начинающихЗначение оператора sizeof вычисляется во время компиляции и считается константой.Оно может быть использовано везде, где требуется константное значение, в том числе в// правильно: константное выражениекачестве размера встроенного массива. Например:int array[ sizeof( some_type_T )];4.9. Операторы new и deleteКаждая программа во время работы получает определенное количество памяти, которуюможно использовать. Такое выделение памяти под объекты во время выполненияназывается динамическим, а сама память выделяется из хипа (heap).
(Мы уже касалисьвопроса о динамическом выделении памяти в главе 1.) Напомним, что выделение памятиобъекту производится с помощью оператора new, возвращающего указатель на вновьсозданный объект того типа, который был ему задан. Например:int *pi = new int;размещает объект типа int в памяти и инициализирует указатель pi адресом этогообъекта.
Сам объект в таком случае не инициализируется, но это легко изменить:int *pi = new int( 1024 );Можно динамически выделить память под массив:int *pia = new int[ 10 ];Такая инструкция размещает в памяти массив встроенного типа из десяти элементов типаint. Для подобного массива нельзя задать список начальных значений его элементов придинамическом размещении. (Однако если размещается массив объектов типа класса, тодля каждого из элементов вызывается конструктор по умолчанию.) Например:string *ps = new string;размещает в памяти один объект типа string, инициализирует ps его адресом ивызывает конструктор по умолчанию для вновь созданного объекта типа string.Аналогичноstring *psa = new string[10];размещает в памяти массив из десяти элементов типа string, инициализирует psa егоадресом и вызывает конструктор по умолчанию для каждого элемента массива.Объекты, размещаемые в памяти с помощью оператора new, не имеют собственногоимени.
Вместо этого возвращается указатель на безымянный объект, и все действия сэтим объектом производятся посредством косвенной адресации.После использования объекта, созданного таким образом, мы должны явно освободитьпамять, применив оператор delete к указателю на этот объект. (Попытка применить158С++ для начинающихоператор delete к указателю, не содержащему адрес объекта, полученного описаннымспособом, вызовет ошибку времени выполнения.) Например:delete pi;освобождает память, на которую указывает объект типа int, на который указывает pi.Аналогичноdelete ps;освобождает память, на которую указывает объект класса string, адрес которогосодержится в ps.
Перед уничтожением этого объекта вызывается деструктор. Выражениеdelete [] pia;освобождает память, отведенную под массив pia. При выполнении такой операциинеобходимо придерживаться указанного синтаксиса.(Об операциях new и delete мы еще поговорим в главе 8.)Упражнение 4.11Какие из следующих выражений ошибочны?(a) vector<string> svec( 10 );(b) vector<string> *pvecl = new vector<string>(10);(c) vector<string> **pvec2 = new vector<string>[10];(d) vector<string> *pvl = &svec;(e) vector<string> *pv2 = pvecl;(f) delete svec;(g) delete pvecl;(h) delete [] pvec2;(i) delete pvl;(j) delete pv2;4.10.
Оператор “запятая”Одно выражение может состоять из набора подвыражений, разделенных запятыми; такиеподвыражения вычисляются слева направо. Конечным результатом будет результатсамого правого из них. В следующем примере каждое из подвыражений условногооператора представляет собой список. Результатом первого подвыражения условногооператора является ix, второго – 0.159С++ для начинающих160int main(){// примеры оператора "запятая"// переменные ia, sz и index определены в другом месте ...int ival = (ia != 0)? ix=get_va1ue(), ia[index]=ix: ia=new int[sz], ia[index]=0;// ...}4.11. Побитовые операторыТаблица 4.3.
Побитовые операторыСимвол операцииЗначениеИспользование~Побитовое НЕ~expr<<Сдвиг влевоexpr1 << expr2>>Сдвиг вправоexpr1 >> expr2&Побитовое Иexpr1 & expr2^ПобитовоеИСКЛЮЧАЮЩЕЕ ИЛИexpr1 ^ expr2|Побитовое ИЛИexpr1 | expr2&=Побитовоеприсваиванием^=expr1 ^= expr2ПобитовоеИСКЛЮЧАЮЩЕЕ ИЛИ сприсваиванием|=ПобитовоеИЛИприсваиваниемс expr1 |= expr2<<=Сдвигвлевоприсваиваниемс expr1 <<= expr2>>=Сдвигвправоприсваиваниемс expr1 >>= expr2Ис expr1 &= expr2Побитовые операции рассматривают операнды как упорядоченные наборы битов,каждый бит может иметь одно из двух значений – 0 или 1.
Такие операции позволяютпрограммисту манипулировать значениями отдельных битов. Объект, содержащий наборбитов, иногда называют битовым вектором. Он позволяет компактно хранить наборфлагов – переменных, принимающих значение “да” “нет”. Например, компиляторызачастую помещают в битовые векторы спецификаторы типов, такие, как const иvolatile. Библиотека iostream использует эти векторы для хранения состоянияформата вывода.Как мы видели, в С++ существуют два способа работы со строками: использование Cстрок и объектов типа string стандартной библиотеки – и два подхода к массивам:массивы встроенного типа и объект vector.