从C语言到C++⑧(第二章_类和对象_下篇_续)笔试选择题和OJ题

简介: 从C语言到C++⑧(第二章_类和对象_下篇_续)笔试选择题和OJ题

一. 笔试选择题

1. 有一个类A,其数据成员如下: 则构造函数中,成员变量一定要通过初始化列表来初始化的是:( )

class A {
 
...
private:
   int a;
public:
   const int b;
   float* &c;
   static const char* d;
   static double* e;
};

A.a b c

B.b c

C.b c d e

D.b c d

E.b

F.c

2. 下面程序的运行结果是( )?

class A
{
public:
  A(int a)
    :_a1(a)
    ,_a2(_a1)
  {}
  void Print()
  {
    cout<<_a1<<" "<<_a2<<endl;
  }
private:
  int _a2;
  int _a1;
}
 
int main()
{
  A aa(1);
  aa.Print();
}

A.输出1 1

B.程序崩溃

C.编译不通过

D.输出1 随机值

3. 在一个cpp文件里面,定义了一个static类型的全局变量,下面一个正确的描述是:( )

A.只能在该cpp所在的编译模块中使用该变量

B.该变量的值是不可改变的

C.该变量不能在类的成员函数中引用

D.这种变量只能是基本类型(如int,char)不能是C++类型

4. 关于C++类中static 成员和对象成员的说法正确的是( )

A.static 成员变量在对象构造时生成

B.static 成员函数在对象成员函数中无法调用

C.static 成员函数没有this指针

D.static 成员函数不能访问static 成员变量

5. 下面程序段包含4个函数,其中具有隐含this指针的是( )

int f1();
 
class T
{
  public:static int f2();
 
  private:friend int f3();
 
  protect:int f4();
};

A.f1

B.f2

C.f3

D.f4

6. 一个类的友元函数能够访问类的( )

A.私有成员

B.保护成员

C.公有成员

D.所有成员

7. 下面有关友元函数与成员函数的区别,描述错误的是?( )

A.友元函数不是类的成员函数,和普通全局函数的调用没有区别

B.友元函数和类的成员函数都可以访问类的私有成员变量或者是成员函数

C.类的成员函数是属于类的,调用的时候是通过指针this调用的

D.友元函数是有关键字friend修饰,调用的时候也是通过指针this调用的

答案解析:

1. B

A.a是不同数据成员,可以通过构造函数进行赋值

B.正确,常量以及引用只能通过初始化列表初始化


C.d,e是静态成员,只能在类外初始化


D.d是静态成员,只能在类外初始化


E.b常量只能通过初始化列表初始化,但不是最佳答案


F.c引用只能通过初始化列表初始化,但不是最佳答案


2. D


A.初始化顺序由定义类时的声明顺序决定,所以先初始化_a2,由于初始化_a2时_a1还未初始化,所以为随机值,故错误


B.程序正常运行


C.能编译通过


D.正确


3. A


A.正确,static限制了变量具有文件域


B.static变量是可以被改变的


C.可以被正常访问使用,以及通过成员来进行引用


D.静态变量也可以是自定义类型的变量


4. C


A.static成员变量在对象生成之前生成


B.普通成员函数是可以调用static函数的


C.static函数属于所有对象共享,不具备this指针


D.static函数唯一能够访问的就是static变量或者其他static函数


5. D


A.全局函数不具备this指针


B.static函数不具备this指针


C.友元函数不具备this指针


D.正确,普通成员方法具有隐藏的this指针


6. D


A.可以访问,这也把一个函数声明为友元的目的


B.可以访问


C.可以访问


D.友元函数对一个类里面的所有成员,全部通吃,正确


7. D


A.友元函数不是类的成员函数,就相当于你的朋友再亲密也不是你的家人,既然不是类成员函数,那和普通成员函数调用一样,不需要通过对象调用


B.友元的目的就是为了访问类的私有数据,成员函数可以直接访问类的私有数据


C.类的成员函数属于类,调用时其内部数据会通过this指针来调用


D.友元函数不具备this指针,更谈不上通过this调用,故错误

二. OJ题

