C++ Primer 学习笔记_91_用于大型程序的工具 --命名空间

用于大型程序的工具

--命名空间

引言:

在一个给定作用域中定义的每个名字在该作用域中必须是唯一的,对庞大、复杂的应用程序而言,这个要求可能难以满足。这样的应用程序的全局作用域中一般有许多名字定义。由独立开发的库构成的复杂程序更有可能遇到名字冲突 —— 同样的名字既可能在我们自己的代码中使用,也可能(更常见地)在独立供应商提供的代码中使用

库倾向于定义许多全局名字 —— 主要是模板名、类型名或函数名。在使用来自多个供应商的库编写应用程序的时候,这些名字中有一些几乎不可避免地会发生冲突,这种名字冲突问题称为命名空间污染问题。

传统上,程序员通过将全局实体的名字设得很长来避免命名空间污染,经常用特定字符序列作为程序中名字的前缀:

class cplusplus_primer_Query
{
    //...
};
ifstream &
cplusplus_primer_open_file(ifstream &,const string &);

程序员编写和阅读使用这种长名字的程序非常麻烦。

命名空间为防止名字冲突提供了更加可控的机制,命名空间能够划分全局名字空间,这样使用独立开发的库就更加容易了。一个命名空间是一个作用域,通过在命名空间内部定义库中的名字,库的作者(以及用户)可以避免全局名字固有的限制。

一、命名空间的定义

以关键字namespace开始,后接命名空间的名字:

namespace cplusplus_primer
{
class Sales_item
{
    //...
};
Sales_item operator+(const Sales_item &,
                     const Sales_item &);

class Query
{
public:
    Query(const std::string &);
    std::ostream &display(std::ostream &) const;

    //...
};

class Query_base
{
    //...
};
}

定义了cplusplus_primer的命名空间,它有四个成员:两个类,一个重载的+操作符,一个函数。

像其他名字一样,命名空间的名字在定义该命名空间的作用域中必须是唯一的。命名空间可以在全局作用域或其他作用域内部定义,但不能在函数或类内部定义

命名空间名字后面接着由花括号括住的一块声明和定义,可以在命名空间中放入可以出现在全局作用域的任意声明:类、变量(以及它们的初始化)、函数(以及它们的定义)、模板以及其他命名空间

命名空间作用域不能以分号结束

1、每个命名空间是一个作用域

定义在命名空间中的实体称为命名空间成员。其中的每个名字必须引用该命名空间中的唯一实体。不同命名空间可以具有同名成员

在命名空间中定义的名字可以被命名空间中的其他成员直接访问,命名空间外部的代码必须指出名字定义在哪个命名空间中:

    cplusplus_primer::Query q = cplusplus_primer::Query("hello");
    q.display(cout);

如果另一命名空间(如AddisonWesley)也提供TextQuery类,而且我们想要使用那个类代替cplusplus_primer中定义的TextQuery,可以通过这样修改代码而实现:

    AddisonWesley::Query q = AddisonWesley::Query("hello");
    q.display(cout);

2、从命名空间外部使用命名空间成员

可以使用using声明来获得对我们知道将经常使用的名字的直接访问:

    using cplusplus_primer::Query;
    Query q = Query("world");
    q.display(cout);

3、命名空间可以是不连续的

与其他作用域不同,命名空间可以在几个部分中定义。命名空间由它的分离定义部分的总和构成,命名空间是累积的。一个命名空间的分离部分可以分散在多个文件中,在不同文本文件中的命名空间定义也是累积的。当然,名字只在声明名字的文件中可见,这一常规限制继续应用,所以,如果命名空间的一个部分需要定义在另一文件中的名字,仍然必须声明该名字

编写命名空间定义:

namespace namespace_name
{
//...
}

既可以定义新的命名空间,也可以添加到现存命名空间中。

如果名字namespace_name不是引用前面定义的命名空间,则用该名字创建新的命名空间,否则,这个定义打开一个已存在的命名空间,并将这些新声明加到那个命名空间。

4、接口和实现的分离

可以用分离的接口文件和实现文件构成命名空间,因此,可以用与管理自己的类和函数定义相同的方法来组织命名空间:

1.定义类的命名空间成员,以及作为类接口的一部分的函数声明与对象声明,可以放在头文件中,使用命名空间成员的文件可以包含这些头文件

2.命名空间成员的定义可以放在单独的源文件中。

