C++ 一些知识点

2023-10-26

https://www.nowcoder.com/profile/7838045/myFollowings/detail/3445247

链接:https://www.nowcoder.com/questionTerminal/087006d1e4ff46ca927c812294facbc4
来源:牛客网

答案:D;
第一个printf:由于char***cpp,可以读成cpp为一个指向char**类型的指针,并且初始化为cp,而cp是一个指针数组,数组里面存储的类型为char**,也就是cpp指向cp[0],故*cpp=cp[0];因此++cpp使得cpp指向cp[1],故*++cpp=cp[1];而cp[1]是一个指向c[2]的指针,因此*cp[1] = c[2],故**++cpp=*cp[1]=c[2],故输出WORLD;
第二个printf:由优先级可得知,单目运算符高于算术运算符;而*和++是单目运算符,+是算术运算符,又cpp本身是指向的cp[1],故++cpp使得cpp指向cp[2],故*++cpp则为cp[2],而此时的cp[2]是一个指向c[1]的地址,故对指向c[1]的指针cp[2]进行--,故使得cp[2]指向的是c[0],故*--*++cpp+3=*--cp[2]+3=c[0]+3,而这里的c[0],其实存放的是指向HELLO字符串的指针故c[0]+3,使得该指针指向HELLO中的第二个L的位置,故输入LO;
第三个printf:由于cpp目前指向的是cp[2],而cpp[-2]是等于*(cpp-2),而在第二个printf时cpp指向cp[2],则*(cpp-2) = cp[0](注:这里cpp-2后,cpp本身的指针地址没有改变,因此在第四个printf中,cpp依旧是指向cp[2]),又cp[0]是一个指向c[3]的指针,对cp[0]进行*运算,使得*cp[0]=c[3],故**(cpp-2)=*cpp[-2]=c[3],同时c[3]是一个指向SAYHI的指针,故c[3]+3,使得指向H的位置,故输出HI;
第四个printf:在第三个printf并没有改变cpp的地址,故cpp依旧是指向cp[2],又cpp[-1][-1]+1=*(*(cpp-1)-1)+1,故cpp-1,则cpp指向cp[2],故*(cpp-1)=cp[1];此时的cp[1]是一个指针, 指向c[2] ,故cp[1]-1是从指向c[2]的指针地址-1,故是指向c[1],则*(*(cpp-1)-1)=*(cp[1]-1)=c[1],故 cpp[-1][-1]+1=*(*(cpp-1)-1)+1=c[1]+1,此时的c[1]同样是一个指针,故c[1]+1是指向NEW中的E的位置,故输出EW。
故答案是D。


一)

1 友元函数和友元类均可以访问private \protected成员;

2 友元函数和友元类能够引用类中的private \protected成员,但是在其内部引用时必须通过对象引用,比如函数体中a.x,b.x,a.y,b.y都是类的私有成员,他们是通过对象引用的。

3 友元函数在外部调用时,不需要通过基类的对象 “  . ”  或者对象指针 “-> ” 来调用,因为它是一个普通函数,不是类成员函数,调用跟普通函数方式一样;

4 友元函数在基类外部定义,且定义时不需要加 :: 符号。

5 友元函数和友元类在基类内部声明时在哪个位置都一样,包括public\private\protected位置;

int a={1,2,3,4}; int *p=(int *)((int)a+1); P指向了什么?
1. int a = {}; 貌似应该是int a[] = {};
2. "(int)a",已经把a的地址强转成int型数字(32bit),所以“((int)a+1); ”实际指向了整个数组内存块的第二个地址,即a[0]的第二个字节。所以p的指向就向前移动了一个字节,指向了a【0】的第二个字节的地址。

剑指offer T48

1 char pData[] = "i am a student.";                      char *pData = "i am a student.";

后者字符串如果为指针型表示,则此字符串为const类型,不可以更改此字符串内容,但是前者数组的形式可以。

比如char *func(char *pData)   此函数的指针输入参数只能选择前者char pData[] = "i am a student.";   选择后者作为参数会出错!!


五-1

class Base {

public:

    virtual void f() { cout << "Base::f" << endl; }

    virtual void g() { cout << "Base::g" << endl; }

    virtual void h() { cout << "Base::h" << endl; }
};


class Derive :public Base
{
public:

      void f() { cout << "Derive::f1" << endl; }

     void g1() { cout << "Derive::g1" << endl; }

    virtual void h1() { cout << "Derive::h1" << endl; }
};

