【C++OJ拷贝构造】身份证设定(复合类+拷贝构造)

该博客主要介绍了如何使用C++实现一个身份证类(PID),包括身份证类型、身份证号码和出生日期等属性,并提供了构造函数、拷贝构造函数和打印方法。拷贝构造函数特别处理了15位身份证升级到18位的情况,规则包括扩展年份、计算校验位等。博客中给出了具体的示例输入和输出,以及参考代码实现。

【C++OJ拷贝构造】身份证设定(复合类+拷贝构造)

题目描述

定义一个身份证类PID,包含私有属性:身份证类型、身份证号码、出生日期;另外包含方法:构造、拷贝构造打印等。

身份证类型表示一代身份证或者二代身份证,分别用1和2表示

身份证号码是一个字符串,长度为15或者18

出生日期是一个类,包含私有属性年、月、日,以及构造函数等(根据需要添加其他方法)

构造函数要注意是复合类,要考虑复合类成员的构造

打印函数把身份证的所有属性都输出,输出格式看示例

拷贝构造作用:如果身份证号码是15位的就升级为18位,包括把身份证类型改为2,然后把号码扩展,规则如下:

\1. 原15位身份证的第7位到12位表示出生年月日,每个两位;把年份的2位扩展为四位。

\2. 把扩展后的17个数字求和,取总和的末尾数字,如果末尾数字是0,则将0改为X,然后把这个数字作为第18位

3.如果身份证号码已经是18位,就无需升级

例如身份证123456910203000,表示91年2月3日出生,然后根据类属性出生日期知道是1991年,不是2091年。因此扩展为12345619910203000

接着把17个数字相加得到46,取末尾6,最终扩展为123456199102030006

输入

第一行输入t表示t个示例

第二行输入一个身份证的5个属性,顺序为:类型、号码、出生年、月、日

依次输入t行

输出

采用拷贝构造函数的方法对身份证号码升级,然后输出

输入样例

3
1 123456910203000 1991 2 3
2 654321200001018889 2000 1 1
1 234567001217000 2000 12 17

输出样例

type=2 birth=1991.02.03
ID=123456199102030006
type=2 birth=2000.01.01
ID=654321200001018889
type=2 birth=2000.12.17
ID=23456720001217000X

参考代码

#include <iostream>
#include <cstring>
#include <iomanip>
using namespace std;

class Date
{
private:
    int year;
    int month;
    int day;

public:
    Date() {}
    Date(int y, int m, int d) : year(y), month(m), day(d) {}
    void set(int y, int m, int d)
    {
        year = y;
        month = m;
        day = d;
    }
    int getyear() { return year; }
    int getmonth() { return month; }
    int getday() { return day; }
    ~Date() {}
};
class PID
{
private:
    int type; //身份证类型表示一代身份证或者二代身份证,分别用1和2表示
    char *num;
    Date date1;

public:
    PID() {}                                 //无参构造
    PID(int t, char *n, int y, int m, int d) //有参构造,进行初始化,注意复合类的构造
    {
        type = t;
        num = new char[20];
        strcpy(num, n);
        date1.set(y, m, d);
    }
    PID(PID &p) //深拷贝构造
    {
        num = new char[20];                                                 //内存分配给num,最多18,这里应该至少分配19
        type = 2;                                                           //设置type,输出都是2
        date1.set(p.date1.getyear(), p.date1.getmonth(), p.date1.getday()); //初始化date1

        if (strlen(p.num) == 18)
            strcpy(num, p.num); //如果本身18位身份证,直接设置为num
        else                    //如果传入的是15位数,进行变化
        {
            int s = 0;

            for (int i = 0; i < 6; i++)
            {
                num[i] = p.num[i]; //前6位数保持不变,
                s += num[i] - '0'; //相加
            }

            for (int i = 8; i < 17; i++)
            {
                num[i] = p.num[i - 2]; // 8-18位保持不变,
                s += num[i] - '0';     //相加
            }

            int t = date1.getyear(); //获取year,设置7、8位数
            num[6] = t / 1000 + '0';
            num[7] = t / 100 % 10 + '0';
            s += num[6] - '0';
            s += num[7] - '0';

            if (s % 10 == 0)
                num[17] = 'X';
            else
                num[17] = s % 10 + '0';
        }
        num[18] = '\0'; //结束符
    }
    void print()
    {
        cout << "type=2" << ' ' << "birth=" << date1.getyear() << '.'
             << setw(2) << setfill('0') << date1.getmonth() << '.'
             << setw(2) << setfill('0') << date1.getday() << endl;
        cout << "ID=" << num << endl;
    }
    ~PID() { delete[] num; }
};

int main()
{
    int t;
    cin >> t;
    int ty, y, m, d;
    char n[20];
    while (t--)
    {
        cin >> ty >> n >> y >> m >> d;
        PID p1(ty, n, y, m, d);
        PID p2(p1);
        p2.print();
    }
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Ferry_xie

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值