1. 求1+2+3+...+n_牛客题霸_牛客网 (nowcoder.com)

HJ73 计算日期到天数转换


中等  通过率:43.38%  时间限制:1秒  空间限制:64M


描述


求1+2+3+...+n,要求不能使用乘除法、for、while、if、else、switch、case等关键字及条件判断语句(A?B:C)。


数据范围:0<n≤200

进阶: 空间复杂度 O(1) ,时间复杂度 O(n)


示例1


输入:


5


返回值:


15


示例2


输入:


1


返回值:


1

class Solution {
public:
    int Sum_Solution(int n) {
        
    }
};

解析代码:

* 1. 题目要求不能使用乘除法、for、while、if、else、switch、case等关键字及条件判断语句A?B:C

  说明不能采用常规方法进行累加求和

* 2. 充分利用C++特性,构造对象构造函数会自动调用的特点,让求和在构造函数中进行完成

* 3. 由于所有对象要针对相同的和进行更新,所以其成员定义为static

class Sum
{
public:
    Sum()
    {
        _sum+=_N;
        ++_N;
    }
 
    static int GetSum()
    {
        return _sum;
    }
private:
    static int _N;
    static int _sum;
};
int Sum::_sum=0;
int Sum::_N=1;
 
class Solution {
public:
    int Sum_Solution(int n) {
        Sum arr[n];// 调用n次Sum的构造函数
        return Sum::GetSum();
    }
};

2. 计算日期到天数转换_牛客题霸_牛客网 (nowcoder.com)

HJ73 计算日期到天数转换

描述

根据输入的日期,计算是这一年的第几天。

保证年份为4位数且日期合法。

进阶:时间复杂度:O(n) ,空间复杂度: O(1)

输入描述:

输入一行,每行空格分割,分别是年,月,日

输出描述:

输出是这一年的第几天

示例1

输入:

2012 12 31

输出:

366

示例2

输入:

1982 3 4

输出:

63

#include <iostream>
using namespace std;
 
int main() {
    int a, b;
    while (cin >> a >> b) { // 注意 while 处理多个 case
        cout << a + b << endl;
    }
}
// 64 位输出请用 printf("%lld")

解析代码:

可以用以前写的函数,比较灵活

#include <iostream>
using namespace std;
int GetMonthDay(int year, int month)
{
  static int monthDatArr[13] = { 0, 31, 28, 31, 30, 31, 30,31, 31, 30, 31, 30, 31 };
  int day = monthDatArr[month];
  // 先二月再判断是否是闰年 (效率快一点)
  if (month == 2 && ((year % 4 == 0 && year % 100 != 0) || (year % 400 == 0)))
  {
    day += 1;                               // 是闰年,天数+1
  }
  return day;                                 // 返回计算的天数
}
int main()
{
  int year = 0, month = 0, day = 0, sum = 0;
  cin >> year >> month >> day;
  for (int i = 1;i < month;i++)
  {
    sum += GetMonthDay(year, i);
  }
  cout << sum + day;
  return 0;
}

3. 日期差值_牛客题霸_牛客网 (nowcoder.com)

KY111 日期差值

描述

有两个日期,求两个日期之间的天数,如果两个日期是连续的我们规定他们之间的天数为两天

输入描述:

有多组数据,每组数据有两行,分别表示两个日期,形式为YYYYMMDD

输出描述:

每组数据输出一行,即日期差值

示例1

输入:

20110412

20110422

输出:

11

#include <iostream>
using namespace std;
 
int main() {
    int a, b;
    while (cin >> a >> b) { // 注意 while 处理多个 case
        cout << a + b << endl;
    }
}
// 64 位输出请用 printf("%lld")

解析代码:

思路:

1. 分别求出每一个日期与0000年0月1日距离的天数

2. 两个距离天数相减即可得到两个日期相差的天数