虚函数的多态性只发生在指针或者引用调用的情况下!!!!!!!看下面运行的对比例子!!

        Base *b1 =& b;
        b1->f(); ///Derive::f1

        Base b1 =b;
        b1.f(); ///Base::f

五-2

class Base {

public:

    Base()
    {
        f();
    }
    virtual void f()
    {
        cout <<" gouzao base" << endl;
    }
};

class Derived :public Base
{
public:
    Derived()
    {};
    void f()
    {
        cout << "gou zao derived" << endl;
    }

};


int main()

{
    Base *p = new Derived;
    p->f();

    system("pause");
    return 0;

}

输出结果  gouzao base

gou zao derived

在一开始调用构造函数的时候,运行构造函数前,对象还没有生成,更谈不上调用动态类型了!!尽管对象是个derived,但是构造基类部分时,还只是个base,所以会调用基类的虚函数f().

五-3

class Base {

public:

    Base()
    {
        f();
    }
    virtual void f()
    {
        cout << " gou zao base" << endl;
    }
    ~Base()
    {
        cout << "base~" << endl;
        f();
    }
};

class Derived :public Base
{
public:
    Derived()
    {};
    void f()
    {
        cout << "gou zao derived" << endl;
    }
     ~Derived()
    {
         cout << "derived~" << endl;
        f();
    }

};

int main()

{
    Base *p = new Derived;
    p->f();
    delete p;//注意此时应该是如果析构函数不是虚函数,只掉用基类析构函数,我看输出结果,此时f() 也是只掉用基类的f()" gou zao base",不涉及继承类的多态问题!!
    p = NULL;
    system("pause");
    return 0;
}

基类指针指向继承类时,此时当用基类指针delete, 需要将析构函数设置为virtual函数形式,否则只会调用基类的析构函数,不会调用派生类对象自身的析构函数!

(类中的静态函数不可以设置为虚函数,因为虚函数都是通过具体对象的虚函数表来实现的, 而静态函数不属于具体对象,自然没有虚函数表,它在创建具体对象之前就已经存在了!)


int main(){
     unsigned char a= 0xA5 ;
     unsigned char b=~a>> 4 + 1 ;
     printf( "%d\n" ,b);
     return 0 ;
}
所以优先级顺序是:~、+大于>>大于=,先做a的取反操作,然后4+1为5,然后~a>>5,然后赋值给b

下面,如果按照我们的常识,a取反右移5位应该是2才对,但是答案是250,对250分析可知,当~a右移时,前面补的是1,而不是0,为什么呢?

因为,无论是unsigned char,还是unsigned short、short、char等,都是转换成int来处理的,一般情况下,int为32位,所以a取反后是ffffff5a,右移5位,前面补1(最高位为1,在计算机中认为是负数,补1,如果是unsigned int型的变量的话,前面补0),最后转化成unsigned char型,所以取后8位是fa,即250


面试题:构造函数

#include <iostream>
using namespace std;

class Base
{
private:
int i;
public:
Base(int x)
{
i = x;
}
};

class Derived : public Base
{
private:
int i;
public:
Derived(int x, int y)
{
i = x;
}
void print()
{
cout << i + Base::i << endl;
}
};

int main()
{
Derived A(2,3);
A.print();
return 0;
}

首先,是访问权限问题,子类中直接访问Base::i是不允许的,应该将父类的改为protected或者public(最好用protected)

其次,统计父类和子类i的和,但是通过子类构造函数没有对父类变量进行初始化;此处编译会找不到构造函数,因为子类调用构造函数会先找父类构造函数,但是没有2个参数的,所以可以在初始化列表中调用父类构造函数

最后个问题,是单参数的构造函数,可能存在隐式转换的问题,因为单参数构造函数,和拷贝构造函数形式类似,调用时很可能会发生隐式转换,应加上explicit关键字

#include <iostream>
using namespace std;

class Base
{
protected:
int i;
public:
explicit Base(int x)
{
i = x;
}
};

class Derived : public Base
{
private:
int i;
public:
Derived(int x, int y):Base(x)
{
i = y;
}
void print()
{
cout << i + Base::i << endl;
}
};

int main()
{
Derived A(2,3);
A.print();
return 0;
}

2、初始化列表

1)使用初始化列表提高效率

class Student
{
public:
Student(string in_name, int in_age)
{
name = in_name;
age = in_age;
}
private :
string name;
int age;
};

因为在构造函数中,是对name进行赋值,不是初始化,而string对象会先调用它的默认构造函数,再调用string类(貌似是basic_string类)的赋值构造函数;对于上例的age,因为int是内置类型,应该是赋值的时候获得了初值。

