C++实践参考:IP地址类

简介: 【项目-IP地址类】   在互联网中使用的IP地址占4字节,可以用四段法表示,每段值的范围为0-255,中间用“.”隔开,例如202.194.116.97。其实,也可以看看一个有4字节的无符号整型值3401741409。   现设计一个IP地址类,用于保存IP地址,并实施在IP地址上的一些操作。如下所示:class IP{private: union //

【项目-IP地址类】
  在互联网中使用的IP地址占4字节,可以用四段法表示,每段值的范围为0-255,中间用“.”隔开,例如202.194.116.97。其实,也可以看看一个有4字节的无符号整型值3401741409。
  现设计一个IP地址类,用于保存IP地址,并实施在IP地址上的一些操作。如下所示:

class IP
{
private:
    union //由此匿名联合体可以看出,IP地址共占4个字节
    {
        struct //这是一个由4个字节构成的匿名结构体
        {
            unsigned char seg0;
            unsigned char seg1;
            unsigned char seg2;
            unsigned char seg3;
        };  //4字节的IP地址可以看作4部分,每部分1字节
        unsigned int address; //4字节的IP地址可以看成一个4字节的整体
    };
public:
    IP(int=0,int=0,int=0,int=0);  //构造函数
    void showIP();  //用四段法显示IP地址
    bool sameSubnet(const IP &ip, const IP &mark);  //判断是否处于同一子网
    char whatKind();  //返回属于哪一类网络
};
//实现成员函数

int main()
{
    IP ip1(202,194,116,97), ip2(202,194,119,102), mark(255,255,248,0);
    cout<<"ip1: ";
    ip1.showIP();
    cout<<"ip2: ";
    ip2.showIP();
    if(ip1.sameSubnet(ip2,mark))
        cout<<"两个IP在同一子网"<<endl;
    else
        cout<<"两个IP不在同一子网"<<endl;
    cout<<"ip1属于"<<ip1.whatKind()<<"类网络"<<endl;
    return 0;
}

  相关的背景知识请通过搜索引擎找到。在给出的代码中,也通过注释给了一些背景知识的介绍。
  如果觉得写出来太难,可以将本题作为代码阅读题看待。

参考解答1:

#include <iostream>
using namespace std;
class IP
{
private:
    union
    {
        struct
        {
            unsigned char seg0;
            unsigned char seg1;
            unsigned char seg2;
            unsigned char seg3;
        };  //IP地址共4字节,或者看成结构体所指4部分
        unsigned int address; //或者看成一个整体
    };
public:
    IP(int=0,int=0,int=0,int=0);  //构造函数
    void showIP();  //用四段法显示IP地址
    bool sameSubnet(const IP &ip, const IP &mark);  //判断是否处于同一子网
    char whatKind();  //返回属于哪一类网络
};

IP::IP(int s0,int s1,int s2,int s3)
{
    //下面赋值的顺序,涉及整型数据存储结构,参考[《整型数据在内存中如何存储?》](http://blog.csdn.net/sxhelijian/article/details/51015706)
    seg3=s0;
    seg2=s1;
    seg1=s2;
    seg0=s3;
}

void IP::showIP()
{
    //显示结果,是给人看的,分4段合适
    cout<<int(seg3)<<"."<<int(seg2)<<"."<<int(seg1)<<"."<<int(seg0)<<endl;
    return;
}

bool IP::sameSubnet(const IP &ip, const IP &mark)
{
    //与子网掩码按位与,是计算机内部的操作,直接以一个整体操作更方便
    unsigned int i1, i2;
    i1=address&mark.address; //和子网掩码作逻辑与运算
    i2=ip.address&mark.address; //和子网掩码作逻辑与运算
    return (i1==i2); //与子网掩码的逻辑与运算结果相同,则属于同一子网
}

char IP::whatKind()
{
    //IP地址的类型,请到百度百科中搜索《IP地址》
    if(seg3<128)
        return 'A';
    else if(seg3<192)
        return 'B';
    else if(seg3<224)
        return 'C';
    else if(seg3<240)
        return 'D';
    else
        return 'E';
}

int main()
{
    IP ip1(202,194,116,97), ip2(202,194,119,102), mark(255,255,248,0);
    cout<<"ip1: ";
    ip1.showIP();
    cout<<"ip2: ";
    ip2.showIP();
    if(ip1.sameSubnet(ip2,mark))
        cout<<"两个IP在同一子网"<<endl;
    else
        cout<<"两个IP不在同一子网"<<endl;
    cout<<"ip1属于"<<ip1.whatKind()<<"类网络"<<endl;
    return 0;
}

参考解答2:

#include <iostream>
using namespace std;
class IP
{
private:
    union
    {
        unsigned char seg[4]; //IP地址共4字节,或者看成无符号字符数组所指4部分
        unsigned int address; //或者看成一个整体

    };
public:
    IP(int=0,int=0,int=0,int=0);
    void showIP();
    bool sameSubnet(const IP &ip, const IP &mark);
    char whatKind();
};

IP::IP(int s0,int s1,int s2,int s3)
{
    //下面赋值的顺序,涉及整型数据存储结构,参考[《整型数据在内存中如何存储?》](http://blog.csdn.net/sxhelijian/article/details/51015706)
    seg[0]=s3;
    seg[1]=s2;
    seg[2]=s1;
    seg[3]=s0;
}

void IP::showIP()
{
    //显示结果,是给人看的,分4段合适
    cout<<int(seg[3])<<"."<<int(seg[2])<<"."<<int(seg[1])<<"."<<int(seg[0])<<endl;
    return;
}

bool IP::sameSubnet(const IP &ip, const IP &mark)
{
    //与子网掩码按位与,是计算机内部的操作,直接以一个整体操作更方便
    unsigned int i1, i2;
    i1=address&mark.address; //和子网掩码作逻辑与运算
    i2=ip.address&mark.address; //和子网掩码作逻辑与运算
    return (i1==i2); //与子网掩码的逻辑与运算结果相同,则属于同一子网
}

char IP::whatKind()
{
    //IP地址的类型,请到百度百科中搜索《IP地址》
    if(seg[3]<128)
        return 'A';
    else if(seg[3]<192)
        return 'B';
    else if(seg[3]<224)
        return 'C';
    else if(seg[3]<240)
        return 'D';
    else
        return 'E';
}

int main()
{
    IP ip1(202,194,116,97), ip2(202,194,119,102), mark(255,255,248,0);
    cout<<"ip1: ";
    ip1.showIP();
    cout<<"ip2: ";
    ip2.showIP();
    if(ip1.sameSubnet(ip2,mark))
        cout<<"两个IP在同一子网"<<endl;
    else
        cout<<"两个IP不在同一子网"<<endl;
    cout<<"ip1属于"<<ip1.whatKind()<<"类网络"<<endl;
    return 0;
}
目录
相关文章
|
4天前
|
设计模式 安全 编译器
【C++11】特殊类设计
【C++11】特殊类设计
24 10
|
9天前
|
C++
C++友元函数和友元类的使用
C++中的友元(friend)是一种机制,允许类或函数访问其他类的私有成员,以实现数据共享或特殊功能。友元分为两类:类友元和函数友元。类友元允许一个类访问另一个类的私有数据,而函数友元是非成员函数,可以直接访问类的私有成员。虽然提供了便利,但友元破坏了封装性,应谨慎使用。
40 9
|
4天前
|
存储 编译器 C语言
【C++基础 】类和对象(上)
【C++基础 】类和对象(上)
|
12天前
|
编译器 C++
【C++】string类的使用④(字符串操作String operations )
这篇博客探讨了C++ STL中`std::string`的几个关键操作,如`c_str()`和`data()`,它们分别返回指向字符串的const char*指针,前者保证以&#39;\0&#39;结尾,后者不保证。`get_allocator()`返回内存分配器,通常不直接使用。`copy()`函数用于将字符串部分复制到字符数组,不添加&#39;\0&#39;。`find()`和`rfind()`用于向前和向后搜索子串或字符。`npos`是string类中的一个常量,表示找不到匹配项时的返回值。博客通过实例展示了这些函数的用法。
|
12天前
|
C++
【C++】string类的使用④(常量成员Member constants)
C++ `std::string` 的 `find_first_of`, `find_last_of`, `find_first_not_of`, `find_last_not_of` 函数分别用于从不同方向查找目标字符或子串。它们都返回匹配位置,未找到则返回 `npos`。`substr` 用于提取子字符串,`compare` 则提供更灵活的字符串比较。`npos` 是一个表示最大值的常量,用于标记未找到匹配的情况。示例代码展示了这些函数的实际应用,如替换元音、分割路径、查找非字母字符等。
|
12天前
|
存储 C++
【C++】string类的使用③(非成员函数重载Non-member function overloads)
这篇文章探讨了C++中`std::string`的`replace`和`swap`函数以及非成员函数重载。`replace`提供了多种方式替换字符串中的部分内容,包括使用字符串、子串、字符、字符数组和填充字符。`swap`函数用于交换两个`string`对象的内容,成员函数版本效率更高。非成员函数重载包括`operator+`实现字符串连接,关系运算符(如`==`, `&lt;`等)用于比较字符串,以及`swap`非成员函数。此外,还介绍了`getline`函数,用于按指定分隔符从输入流中读取字符串。文章强调了非成员函数在特定情况下的作用,并给出了多个示例代码。
|
17天前
|
C++
【C++】日期类Date(详解)②
- `-=`通过复用`+=`实现,`Date operator-(int day)`则通过创建副本并调用`-=`。 - 前置`++`和后置`++`同样使用重载,类似地,前置`--`和后置`--`也复用了`+=`和`-=1`。 - 比较运算符重载如`&gt;`, `==`, `&lt;`, `&lt;=`, `!=`,通常只需实现两个,其他可通过复合逻辑得出。 - `Date`减`Date`返回天数,通过迭代较小日期直到与较大日期相等,记录步数和符号。 ``` 这是236个字符的摘要,符合240字符以内的要求,涵盖了日期类中运算符重载的主要实现。
|
12天前
|
C++
C++】string类的使用③(修改器Modifiers)
这篇博客探讨了C++ STL中`string`类的修改器和非成员函数重载。文章介绍了`operator+=`用于在字符串末尾追加内容,并展示了不同重载形式。`append`函数提供了更多追加选项,包括子串、字符数组、单个字符等。`push_back`和`pop_back`分别用于在末尾添加和移除一个字符。`assign`用于替换字符串内容,而`insert`允许在任意位置插入字符串或字符。最后,`erase`函数用于删除字符串中的部分内容。每个函数都配以代码示例和说明。
|
12天前
|
安全 编译器 C++
【C++】string类的使用②(元素获取Element access)
```markdown 探索C++ `string`方法:`clear()`保持容量不变使字符串变空;`empty()`检查长度是否为0;C++11的`shrink_to_fit()`尝试减少容量。`operator[]`和`at()`安全访问元素,越界时`at()`抛异常。`back()`和`front()`分别访问首尾元素。了解这些,轻松操作字符串!💡 ```
|
12天前
|
存储 编译器 Linux
【C++】string类的使用②(容量接口Capacity )
这篇博客探讨了C++ STL中string的容量接口和元素访问方法。`size()`和`length()`函数等价,返回字符串的长度;`capacity()`提供已分配的字节数,可能大于长度;`max_size()`给出理论最大长度;`reserve()`预分配空间,不改变内容;`resize()`改变字符串长度,可指定填充字符。这些接口用于优化内存管理和适应字符串操作需求。