这个要求同样适用于命名空间中定义的名字。通过将接口和实现分离,可以保证函数和其他我们需要的名字只定义一次,但相同的声明可以在任何使用该实体的地方见到

【最佳实践】

定义多个不相关类型的命名空间应该使用分离的文件,表示该命名空间定义的每个类型。

5、定义本书的命名空间

使用将接口和实现分离的策略:

//1 ---Sales_item.h---
#ifndef SALES_ITEM_H_INCLUDED
#define SALES_ITEM_H_INCLUDED

namespace cplusplus_primer
{
class Sales_item
{
    //...
};

Sales_item operator+(const Sales_item &,const Sales_item &);
}

#endif // SALES_ITEM_H_INCLUDED

//2 ---Query.h---
#ifndef QUERY_H_INCLUDED
#define QUERY_H_INCLUDED

#include <fstream>
#include <string>

namespace cplusplus_primer
{
class Query
{
public:
    Query(const std::string &);
    std::ostream &display(std::ostream &) const;
};

class Query_base
{
    //...
};
}

#endif // QUERY_H_INCLUDED

//3 ---Sales_item.cpp---
#include "Sales_item.h"
namespace cplusplus_primer
{
//定义
}

//4 ---Query.cpp---
#include "Query.h"
namespace cplusplus_primer
{
//定义
}

这种程序组织给予开发者库用户必要的模块性。每个类仍组织在自己的接口和实现文件中,一个类的用户不必编译与其他类相关的名字。如果允许 Sales_item.cpp和 main.cpp文件编译和链接到一个程序而不会导致编译时错误和运行时错误,就可以对用户隐藏实现。库的开发者可以独立工作于每个类型的实现。

使用我们的库的程序可以包含需要的头文件,那些头文件中的名字定义在命名空间 cplusplus_primer内部:

// --- main.cpp ---
#include "Sales_item.h"

int main()
{
    cplusplus_primer::Sales_item trans1,trans2;
    //...
}

6、定义命名空间成员

在命名空间内部定义的函数可以使用同一命名空间中定义的名字的简写形式:

namespace cplusplus_primer
{
std::istream &operator>>(std::istream &in,Sales_item &)
{
    //...
    return in;
}
}

在命名空间的外部定义命名空间成员:名字的命名空间声明必须在作用域中,并且定义必须指定该名字所属的命名空间:

cplusplus_primer::Sales_item
cplusplus_primer::operator+(const Sales_item &lhs,
                            const Sales_item &rhs)
{
    Sales_item ret(lhs);
    //...
    return ret;
}

定义看起来类似于定义在类外部的类成员函数,返回类型和函数名由命名空间名字限定。一旦看到完全限定的函数名,就处于命名空间的作用域中。因此,形参表和函数体中的命名空间成员引用可以使用非限定名引用Sales_item。

7、不能在不相关的命名空间中定义成员

虽然可以在命名空间定义的外部定义命名空间成员,对这个定义可以出现的地方仍有些限制:只有包围成员声明的命名空间可以包含成员的定义。例如,operator+ 既可以定义在命名空间cplusplus_primer中,也可以定义在全局作用域中,但它不能定义在不相关的命名空间中。

8、全局命名空间

定义在全局作用域的名字(在任意类、函数或命名空间外部声明的名字)是定义在全局命名空间中的。全局命名空间是隐式声明的,存在于每个程序中。在全局作用域定义实体的每个文件将那些名字加到全局命名空间。

可以用作用域操作符引用全局命名空间的成员。因为全局命名空间是隐含的,它没有名字,所以记号

    ::member_name;

引用全局命名空间的成员。

//P603 习题17.13/14/15
//--- Bookstore.h ---
#ifndef BOOKSTORE_H_INCLUDED
#define BOOKSTORE_H_INCLUDED

#include <stdexcept>
#include <string>
#include <iostream>

namespace Bookstore
{
class out_of_stock : public std::runtime_error
{
public:
    out_of_stock(const std::string &s):
        runtime_error(s) {}
};

class isbn_mismatch : public std::logic_error
{
public:
    isbn_mismatch(const std::string &s):logic_error(s) {}
    isbn_mismatch(const std::string &s,
                  const std::string &lhs,
                  const std::string &rhs):
        logic_error(s),left(lhs),right(rhs) {}