//思路:
// 1. 分别求出每一个日期与0000年0月1日距离的天数
// 2. 两个距离天数相减即可得到两个日期相差的天数
#include <iostream>
using namespace std;
//给出年月日,计算距离0000年0月1日的天数和
int CountDay(int y, int m, int d)
{
    // 计算0-y年的天数
    int yearDay = y * 365 + y / 4 - y / 100 + y / 400;
    //平年从1月到n月的天数
    int mon[12] = { 0,31,59,90,120,151,181,212,243,273,304,334 };
    // 计算到0-m月的天数
    int monthDay = mon[m - 1];
    if (m > 2 && ((y % 4 == 0 && y % 100 != 0) || y % 400 == 0))
    {
        monthDay += 1;
    }
    return yearDay + monthDay + d;
}
 
int main()
{
    int year1 = 1, month1 = 1,day1 = 1;
    scanf("%4d%2d%2d", &year1, &month1, &day1);
    int n1 = CountDay(year1, month1, day1);
 
    int year2 = 1, month2 = 1, day2 = 1;
    scanf("%4d%2d%2d", &year2, &month2, &day2);
    int n2 = CountDay(year2, month2, day2);
 
    cout << abs(n1 - n2) + 1 << endl;
}

4. 打印日期_牛客题霸_牛客网 (nowcoder.com)

KY222 打印日期

描述

给出年分m和一年中的第n天,算出第n天是几月几号。

输入描述:

输入包括两个整数y(1<=y<=3000),n(1<=n<=366)。

输出描述:


可能有多组测试数据,对于每组数据, 按 yyyy-mm-dd的格式将输入中对应的日期打印出来。


示例1


输入:


2000 3

2000 31

2000 40

2000 60

2000 61

2001 60


输出:


2000-01-03

2000-01-31

2000-02-09

2000-02-29

2000-03-01

2001-03-01

#include <iostream>
using namespace std;
 
int main() {
    int a, b;
    while (cin >> a >> b) { // 注意 while 处理多个 case
        cout << a + b << endl;
    }
}
// 64 位输出请用 printf("%lld")

解析代码:

利用日期类写的日期加天数的思路就很容易写出来了,这里认为天数全都加到day上了。

#include <iostream>
using namespace std;
 
int GetMonthDay(int year, int month)
{
    static int monthDatArr[13] = { 0, 31, 28, 31, 30, 31, 30,31, 31, 30, 31, 30, 31 };
    int day = monthDatArr[month];
    // 先二月再判断是否是闰年 (效率快一点)
    if (month == 2 && ((year % 4 == 0 && year % 100 != 0) || (year % 400 == 0)))
    {
        day += 1; // 是闰年,天数+1
    }
    return day; // 返回计算的天数
}
 
int main()
{
    int year = 1, day = 1;
    while (cin >> year >> day)
    {
        int month = 1;
        while (day > GetMonthDay(year, month))
        {
            day -= GetMonthDay(year, month);
            month++;
        }
        printf("%4d-%02d-%02d\n", year, month, day);
    }
    return 0;
}

5. 日期累加_牛客题霸_牛客网 (nowcoder.com)

KY258 日期累加

中等  通过率:20.65%  时间限制:1秒  空间限制:32M

描述

设计一个程序能计算一个日期加上若干天后是什么日期。

输入描述:


输入第一行表示样例个数m,接下来m行每行四个整数分别表示年月日和累加的天数。


输出描述:


输出m行,每行按yyyy-mm-dd的个数输出。


示例1


输入:


1

2008 2 3 100


输出:


2008-05-13

解析代码:

这题完全就是日期类中日期加天数的运算符重载了,所以写好日期类还是很重要的。

#include <iostream>
using namespace std;
 
int GetMonthDay(int year, int month)
{
    static int monthDatArr[13] = { 0, 31, 28, 31, 30, 31, 30,31, 31, 30, 31, 30, 31 };
    int day = monthDatArr[month];
    // 先二月再判断是否是闰年 (效率快一点)
    if (month == 2 && ((year % 4 == 0 && year % 100 != 0) || (year % 400 == 0)))
    {
        day += 1; // 是闰年,天数+1
    }
    return day; // 返回计算的天数
}
 