要对成员进行初始化,而不是赋值,可以采用初始化列表(member initialization list)

class Student
{
public:
Student(string in_name, int in_age):name(in_name),age(in_age) {}
private :
string name;
int age;
};

在初始化的时候调用的是string的拷贝构造函数,而上例会调用两次构造函数,从性能上会有不小提升

有的情况下,是必须使用初始化列表进行初始化的:const对象、引用对象

2)初始化列表初始顺序

#include <iostream>
using namespace std;

class Base
{
public:
Base(int i) : m_j(i), m_i(m_j) {}
Base() : m_j(0), m_i(m_j) {}
int get_i() const
{
return m_i;
}
int get_j() const
{
return m_j;
}

private:
int m_i;
int m_j;

};

int main()
{
Base obj(98);
cout << obj.get_i() << endl << obj.get_j() << endl;
return 0;
}

输出为一个随机数和98,为什么呢?因为对于初始化列表而言,对成员变量的初始化,是严格按照声明次序,而不是在初始化列表中的顺序进行初始化,如果改为赋值初始化则不会出现这个问题,当然,为了使用初始化列表,还是严格注意声明顺序吧,比如先声明数组大小,再声明数组这样。

 

C++构造函数初始化按下列顺序被调用:

 

  • 首先,任何虚拟基类的构造函数按照它们被继承的顺序构造;
  • 其次,任何非虚拟基类的构造函数按照它们被继承的顺序构造;
  • 最后,任何成员对象的构造函数按照它们声明的顺序调用;

#include <iostream>
using namespace std;
class OBJ1{
public:
OBJ1(){ cout<<"OBJ1\n"; }
};
class OBJ2{
public:
OBJ2(){ cout<<"OBJ2\n";}
}
class Base1{
public:
Base1(){ cout<<"Base1\n";}
}
class Base2{
public:
Base2(){ cout <<"Base2\n"; }
};
class Base3{
public:
Base3(){ cout <<"Base3\n"; }
};
class Base4{
public:
Base4(){ cout <<"Base4\n"; }
};
class Derived :public Base1, virtual public Base2,public Base3, virtual public Base4//继承顺序{
public:
Derived() :Base4(), Base3(), Base2(),Base1(), obj2(), obj1(){//初始化列表
cout <<"Derived ok.\n";
}
protected:
OBJ1 obj1;//声明顺序
OBJ2 obj2;
};

int main()
{
Derived aa;//初始化
cout <<"This is ok.\n";
return 0;
}
结果:
Base2 //虚拟基类按照被继承顺序初始化
Base4 //虚拟基类按照被继承的顺序
Base1 //非虚拟基类按照被继承的顺序初始化
Base3 //非虚拟基类按照被继承的顺序
OBJ1 //成员函数按照声明的顺序初始化
OBJ2 //成员函数按照声明的顺序
Derived ok.
This is ok.



C++中如果子类一个变量与基类的某个变量重名了怎么办?如果B中有同名成员,则会使得父类的同名成员不可见(需要通过名空间访问),此时函数中的a实际上指B中的a,访问父类中的a需要通过A::来访问。


cout<<2^3<<endl;//error 运算符优先级不对应该为

cout<<(2^3)<<endl;


链接:https://www.nowcoder.com/questionTerminal/2b66de6a28234cd4ab8050c096476b92?pos=136&mutiTagIds=138&orderByHotValue=1
来源:牛客网

int a[3][4],下面哪个不能表示 a[1][1]?
  • *(&a[0][0]+5)
  • *(*(a+1)+1)
  • *(&a[1]+1)
  • *(a[1]+1)
正确答案:C

在二维数组中a[1]表示的是a[1][0]的地址,数组在内存中连续存储,所以a[1]+1表示的是a[1][1]的地址,所以D可以取得正确的值;
指针操作*(a+1)与a[1]等价,所以B也可以取得正确的值;
二维数组在内存中是行优先存储的,所以A中a[0][0]的地址加5可以取得正确值;
C选项错误,应改为*(&a[1][0]+1),因为a[1]就表示a[1][0]的地址。

在VS中编译,输出a[1]和&a[1]的地址是一样的,但是a[1]+1和&a[1]+1却不一样,这是为什么呢?

