246071-Либерти-Освой-самостоятельно-С-за-21-день (852741), страница 88
Текст из файла (страница 88)
Видимость этой переменной, заданнаяоператором using, закончилась сразу за закрывающими фигурными скобками в предыдущейстрокепрограммы.В случае объявления внутри модуля локальных переменных все одноименные переменныепространства имен, открытые в этом модуле, будут скрыты. Это аналогично сокрытиюглобальных переменных локальными в случае совпадения их областей видимости.
Даже еслипеременная, объявленная в пространстве имен, будет открыта с помощью using послеобъявления локальной переменной, последняя все равно будет иметь приоритет. Это нагляднопоказановследующемпримере:namespaceWindow{intvalue1=20;intvalue2=40;}//...voidf(){intvalue2=10;usingnamespaceWindow;std::cout<<value2<<std::endl;}При выполнения этой функции на экране появится значение 10, а не 40, подтверждая тотфакт,чтопеременнаяvalue2пространстваименWindowскрываетсяпеременнойvalue2функцииf(). Если все же требуется использовать переменную пространства имен, явно укажите имяпространства.При использовании одноименных идентификаторов, один из которых объявлен какглобальный,адругой—внутрипространстваимен,такжеможетвозникнутьдвусмысленность.Чтобы избежать ее, всегда явно указывайте имя пространства при вызове объекта, как вследующемфрагментепрограммы:namespaceWindow{intvalue1=20;}//...usingnamespaceWindow;intvalue1=10;voidf(){value1=10;}В данном примере неопределенность возникает внутри функции f().
Оператор usingсообщает переменной Window::value1 глобальную область видимости. Однако в программеобъявляется другая глобальная переменная с таким же именем. Какая из них используется вфункции f()? Обратите внимание, что ошибка будет показана не во время объявленияодноименнойглобальнойпеременной,априобращениикнейвтелефункцииf().ИспользованиеusingвобъявленияхНазначение using в объявлениях идентификаторов аналогично использованию using какоператора с той лишь разницей, что обеспечивается более высокий уровень контроля. Этотспособиспользуетсядляоткрытиявидимоститолькодляодногоидентификатора,объявленноговпространствеимен,какпоказановследующемпримере:namespaceWindow{intvalue1=20;intvalue2=40;intvalue3=60;}//...usingWindow::value2;//открытиедоступакvalue2втекущеммодулеWindow::value1=10;//дляvalue1необходимоуказаниепространстваименvalue2=30;Window::value3=10;//дляvalue3необходимоуказаниепространстваименИтак,спомощьюusingможнооткрытьдоступвтекущуюобластьвидимостикотдельномуидентификаторупространстваимен,неповлиявнаостальныеидентификаторы,заданныевэтомпространстве.
В предыдущем примере переменная value2 вызывается без явного указанияпространстваимен,чтоневозможноприобращениикvalue1иvalue3.Использованиеusingприобъявленииобеспечиваетдополнительныйконтрольнадвидимостьюкаждогоидентификаторапространства имен. В этом и заключается отличие от использования using как оператора,открывающегодоступсразуковсемидентификаторампространстваимен.Видимостьименираспространяетсядоконцаблока,чтоможносказатьиолюбомдругомобъявлении. С помощью using идентификаторы можно объявлять как глобально, так и влокальнойобласти.Если в локальную область, где уже объявлен идентификатор из пространства имен,добавляется другой идентификатор с таким же именем, это приводит к ошибке компиляции.Ошибкой будет и объявление идентификатора из пространства имен в области, где ужесуществуетдругойидентификаторстакимжеименем.Этопоказановследующемпримере:namespaceWindow{intvalue1=20;intvalue2=40;}//...voidf(){intvalue2=10;usingWindow::value2;//рядобьявлеиийstd::cout<<value2<<std::endl;}Компиляция второй строки функции f() приведет к ошибке, поскольку переменная сименемvalue2вэтомблокеужеобъявлена.Тотжерезультатполучится,еслиобъявлениесusingразместитьпередобъявлениемлокальнойпеременнойvaluo2.Идентификатор пространства имен, введенный в локальную область с помошью using,скрывает аналогичный идентификатор, объявленный за пределами этой области.Проанализируйтеследующийпример:namespaceWindow{intvalue1-20;intva]ue2-40;}intvalue2=10;//...voidf(){usingWindow::value2;std::cout<<value2<<std::endl;}Объявлениепеременнойспомощьюusingвфункцииf()скрываетглобальнуюпеременнуюvalue2.Как отмечалось ранее, этот способ использования using позволяет дополнительноконтролировать области видимости отдельных идентификаторов пространства имен.
Операторusing открываетдоступ n локальной области ко всем идентификаторам, объявленным впространстве имен. Поэтому предпочтительней использовать using в объявлениях, а не какоператор, чтобы п полной мере воспользоваться всеми преимуществами, предоставляемымипространством имени. Явное расширение области видимости для отдельных идентификаторовпозволяетснизитьвероятностьвозникновенияконфликтовимен.Использованиеоператораusingоправданотольковтомслучае,еслинеобходимооткрытьдоступсразуковсемидентификаторампространстваимен.ПсевдонимыпространстваименПсевдонимы пространства имен используется для создания дополнительного имениименованного пространства.
Как правило, псевдоним представляет собой информативныйтермин, используемый для ссылки на пространство имен. Это весьма эффективно, если имяпространства очень длинное. Создание псевдонимов поможет упростить дальнейшую работу спространствамиимен.Рассмотримследующийпример:namespacethe_software_company{intvalue;//...}the_software_company::value=10;...namespaceTSC=the_software_company;TSC::value=20;Недостатокэтогометодасостоитвтом,чтоподобныйпсевдонимможетужесуществоватьвуказанной области видимости. В этом случае компилятор сообщит об ошибке, и вы сможетепростоизменитьпсевдоним.НеименованныепространстваименТакиепространстваименотличаютсяотименованныхтем,чтонеимеютимени.Наиболеечасто они используются для защиты глобальных данных от потенциальных конфликтов имен.Каждая единица программы имеет собственное уникальное неименованное пространство.
Всеидентификаторы,объявленныевнутритакогопространстваимен,вызываютсяпростопоименибез каких-либо префиксов. В следующем коде представлены примеры двух неименованныхпространств,расположенныхвдвухразныхфайлах.//файл:one.cppnamespace{intvalue;charp(char*p);//...}//файл:two.cppnamespace{intvalue;charp(char*p);//...}intmain(){charс=p(ptr);}В каждом файле объявляется переменная value и функция p().
Благодаря тому что длякаждого файла задано свое неименованное пространство, обращения к одноименнымидентификаторамвнутрифайловоднойпрограммынеприводиткконфликтамимен.Этохорошовидно при вызове функции p(). Функционирование неименованного пространства именаналогичноработестатическогообъектасвнешнейсвязью,такогокакstaticintvalue=10;Не забывайте, что подобное использование ключевого слова static не рекомендуетсякомитетомпостандартизации.Длярешенияподобныхзадачтеперьиспользуютсяпространстваимен. Можно провести еще одну аналогию с неименованными пространствами: они оченьпохожинаглобальныепеременныесвнутреннейсвязью.СтандартноепространствоименstdНаилучший пример пространств имен можно найти в стандартной библиотеке C++.
Всефункции,классы,объектыишаблоныстандартнойбиблиотекиобъявленывнутрипространстваименstd.Вероятно,вамприходилосьвидетьподобныевыражения:#include<iostream>usingnamespacestd;Не забывайте, что использование директивы using открывает доступ ко всемидентификаторам именованного пространства имен. Поэтому лучше не обращаться к помощиданного оператора при работе со стандартными библиотеками. Почему? Да потому, что такимобразом вы нарушите основное предназначение пространств имен. Глобальное пространствобудет буквально заполнено именами различных идентификаторов из файлов заголовковстандартнойбиблиотеки,большаячастькоторыхнеиспользуетсявданнойпрограмме.Помните,что во всех файлах заголовков используется средство пространства имен, поэтому, если вывключите в программу несколько файлов заголовков и используете оператор using, всеидентификаторы, объявленные в этих файлах заголовков, получат глобальную видимость.
Вымогли заметить, что в большинстве примеров данной книги это правило нарушается. Этосделано исключительно для краткости изложения примеров. Вам же следует использовать всвоихпрограммахобъявлениясключевымсловомusing,каквследующемпримере:#include<iostream>usingstd::cin;usingstd::cout;usingstd::endl;intmain(){intvalue=0;cout<<"So,howmanyeggsdidyousayyouwanted?"<<endl;cin>>value;cout<<value<<"eggs,sunny-sideup!"<<endl;return(0);}Выполнениеэтойпрограммыприведеткследующемувыводу:So,howmanyeggsdidyousayyouwanted?44eggs,sunny-sideup!В качестве альтернативы можно явно обращаться к идентификаторам, объявленным впространствеимен:#include<iostream>intmain(){intvalue=0;std::cout<<"Howmanyeggsdidyouwant?"<<std::endl;std::cin>>value;std::cout<<value<<"eggs,sunny-sideup!"<<std::endl;return(0);}Программавыведетследующиеданные:Howmanyeggsdidyouwant?44eggs,sunny-sideup!Такойподходвполнегодитсядлянебольшойпрограммы,новбольшихприложенияхбудетдовольно сложно проследить за всеми явными обращениями к идентификаторам пространстваимен.Толькопредставьтесебе:вампридетсядобавлятьstd::длякаждогоимениизстандартнойбиблиотеки!РезюмеСоздатьпространствоименитакжепросто,какописатькласс.Естьнесколькоразличий,ноонивесьманезначительны.Во-первых,послезакрывающейфигурнойскобкипространстваименнеследуетточкасзапятой.Во-вторых,пространствоименвсегдаоткрыто,втовремякакклассзакрыт.Этоозначает,чтовыможетепродолжитьобъявлениепространстваименвдругихфайлахиливразныхместаходногофайла.Вставлятьвпространствоименможновсе,чтоподлежитобъявлению.Создаваяклассыдлясвоей будущей библиотеки, вам следует взять на вооружение средство пространства имен.Объявленные внутри пространства имен функции должны определяться за его пределами.Благодаряэтомуинтерфейспрограммыотделяетсяотеевыполнения.Можно вкладывать одно пространство имен в другое.