int main()
{
    int year = 0, month = 0, day = 0, nday = 0, n = 0;
    cin >> n;
    for (int i = 0;i < n;i++)
    {
        cin >> year >> month >> day >> nday;
        day += nday;
        while (day > GetMonthDay(year, month))
        {
            day -= GetMonthDay(year, month );
            month++;
            if (month == 13)
            {
                year++;
                month = 1;
            }
        }
        printf("%4d-%02d-%02d\n", year, month, day);
    }
    return 0;
}

本章完。

目录
相关文章
|
4月前
|
安全 C语言 C++
比较C++的内存分配与管理方式new/delete与C语言中的malloc/realloc/calloc/free。
在实用性方面,C++的内存管理方式提供了面向对象的特性,它是处理构造和析构、需要类型安全和异常处理的首选方案。而C语言的内存管理函数适用于简单的内存分配,例如分配原始内存块或复杂性较低的数据结构,没有构造和析构的要求。当从C迁移到C++,或在C++中使用C代码时,了解两种内存管理方式的差异非常重要。
179 26
|
5月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
149 0
|
5月前
|
存储 编译器 程序员
c++的类(附含explicit关键字,友元,内部类)
本文介绍了C++中类的核心概念与用法,涵盖封装、继承、多态三大特性。重点讲解了类的定义(`class`与`struct`)、访问限定符(`private`、`public`、`protected`)、类的作用域及成员函数的声明与定义分离。同时深入探讨了类的大小计算、`this`指针、默认成员函数(构造函数、析构函数、拷贝构造、赋值重载)以及运算符重载等内容。 文章还详细分析了`explicit`关键字的作用、静态成员(变量与函数)、友元(友元函数与友元类)的概念及其使用场景,并简要介绍了内部类的特性。
227 0
|
7月前
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
278 12
|
8月前
|
编译器 C++
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。
|
8月前
|
存储 编译器 C++
类和对象(上)(C++)
本篇内容主要讲解了C++中类的相关知识,包括类的定义、实例化及this指针的作用。详细说明了类的定义格式、成员函数默认为inline、访问限定符(public、protected、private)的使用规则,以及class与struct的区别。同时分析了类实例化的概念,对象大小的计算规则和内存对齐原则。最后介绍了this指针的工作机制,解释了成员函数如何通过隐含的this指针区分不同对象的数据。这些知识点帮助我们更好地理解C++中类的封装性和对象的实现原理。
|
8月前
|
编译器 C++
类和对象(下)C++
本内容主要讲解C++中的初始化列表、类型转换、静态成员、友元、内部类、匿名对象及对象拷贝时的编译器优化。初始化列表用于成员变量定义初始化,尤其对引用、const及无默认构造函数的类类型变量至关重要。类型转换中,`explicit`可禁用隐式转换。静态成员属类而非对象,受访问限定符约束。内部类是独立类,可增强封装性。匿名对象生命周期短,常用于临时场景。编译器会优化对象拷贝以提高效率。最后,鼓励大家通过重复练习提升技能!
|
2月前
|
存储 C语言
`scanf`是C语言中用于按格式读取标准输入的函数
`scanf`是C语言中用于按格式读取标准输入的函数,通过格式字符串解析输入并存入指定变量。需注意输入格式严格匹配,并建议检查返回值以确保读取成功,提升程序健壮性。
913 0
|
4月前
|
安全 C语言
C语言中的字符、字符串及内存操作函数详细讲解
通过这些函数的正确使用,可以有效管理字符串和内存操作,它们是C语言编程中不可或缺的工具。
304 15
|
10月前
|
存储 算法 C语言
【C语言程序设计——函数】素数判定(头歌实践教学平台习题)【合集】
本内容介绍了编写一个判断素数的子函数的任务,涵盖循环控制与跳转语句、算术运算符(%)、以及素数的概念。任务要求在主函数中输入整数并输出是否为素数的信息。相关知识包括 `for` 和 `while` 循环、`break` 和 `continue` 语句、取余运算符 `%` 的使用及素数定义、分布规律和应用场景。编程要求根据提示补充代码,测试说明提供了输入输出示例,最后给出通关代码和测试结果。 任务核心:编写判断素数的子函数并在主函数中调用,涉及循环结构和条件判断。
535 23