只是输出来的值一样,但变量类型不同。a[1]是int *类型,而&a[1]是int (*)[4]类型。因此同样加1的时候,偏移不一样。前者只移一位,而后者要移动四维,即到二位数组的下一行。*(*(&a[1]+1))表示a[2][0]的值,也就是*(&a[1]+1)是a[2][0]的地址。
对于二维数组a[m][n]而言,a和a[0]都是首地址等价于&a[0][0],不同之处在于a+1移动至下一行,而a[0]+1移动至下一列。即a+1指针指到a[1][0]处,而a[0]+1指针指到a[0][1];

     int a[2][4] = { { 2, 5, 6, 8 }, { 22, 55, 66, 88 } };

     int c[4] = { 5, 8, 9, 4 };

     int d[3] = { 23, 12, 443 };

     int *p[4], (*q)[4];
//(我不明白为什么q=a可以,然而 p=a不可以!!答谢可否告知!!只是因为p作为第一个int数组不确定大小,但是a作为第一个数组首地址已经确定了数组大小是4个4,维的吗??)
     q = a;

     *p = c;
     p[0] = c;
     *(p + 1) = d;
     p[1] = d;

     int i, j;

     for (i = 0; i<2; i++)

     for (j = 0; j<4; j++)

     {

         if ((i == 1) && (j == 3)) break;

         printf("*(*(p+%d)+%d)=%d\n", i, j, *(*(p + i) + j));

     }

     puts("===============");

     for (i = 0; i<2; i++)

     for (j = 0; j<4; j++)

         printf("*(*(q+%d)+%d)=%d\n", i, j, *(*(q + i) + j));

     cout << p[1][1] << endl;
     cout << *(*(p + 1) + 1) << endl;

int *p[4]与int (*q)[4]的区别

   

 以上定义涉及两个运算符:“*”(间接引用)、“[]”(下标),“[]”的优先级别大于“*”的优先级别。

  首先看int*p[4],“[]”的优先级别高,所以它首先是个大小为4的数组,即p[4];剩下的“int*”作为补充说明,即说明该数组的每一个元素为指向一个整型类型的指针。int*p[4]的存储结构如下:(存储方格横向排列或竖向排列没区别,只要按内存地址顺序排列就行,此处只是为画图方便)

int <wbr>*p[4]与int <wbr>(*q)[4]的区别



   再看int (*q)[4]。它首先是个指针,即*q,剩下的“int [4]”作为补充说明,即说明指针q指向一个长度为4的数组。int(*q)[4]的存储结构如下:

int <wbr>*p[4]与int <wbr>(*q)[4]的区别

请看以下定义:

int a[2][4]={{2,5,6,8},{22,55,66,88}};

int c[4]={5,8,9,4};

int d[3]={23,12,443};

int *p[4],(*q)[4];

q=a;

*p=c;

*(p+1)=d;

则int *p[4]和int (*q)[4]的存储数据为:

int <wbr>*p[4]与int <wbr>(*q)[4]的区别

int <wbr>*p[4]与int <wbr>(*q)[4]的区别


本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

C++ 一些知识点 的相关文章

