C语言 结构体笔记
关于C语言中的结构体的语法,这是C++自定义类型的基础。 C++的类提供了非常丰富的功能,但是也会尽量兼容C语言的结构体。 简单示例 下面是针对学生信息的结构体的使用示例,包括了结构体对象的定义和读写,以及通过指针传递给函数。 123456789101112131415161718192021222324252627282930313233343536373839#include <stdio.h>#include <string.h>struct Person { char name[50]; int age; double height;};void show(struct Person *person) { printf("Name: %s\n", person->name); printf("Age: %d\n", person->age); printf("Height: %.2f\n",...
Cpp new/delete 语句
new和delete是C++中最基础且重要的申请和释放堆内存的语法,以它为线索可以引出C++堆内存管理的相关内容,它也是智能指针所需要的基础知识,值得整理一下。 基本使用 内置类型 new和delete最基础的用法是被设计用来替代C语言中的malloc和free的,分配堆内存来构造指定类型的对象,返回对应类型的指针,例如 12int *p = new int; // uninitializeddelete p; 这两个语句都是非常危险的: 最简单的new语句对内置数据类型不会对内存进行初始化,我们得到值是随机的;(Debug模式下可能内存被清空,改成Release模式可以看到随机值) delete语句不会在释放内存之后将指针置空,指针会变成空悬指针。 我们可以用下面的new语句进行初始化 12345double *p1 = new double; // uninitializeddouble *p2 = new double(); // 0double *p3 = new double{}; // 0double *p4 =...
Cpp 显式加载动态库
在 C++ 中,加载动态库通常有两种方式:显式加载和隐式加载。 隐式加载比较简单,只需要设置链接选项即可自动进行,本文主要学习一下显式加载。 概述 在 C++...
Cpp 奇异递归模板模式 CRTP
CRTP 是一个非常经典的C++静态多态实现方案,在很多库中都有应用,这里整理一下。 概述 奇异递归模板模式(Curiously Recurring Template Pattern, CRTP)是...
Cpp static+extern+inline 学习笔记
整理一下关于C++中的几个关键词(static,extern和inline)的笔记。 static static这个关键词的语义非常复杂,在C语言中本来就有多重语义,C++又添加了额外的语义。 静态函数 默认情况下,函数可以被其它编译单元使用,具有外部链接性,可以使用static将其改为具有内部链接性的静态函数,只有在当前文件中才可以使用此函数。 例如 123static void func();void func(){} 静态全局变量 与静态函数类似,全局变量在默认情况下可以被其它编译单元所使用,具有外部链接性, 可以使用static将其改为静态全局变量,只有在当前文件中才可以使用。 例如 1static int s = 100; 静态局部变量 在函数体内部使用static修饰的局部变量会变成静态局部变量,此时静态的语义不再是对外部不可见,而是延长生命周期,此时变量的存储位置从栈区转移到了数据区,不会因为函数调用的结束而销毁,下次进入函数体时会自动忽略定义和初始化语句,例如 12345int func(){ static int s =...
Cpp 面向对象——虚函数进阶
前面简单介绍了重定义和重写,以及重写所涉及到的虚函数的基本概念,这里整理一下虚函数和动态多态的进阶内容。 C++使用虚函数机制来实现运行期的多态(动态多态),与之相对的是编译期的多态(静态多态), 对于C++来说,静态多态可以通过函数重载和泛型编程实现,而动态多态则通过虚函数实现。 在讨论的范围内,基类或派生类指针/引用的使用方法通常是完全等效的,示例主要以指针的使用为主。 为了简化讨论,我们不关注public,private,protected修饰的区别,无论是对类的方法还是继承关系的修饰,统一使用public。 我们不关注多继承、菱形继承以及虚继承等复杂情景,只考虑简单的单继承情景,不考虑模板类的处理。 虚析构函数 对于含有虚函数的类,非常建议将基类的析构函数也设置为虚函数,此时派生类的析构函数也全部会自动变为虚函数,这可以保证基类指针指向派生类时, 在销毁时可以正确调用派生类的析构函数。 考虑下面的例子 123456789101112131415161718192021222324252627282930#include <iostream>struct...
Cpp 面向对象——重定义与重写、虚函数
在一个作用域中,同名函数但是形参列表不同的函数构成重载的关系, 在考虑面向对象时,则会产生一系列新的问题,因为基类和派生类的作用域是一种非常微妙的关系,既不能说它们是简单的两个独立的作用域,也不能说是一个作用域,而且这种关系是天然不对称的。 在面向对象的语法中对基类和派生类的同名函数(相同或者不同的形参列表)都有着特殊的设计,在本文中我们主要讨论的是不涉及虚函数的部分。 为了简化讨论,我们不关注public,private,protected修饰的区别,无论是对类的方法还是继承关系的修饰,统一使用public。 我们不关注多继承、菱形继承以及虚继承等复杂情景,只考虑简单的单继承情景,不考虑模板类的处理。 重定义(隐藏) 例子 假设基类Base定义了一个方法hello,有好几个版本,相互之间构成重载关系。 派生类Derived如果没有实现hello方法,那么派生类对象仍然是可以直接调用hello方法的,包括基类所实现的各种版本,通过重载决议调用,这是继承所赋予的特点。 例如 12345678910111213141516#include...
C/Cpp 类型别名
typedef typedef是用于为类型定义别名的关键字,在C和C++中都可以使用。 最简单的用法是对基本类型起别名,例如 1typedef int Length; 下面的是MSVC的stdint.h中的部分源码,对基本数据类型起了意义更明确的别名 12345678typedef signed char int8_t;typedef short int16_t;typedef int int32_t;typedef long long int64_t;typedef unsigned char uint8_t;typedef unsigned short uint16_t;typedef unsigned int uint32_t;typedef unsigned long long uint64_t; 数组类型和指针类型可以通过typedef起别名达到简化语法和提升可读性的目的,例如 12345typedef int IntArray4[4];typedef...