    std::string left,right;
    virtual ~isbn_mismatch() throw () {}
};

class Sales_item
{
    friend Sales_item operator+(const Sales_item &,const Sales_item &);
    friend std::ostream &operator<<(std::ostream &,const Sales_item &);
    friend std::istream &operator>>(std::istream &,Sales_item &);

public:
    Sales_item(const std::string &s = ""):
        isbn(s),units_sold(0),revenue(0) {}
    Sales_item(std::istream &in)
    {
        in >> *this;
    }

    std::string book() const
    {
        return isbn;
    }

    bool same_isbn(const Sales_item &rhs) const
    {
        return isbn == rhs.isbn;
    }

    void setVal()
    {
        std::cin >> isbn >> units_sold >> revenue;
    }

    Sales_item &operator+=(const Sales_item &rhs)
    {
        units_sold += rhs.units_sold;
        revenue += rhs.revenue;

        return *this;
    }

private:
    std::string isbn;
    unsigned units_sold;
    double revenue;
};
}

#endif // BOOKSTORE_H_INCLUDED

// --- Bookstore.cpp ---
#include "Bookstore.h" 

namespace Bookstore
{
    Sales_item operator+(const Sales_item &lhs,const Sales_item &rhs)
    {
        if (!lhs.same_isbn(rhs))
        {
            throw isbn_mismatch("isbn mismatch ",lhs.book(),rhs.book());
        } 

        Sales_item ret(lhs);
        ret += rhs; 

        return ret;
    } 

    std::ostream &operator<<(std::ostream &os,const Sales_item &rhs)
    {
        os << rhs.book() << "\t" << rhs.units_sold << "\t" << rhs.revenue;
        return os;
    } 

    std::istream &operator>>(std::istream &in,Sales_item &rhs)
    {
        double price;
        in >> rhs.isbn >> rhs.units_sold >> price; 

        if (in)
        {
            rhs.revenue = price * rhs.units_sold;
        }
        else
        {
            rhs = Sales_item();
        } 

        return in;
    }
}

//--- MyApp.h ---
#ifndef MYAPP_H_INCLUDED
#define MYAPP_H_INCLUDED

#include "Bookstore.h"

namespace MyApp
{
void processTrans()
{
    Bookstore::Sales_item item1,item2,sum;

    while (std::cin >> item1 >> item2)
    {
        try
        {
            sum = item1 + item2;
        }
        catch (Bookstore::isbn_mismatch &e)
        {
            std::cerr << e.what() << ": left isbn(" << e.left
                      << "); right isbn(" << e.right << ")" << std::endl;
            continue;
        }

        std::cout << sum << std::endl;
    }
}
}

#endif // MYAPP_H_INCLUDED

C++ Primer 学习笔记_91_用于大型程序的工具 --命名空间

时间: 2024-12-21 15:44:38

C++ Primer 学习笔记_91_用于大型程序的工具 --命名空间的相关文章

C++ Primer 学习笔记_93_用于大型程序的工具 --命名空间[续2]

