Использование спецификатора void

Тип void (пустой) синтаксически ведет себя как основной тип. Однако использовать его можно только как часть производного типа, объектов типа void не существует. Он используется для того, чтобы указать, что функция не возвращает значения, или как базовый тип для указателей на объекты неизвестного типа.

void f() // f не возвращает значение

void* pv; // указатель на объект неизвестного типа

Переменной типа void* можно присваивать указатель любого типа. На первый взгляд это может показаться не особенно полезным, поскольку void* нельзя разыменовать, но именно это ограничение и делает тип void* полезным. Главным образом, он применяется для передачи указателей функциям, которые не позволяют сделать предположение о типе объекта, и для возврата из функций нетипизированных объектов. Чтобы использовать такой объект, необходимо применить явное преобразование типа. Подобные функции обычно находятся на самом нижнем уровне системы, там, где осуществляется работа с основными аппаратными ресурсами. Например:

void* allocate(int size); // выделить

void deallocate(void*); // освободить

f() {

int* pi = (int*)allocate(10*sizeof(int));

char* pc = (char*)allocate(10);

//...

deallocate(pi);

deallocate(pc);

}


 

Преобразование стандартных типов данных.

1.Неявное преобразование

2.Явное преобразование int(i)/int(j)

В языке С использовалась операция приведения типов следующего вида:

“(“имя типа”)”выражение

Наряду со старой формой операции приведения типов в С++ введен новый формат:

имя типа “(”выражение”)”

a = (double)b;

a = double (b); //функциональная форма

Новый формат не может быть использован, если имя состоит из нескольких слов.

 


 

Новые операции new, delete, операция расширения области видимости

Язык С++ имеет все операции языка С, но есть и дополнительные.

  1. Операция расширения области видимости –язык С++ имеет мощную структуру, т.е. одно и то же имя может обозначать различные объекты, в зависимости от вложенности блоков. При этом правила следующие: объявление идентификатора во внутреннем блоке перекрывает соответствующее объявление в охватывающем блоке, а также и глобальное объявление, если оно есть.

“: :” – данная операция позволяет обратиться к глобальным переменным, если последняя перекрывается локальной переменной.

#include <stdio.h>

int i=1;

void main(void)

{

int i=10;

{

int i=100;

printf(“%d %d”,i,::i);

}

printf(“%d %d”,i,::i);

}

Эта операция нарушает значения ввода, поэтому использовать ее надо крайне осторожно.

  1. Операции расширения динамической памяти –в языке С работать с динамической памятью можно было лишь через соответствующие функции распределения памяти, которые громоздкие и требовали ручной работы. В связи с этим, а также из-за специфики создания новых типов данных, в С++ введены новые операции: new и new[] для выделения памяти и delete и delete[] для удаления. С помощью new можно выделить для одиночного элемента, а с помощью new[] для массива. В случае успеха результатом операции будет адрес выделенного блока памяти. В случае неудачи: в старых версиях компилятора будет выделено значение 0, для современных компиляторов будет инициирована исключительная ситуация и программа аварийно остановится.

double *pd = new double;

double *a = new double[20];

double &dr = new double; //ошибка, т.к. new возвращает double

double &dr = *new double;

 

delete pd;

delete [] a;

delete &dr;

Следует отметить, что для освобождения массива данных надо всегда пользоваться операцией delete[]. Для встроенных типов данных разницы нет, но для данных, определяемых программистом, могут быть неприятности, связанные с отсутствием вызова специальных функций, называемых деструктор.В реальных системах операция new может выделить не более 64 кб памяти. Для функции освобождения нельзя пытаться освободить уже освобожденную область памяти.