随机推荐

  • 统计学习方法 例7.1 超详细求解过程

    例7 1 已知一个如图所示的训练数据集 其正例点是 x 1 3 3
  • C#软件开发实例.私人订制自己的屏幕截图工具(九)使用自定义光标,QQ截图时的光标

    本实例全部文章目录 一 功能概览 二 创建项目 注册热键 显示截图主窗口 三 托盘图标及菜单的实现 四 基本截图功能实现 五 针对拖拽时闪烁卡顿现象的优化 六 添加配置管理功能 七 添加放大镜的功能 八 添加键盘操作截图的功能 九 使用自定
  • HashMap实现原理, 扩容机制,面试题和总结

    文章目录 1 讲下对HashMap的认识 2 HashMap的一些参数 3 为什么HashMap的长度必须是2的n次幂 4 HashMap 为什么在获取 hash 值时要进行位运算 5 HashMap在JDK1 7和JDK1 8中有哪些不同
  • PHP实现读取指定目录下的所有文件

    在php中读取指定目录下的文件主要用到了opendir和readdir函数 一 opendir 打开目录句柄 1 语法 opendir path context 2 参数说明 参数 描述 path 必需 规定要打开的目录路径 context
  • 【傻瓜向装系统】电脑重装&&加固态硬盘

    市面上大部分PE制作工具都会在操作系统中内嵌广告或软件等 慎重使用 重装 顺序步骤 重要数据备份 备份到移动硬盘或其他设备 格式化U盘 至少8G 准备作为系统启动盘 安装PE制作工具 如老毛桃等 制作启动盘 下载对应操作系统放到U盘中 某个
  • 极大极小树

    博弈树作为传统AI领域的一个传统又经典的算法 有着广泛的应用 尤其是棋类AI 记得曾经刚学C语言的时候 用控制台写了一个五子棋的程序 后来突发奇想 给它增加可以人机对战的AI 设计了一个简单的根据当前局面判断最优落子的AI 但是只能想到两手
  • DNS、HTTP 与 HTTPS

    DNS HTTP与HTTPS DNS 域名的层级 查找过程 优化 常见的 DNS 攻击 使用的协议 CDN 路由解析 内容分发 HTTP 协议 HTTP 请求报文 GET 和 POST 有什么区别 幂等性和非幂等性 前后端传参 HTTP响应
  • Ubuntu 20.04 开启SSH服务

    更新软件下载源 sudo apt update 安装ssh服务 sudo apt install openssh server 开启防火墙ssh的服务端口 sudo ufw allow ssh 附 还可以查看或更改ssh服务的状态 查看ss
  • Javaweb实现增删改查操作操作

    Javaweb实现增删改查操作操作 一 准备工作 1 Idea编辑器 eclispe和myeclispe都可以 个人推荐使用idea 新建一个web项目 2 数据库mysql 3 需要提前了解的知识点 servlet el和jstl表达式
  • 怎么检测两张照片的相似度,两张图片相似度测试

    计算图像相似度的算法有哪些 SIM StructuralSIMilarity 结构相似性 这是一种用来评测图像质量的一种方法 由于人类视觉很容易从图像中抽取出结构信息 因此计算两幅图像结构信息的相似性就可以用来作为一种检测图像质量的好坏 首
  • 服务器盘符名称修改,linux下powerpath对盘与更改盘符名的教程

    PowerPath 软件在服务器上运行并管理服务器和存储系统中的虚拟磁盘之间的路径 如果一条路径出现故障 它可以将I O 转发到有效路径中 并提供负载平衡来平均分配各条路径中的I O 负载 另外这里的路径由HBA 硬件和驱动程序 光纤 两个
  • docker web mysql_在 Docker 中完整部署 Web 应用

    原标题 在 Docker 中完整部署 Web 应用 一个完整的 Web 应用包含前端页面 数据库 后台逻辑等 按照一般流程去构建需要配置 Nginx MySQL 以及后台服务器 运维涉及到的部分十分复杂 而 Docker 可以将这些东西 数
  • STM8 学习笔记13:PWM

    PWM Gitee 空间跳转 https gitee com galoc stm8 git 1 概述 PWM也叫脉冲宽度调制 是利用微处理器的数字输出来对模拟电路进行控制的一种非常有效的技术 频率 周期 占空比 1 1 PWM 频率 是指在
  • Java常用类System类、Math、BigInteger 和 BigDecimal

    1 System类 public void test1 String javaVersion System getProperty java version System out println java的version javaVersi
  • Windows磁盘管理中的压缩卷操作

    基本磁盘 主分区在压缩卷操作后为黑色的未分配 逻辑分区在压缩卷操作后为绿色的可用空间 动态磁盘 简单卷在压缩卷操作后为黑色的未分配
  • Figma怎么汉化?这个Figma 汉化插件早知道就好了!

    Figma官方目前没有中文版 可能与早期的封禁大疆事件和面向非洲地区扩张策略有关 但是可以使用第三方汉化插件来获得中文版本的Figma Figma cool是一个提供汉化插件的网站 支持多个平台 另外 还有一款名为即时设计的中文设计工具 提
  • java.lang.IllegalArgumentException 异常报错完美解决

    目录 修改JDK使用版本 修改开发工具idea配置 eclipse的直接跳过这个看下面 修改开发工具eclipse配置 学习spring依赖注入的时候碰到这个坑 折腾了许久 记录一下以防其他小伙伴入坑 该异常主要原因是因为JDK与Sprin
  • Lisp-Stat 翻译 —— 第四章 其它Lisp特性

    2019独角兽企业重金招聘Python工程师标准 gt gt gt 第四章 其它Lisp特性 上一章介绍了Lisp编程的基础 在那章里重点展示了对编写Lisp函数有用的编程技术 为了最高效地使用这些技术 知道Lisp和Lisp Stat提供
  • 智能指针之shared_ptr初始化,引用计数,常用操作和自定义删除器等等03

    一 share ptr 1 share ptr基础 1 共享所有权 不是被一个shared ptr拥有 而是被多个shared ptr之间相互协作 shared有额外开销 2 工作原理 利用引用计数的方法管理一片内存 每增加一个shared
  • C++ 一些知识点

    https www nowcoder com profile 7838045 myFollowings detail 3445247 链接 https www nowcoder com questionTerminal 087006d1e4