用于大型程序的工具 --命名空间[续2] 五.类.命名空间和作用域 名字的可见性穿过任意嵌套作用域,直到引入名字的块的末尾. 对命名空间内部使用的名字的查找遵循常规C++查找规则:当查找名字的时候,通过外围作用域外查找.对命名空间内部使用的名字而言,外围作用域可能是一个或多个嵌套的命名空间,最终以全包围的全局命名空间结束.只考虑已经在使用点之前声明的名字,而该使用仍在开放的块中: namespace A { int i; namespace B { int i; int j; int f1()

C++ Primer 学习笔记_94_用于大型程序的工具 --命名空间[续3]

用于大型程序的工具 --命名空间[续3] 六.重载与命名空间 正如我们所见,每个命名空间维持自己的作用域,因此,作为两个不同命名空间的成员的函数不能互相重载.但是,给定命名空间可以包含一组重载函数成员. 1.候选函数与命名空间 命名空间对函数匹配有两个影响.一个影响是明显的:using声明或using 指示可以将函数加到候选集合.另一个影响则微妙得多. 正如前节所见,有一个或多个类类型形参的函数的名字查找包括定义每个形参类型的命名空间.这个规则还影响怎样确定候选集合,为找候选函数而查找定义形参类

C++ Primer 学习笔记_92_用于大型程序的工具 --命名空间[续1]

用于大型程序的工具 --命名空间[续1] 二.嵌套命名空间 一个嵌套命名空间即是一个嵌套作用域 -- 其作用域嵌套在包含它的命名空间内部.嵌套命名空间中的名字遵循常规规则:外围命名空间中声明的名字被嵌套命名空间中同一名字的声明所屏蔽.嵌套命名空间内部定义的名字局部于该命名空间.外围命名空间之外的代码只能通过限定名引用嵌套命名空间中的名字. 嵌套命名空间可以改进库中代码的组织: namespace cplusplus_primer { namespace QueryLib { class Quer

C++ Primer 学习笔记_91_用以大型程序的工具 -命名空间

用于大型程序的工具 --命名空间 引言: 在一个给定作用域中定义的每个名字在该作用域中必须是唯一的,对庞大.复杂的应用程序而言,这个要求可能难以满足.这样的应用程序的全局作用域中一般有许多名字定义.由独立开发的库构成的复杂程序更有可能遇到名字冲突 -- 同样的名字既可能在我们自己的代码中使用,也可能(更常见地)在独立供应商提供的代码中使用. 库倾向于定义许多全局名字 -- 主要是模板名.类型名或函数名.在使用来自多个供应商的库编写应用程序的时候,这些名字中有一些几乎不可避免地会发生冲突,这种名字

C++ Primer 学习笔记_89_用于大型程序的工具 --异常处理[续2]

用于大型程序的工具 --异常处理[续2] 八.自动资源释放 考虑下面函数: void f() { vector<string> v; string s; while (cin >> s) { v.push_back(s); } string *p = new string[v.size()]; //... delete p; } 在正常情况下,数组和vector都在退出函数之前被撤销,函数中最后一个语句释放数组,在函数结束时自动撤销vector. 但是,如果在函数内部发生异常,则将

C++ Primer 学习笔记_90_用于大型程序的工具 --异常处理[续3]

用于大型程序的工具 --异常处理[续3] 九.auto_ptr类[接上] 5.auto_ptr对象的复制和赋值是破坏性操作 auto_ptr和内置指针对待复制和赋值有非常关键的区别.当复制auto_ptr对象或者将它的值赋给其他auto_ptr对象的时候,将基础对象的所有权从原来的auto_ptr对象转给副本,原来的auto_ptr对象重置为未绑定状态. auto_ptr<string> strPtr1(new string("HELLO!")); auto_ptr<

C++ Primer 学习笔记_95_用于大型程序的工具 --多重继承与虚继承

用于大型程序的工具 --多重继承与虚继承 引言: 大多数应用程序使用单个基类的公用继承,但是,在某些情况下,单继承是不够用的,因为可能无法为问题域建模,或者会对模型带来不必要的复杂性. 在这些情况下,多重继承可以更直接地为应用程序建模.多重继承是从多于一个直接基类派生类的能力,多重继承的派生类继承其所有父类的属性. 一.多重继承 1.定义多个类 为了支持多重继承,扩充派生列表: class Bear : public ZooAnimal { //... }; 以支持由逗号分隔的基类列表: cla

C++ Primer 学习笔记_96_用于大型程序的工具 --多重继承与虚继承[续1]

用于大型程序的工具 --多重继承与虚继承[续1] 四.多重继承下的类作用域 成员函数中使用的名字和查找首先在函数本身进行,如果不能在本地找到名字,就继续在本类中查找,然后依次查找每个基类.在多重继承下,查找同时检察所有的基类继承子树 -- 在我们的例子中,并行查找 Endangered子树和Bear/ZooAnimal子树.如果在多个子树中找到该名字,则那个名字的使用必须显式指定使用哪个基类;否则,该名字的使用是二义性的. [小心地雷] 当一个类有多个基类的时候,通过对所有直接基类同时进行名字查

C++ Primer 学习笔记_97_用于大型程序的工具 --多重继承与虚继承[续2]

用于大型程序的工具 --多重继承与虚继承[续2] 七.特殊的初始化语义 从具有虚基类的类继承的类对初始化进行特殊处理:在虚基类中,由最低层派生类的构造函数初始化虚基类.在ZooAnimal示例中,使用常规规则将导致Bear 类和 Raccoon类都试图初始化Panda对象的ZooAnimal类部分. 虽然由最低层派生类初始化虚基类,但是任何直接或间接继承虚基类的类一般也必须为该基类提供自己的初始化式.只要可以创建虚基类派生类类型的独立对象,该类就必须初始化自己的虚基类,这些初始化只在创建中间类型