C++ 学习基础篇(一)—— C++与C 的区别

2023-05-16

      编程的学习学无止境,只掌握一门语言是远远不够的,现在我们开始C++的学习之路,下面先看下C++ 与C 的区别

一、C++概述

1、发展历史

      1980年,Bjarne Stroustrup博士开始着手创建一种模拟语言,能够具有面向对象的程序设计特色。在当时,面向对象编程还是一个比较新的理念,Stroustrup博士并不是从头开始设计新语言,而是在C语言的基础上进行创建。这就是C++语言。

     1985年,C++开始在外面慢慢流行。经过多年的发展,C++已经有了多个版本。为次,ANSI和ISO的联合委员会于1989年着手为C++制定标准。1994年2月,该委员会出版了第一份非正式草案,1998年正式推出了C++的国际标准。

2、C和C++

      C++是C的超集,也可以说C是C++的子集,因为C先出现。按常理说,C++编译器能够编译任何C程序,但是C和C++还是有一些小差别。

      例如C++增加了C不具有的关键字。这些关键字能作为函数和变量的标识符在C程序中使用,尽管C++包含了所有的C,但显然没有任何C++编译器能编译这样的C程序。

     C程序员可以省略函数原型,而C++不可以,一个不带参数的C函数原型必须把void写出来。而C++可以使用空参数列表

     C++中new和delete是对内存分配的运算符,取代了C中的mallocfree

     标准C++中的字符串类取代了C标准C函数库头文件中的字符数组处理函数(C中没有字符串类型)。

     C++中用来做控制态输入输出的iostream类库替代了标准C中的stdio函数库

     C++中的try/catch/throw异常处理机制取代了标准C中的setjmp()和longjmp()函数。


二、关键字和变量

      C++相对与C增加了一些关键字,如下:

typename bool dynamic_cast mutable namespace

static_cast using catch explicit new

virtual operator false private template

volatile const protected this wchar_t

const_cast public throw friend true

reinterpret_cast try

bitor xor_e and_eq compl or_eq

not_eq bitand


在C++中还增加了bool型变量wchar_t型变量

布尔型变量是有两种逻辑状态的变量,它包含两个值:真和假。如果在表达式中使用了布尔型变量,那么将根据变量值的真假而赋予整型值1或0。要把一个整型变量转换成布尔型变量,如果整型值为0,则其布尔型值为假;反之如果整型值为非0,则其布尔型值为真。布儿型变量在运行时通常用做标志,比如进行逻辑测试以改变程序流程。

#include iostream.h
       
int main()
{           
	bool flag;
	flag = true;
	if(flag)
		cout << true << endl;

	return 0;
}

C++中还包括wchar_t数据类型,wchar_t也是字符类型,但是是那些宽度超过8位的数据类型。许多外文字符集所含的数目超过256个,char字符类型无法完全囊括。wchar_t数据类型一般为16位。

标准C++的iostream类库中包括了可以支持宽字符的类和对象。用wout替代cout即可。

#include iostream.h

int main()
{
	wchar_t wc;
	wc = 'b';
	wout << wc;
	wc = 'y';
	wout << wc;
	wc = 'e';
	wout << wc << endl;

	return 0;
}

说明一下:某些编译器无法编译该程序(不支持该数据类型)。


三、强制类型转换

     有时候,根据表达式的需要,某个数据需要被当成另外的数据类型来处理,这时,就需要强制编译器把变量或常数由声明时的类型转换成需要的类型。为此,就要使用强制类型转换说明,格式如下:

int* iptr=(int*) &table;

表达式的前缀(int*)就是传统C风格的强制类型转换说明(typecast),又可称为强制转换说明(cast)。强制转换说明告诉编译器把表达式转换成指定的类型。有些情况下强制转换是禁用的,例如不能把一个结构类型转换成其他任何类型。数字类型和数字类型、指针和指针之间可以相互转换。当然,数字类型和指针类型也可以相互转换,但通常认为这样做是不安全而且也是没必要的。强制类型转换可以避免编译器的警告。

long int el = 123;
short i = (int) el;

float m = 34.56;
int i = (int) m;

上面两个都是C风格的强制类型转换,C++还增加了一种转换方式,比较一下上面和下面这个书写方式的不同:

long int el = 123;
short i = int (el);


float m = 34.56;
int i = int (m); 

使用强制类型转换的最大好处就是:禁止编译器对你故意去做的事发出警告。但是,利用强制类型转换说明使得编译器的类型检查机制失效,这不是明智的选择。通常,是不提倡进行强制类型转换的。除非不可避免,如要调用malloc()函数时要用的void型指针转换成指定类型指针。


四、标准输入输出流

       在C语言中,输入输出是使用语句scanf()printf()来实现的,而C++中是使用类来实现的

#include iostream.h

main()    //C++中main()函数默认为int型,而C语言中默认为void型。
{
	int a;
	cout << input a number: ;
	cin >> a;             /*输入一个数值*/
	cout << a << endl;      //输出并回车换行
	
	return 0;
}

    cin,cout,endl对象,他们本身并不是C++语言的组成部分。虽然他们已经是ANSI标准C++中被定义,但是他们不是语言的内在组成部分。在C++中不提供内在的输入输出运算符,这与其他语言是不同的。输入和输出是通过C++类来实现的,cin和cout是这些类的实例,他们是在C++语言的外部实现。

     在C++语言中,有了一种新的注释方法,就是‘//’,在该行//后的所有说明都被编译器认为是注释,这种注释不能换行。C++中仍然保留了传统C语言的注释风格/*……*/。
C++也可采用格式化输出的方法:
#include iostream.h
       
int main()
{          
	int a;           
	cout << input a number: ;         
	cin >> a;
           
	cout << dec << a << ' '     //输出十进制数               
	<< oct << a << ' '     //输出八进制数               
	<< hex << a << endl;   //输出十六进制数
           
	return 0;       
}

从上面也可以看出,dec,oct,hex也不可作为变量的标识符在程序中出现。


五、函数参数问题

1、无名的函数形参

       声明函数时可以包含一个或多个用不到的形式参数。这种情况多出现在用一个通用的函数指针调用多个函数的场合,其中有些函数不需要函数指针声明中的所有参数。看下面的例子:
int fun(int x,int y)
{
	return x*2;
}
    尽管这样的用法是正确的,但大多数C和C++的编译器都会给出一个警告,说参数y在程序中没有被用到。为了避免这样的警告,C++允许声明一个无名形参,以 告诉编译器存在该参数,且调用者需要为其传递一个实际参数,但是函数不会用到这个参数。下面给出使用了无名参数的C++函数代码:
int fun(int x,int) //注意不同点
{
	return x*2;
}

2、函数的默认参数

      C++函数的原型中可以声明一个或多个带有默认值的参数。如果调用函数时,省略了相应的实际参数,那么编译器就会把默认值作为实际参数。可以这样来声明具有默认参数的C++函数原型:
#include iostream.h

void show(int = 1,float = 2.3,long = 6);

int main()
{
	show();
	show(2);
	show(4,5.6);
	show(8,12.34,50L);
	
	return 0;
}

void show(int first,float second,long third)
{
	cout << first =<< first
		<< second =<< second
		<< third =<< third << endl;
}

上面例子中,第一次调用show()函数时,让编译器自动提供函数原型中指定的所有默认参数,第二次调用提供了第一个参数,而让编译器提供剩下的两个,第三次调用则提供了前面两个参数,编译器只需提供最后一个,最后一个调用则给出了所有三个参数,没有用到默认参数。


六、函数重载

在C++中,允许有相同的函数名,不过它们的参数类型不能完全相同,这样这些函数就可以相互区别开来。而这在C语言中是不允许的。

1、参数个数不同

#include iostream.h

void a(int,int);
void a(int);

int main()
{
	a(5);
	a(6,7);
	
	return 0;
}


void a(int i)
{
	cout << i << endl;  //输出5
} 


void a(int i,int j)
{
	cout << i << j << endl;       //输出67
}

2.参数格式不同

#include iostream.h

void a(int,int);
void a(int,float);

int main()
{
	a(5,6);
	a(6,7.0);

	return 0;
}

void a(int i,int j)
{

	cout << i << j <<endl;          //输出56
} 

void a(int i,float j)
{
	cout << i << j << endl;          //输出67.0

}


七、变量作用域      

    C++语言中,允许变量定义语句在程序中的任何地方,只要在是使用它之前就可以;而C语言中,必须要在函数开头部分。而且C++允许重复定义变量,C语言也是做不到这一点的。看下面的程序:

#include iostream.h

int a;

int main()
{
	cin >> a;
	for(int i = 1;i <= 10; i++) //C语言中,不允许在这里定义变量
	{
		static int a = 0; //C语言中,同一函数块,不允许有同名变量
		a += i;
		cout<<::a<< <<a<<endl;
        }		
	return 0;
}


八、new和delete运算符

在C++语言中,仍然支持malloc()和free()来分配和释放内存,同时增加了new和delete来管理内存。

1.为固定大小的数组分配内存

#include iostream.h

int main()
{
	int *birthday = new int[3];
	birthday[0] = 6;
	birthday[1] = 24;
	birthday[2] = 1940;
	cout << I was born on
		<< birthday[0] << '/' << birthday[1] << '/' << birthday[2] << endl;
	delete [] birthday;      //注意这儿
	
	return 0;
}

在删除数组时,delete运算符后要有一对方括号。

2.为动态数组分配内存

#include iostream.h
#include stdlib.h

int main()
{
	int size;
	cin >> size;
	int *array = new int[size];
	for(int i = 0;i < size;i++)
		array[i] = rand();
        for(i = 0;i < size;i++)
		cout << '\n' << array[i];
        delete [] array;
	
	return 0;
}


九、引用型变量

在C++中,引用是一个经常使用的概念。引用型变量是其他变量的一个别名,我们可以认为他们只是名字不相同,其他都是相同的。

1.引用是一个别名

    C++中的引用是其他变量的别名。声明一个引用型变量,需要给他一个初始化值,在变量的生存周期内,该值不会改变。& 运算符定义了一个引用型变量:

int a;

int& b=a;

先声明一个名为a的变量,它还有一个别名b。我们可以认为是一个人,有一个真名,一个外号,以后不管是喊他a还是b,都是叫他这个人。同样,作为变量,以后对这两个标识符操作都会产生相同的效果。

#include iostream.h

int main()
{
	int a = 123;
	int& b = a;
	cout << a << ','<< b << endl;       //输出123,123
	a++;
	cout << a << ','<< b << endl;       //输出124,124
	b++;
	cout << a<< ',' << b << endl;        //输出125,125

	return 0;
}


2.引用的初始化

和指针不同,引用变量的值不可改变。引用作为真实对象的别名,必须进行初始化,除非满足下列条件之一:

(1) 引用变量被声明为外部的,它可以在任何地方初始化

(2) 引用变量作为类的成员,在构造函数里对它进行初始化

(3) 引用变量作为函数声明的形参,在函数调用时,用调用者的实参来进行初始化


3.作为函数形参的引用

引用常常被用作函数的形参。以引用代替拷贝作为形参的优点:

引用避免了传递大型数据结构带来的额外开销

引用无须象指针那样需要使用*和->等运算符

#include iostream.h

void func1(s p);
void func2(s& p);

struct s
{
	int n;
	char text[10];
};

int main()
{
	static s str = {123,China};
	func1(str);
	func2(str);
	return 0;
}

void func1(s p)
{
	cout << p.n << endl;
	cout << p.text << endl;
}


void func2(s& p)
{
	cout << p.n << endl;
	cout << p.text << endl;
}
从表面上看,这两个函数没有明显区别,不过他们所花的时间却有很大差异,func2()函数所用的时间开销会比func2()函数少很多。它们还有一个差别,如果程序递归func1(),随着递归的深入,会因为栈的耗尽而崩溃,但func2()没有这样的担忧。


4.以引用方式调用

当函数把引用作为参数传递给另一个函数时,被调用函数将直接对参数在调用者中的拷贝进行操作,而不是产生一个局部的拷贝(传递变量本身是这样的)。这就称为以引用方式调用。把参数的值传递到被调用函数内部的拷贝中则称为以传值方式调用。
#include iostream.h

void display(const Date&,const char*);
void swapper(Date&,Date&);

struct Date
{
	int month,day,year;
};

int main()
{
	static Date now={2,23,90};
	static Date then={9,10,60};
	display(now,Now: );
	display(then,Then: );
	swapper(now,then);
	display(now,Now: );
	display(then,Then: );
	
	return 0;
}

void swapper(Date& dt1,Date& dt2)
{
	Date save;
	save=dt1;
	dt1=dt2;
	dt2=save;
}

void display(const Date& dt,const char *s)
{
	cout << s;
	cout << dt.month << '/' << dt.day << '/'<< dt.year << endl;
}

5.以引用作为返回值

#include iostream.h

struct Date
{
	int month,day,year;
};

Date birthdays[]=
{
	{12,12,60};
	{10,25,85};
	{5,20,73};
};

const Date& getdate(int n)
{
	return birthdays[n-1];
}

int main()
{
	int dt=1;
	while(dt!=0)
	{
		cout<<Enter date # (1-3,0 to quit)<<endl;
		cin>>dt;
		if(dt>0 && dt<4)
		{
			const Date& bd = getdate(dt);
			cout << bd.month << '/' << bd.day << '/'<< bd.year << endl;
		}
	}
	return 0;
}


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

C++ 学习基础篇(一)—— C++与C 的区别 的相关文章

  • 云原生周刊:一文读懂 Pod 网络 | 2023.4.10

    文章推荐 一文读懂 Pod 网络 这篇文章旨在帮助读者理解 Pod 网络的概念和原理 Pod 网络是 Kubernetes 中的一个重要概念 xff0c 它描述了如何在一个集群中部署和运行应用程序 Pod 网络是指使用容器网络插件 如 Ca
  • 前辈大公司的面试,重点是他推荐我们应该看得那些书

    应届生上泡了两年 xff0c 一直都是下资料 xff0c 下笔试题 xff0c 面试题 一直都在感谢那些默默付出的人 写这个帖子花了我两个夜晚的时间 xff0c 不是为了炫耀 xff0c 只是为了能给那些 迷惘 的学弟学妹 xff0c 一点
  • 数据库for update 之后未提交事务导致锁表

    在工作的时候 xff0c 操作数据库 xff0c select for update xff0c 忘记提交事务 xff0c 数据库为了防止其他人对该表进行操作 xff0c 对该表进行锁表 xff0c 导致我再次for update 的时候一
  • 芯片驱动程序编写

    实质 利用程序控制单片机与芯片通信 xff0c 目的是读写芯片 xff0c 一般来说 xff0c 驱动程序就是对芯片的读写操作 看数据手册 寄存器表 芯片的所有功能都 映射 在寄存器表上 xff0c 阅读寄存器表就可以了解芯片的功能 这部分
  • 如何快速入门RTOS

    摘要 本文结合自己学习RTOS的经历 xff0c 来谈谈如何快速入门一款RTOS xff0c 希望能够给初学者以启发 xff0c 找到适合自己的学习思路和方法 我的学习经历 ucos学习 我是在上学期间接触到了RTOS xff0c 当时学习
  • RT-Thread快速入门-了解内核启动流程

    首发 xff0c 公众号 一起学嵌入式 xff0c RTOS Linux C 内核是操作系统最基础也是最重要的部分 从本文开始进入 RT Thread 内核相关知识的学习 首先了解内核的基础知识 xff0c 对 RT Thread 内核的设
  • RT-Thread快速入门-中断管理

    首发 xff0c 公众号 一起学嵌入式 经过前面文章的学习 xff0c 对于 RT Thread 处理多任务或者说线程的处理机制 xff0c 基本上入门了 能够上手用 RT thread 进行日常开发了 但是 xff0c 还有一个重要的部分
  • FreeRTOS快速入门-初探FreeRTOS

    首发 xff0c 公众号 一起学嵌入式 对于 RTOS 入门系列文章 xff0c 已经更新完一款 xff08 RT Thread xff09 xff1a 助你快速入门 RT Thread 这个系列的文章结合 RT Thread xff0c
  • 有人看衰OpenStack,这家公司却给出不一样的感觉

    前一段时间我们写过九州云的文章 xff0c 主要谈的是新产品Animbus 7 0和九州云的一些战略规划 xff0c 有兴趣可以翻阅 OpenStack年年痒 xff0c 但并不妨碍越来越多的用户选择它 一文 日前 xff0c 值 双态IT
  • 我读Mongoose源码----程序框架

    Mongoose是一种WEB服务器 xff0c 因为最近在学习网络编程 xff0c 所以打算研究研究它的源码 xff0c 认真看了大部分 xff0c 觉得学到的东西的确不少 xff0c 拿出来分享一下 xff0c 也和大家交流交流 至于什么
  • git的简单使用

    以前一直没有提交过代码 xff0c 这次提交一下代码 xff0c 整理一下 xff47 xff49 xff54 的简单使用 1 首先我们要在github上面创建一个帐号 xff0c 之后创建一个仓库create a new repo xff
  • 二叉查找树的实现

    二叉查找树是这样定义的 xff1a 二叉查找树 xff08 Binary Search Tree xff09 xff0c 或者是一棵空树 xff0c 或者是具有下列性质的二叉树 xff1a 若它的左子树不空 xff0c 则左子树上所有结点的
  • 翻转n个硬币的问题

    今天去面试 面试官问了我这样一个问题 当时答的很近了 但是还差一点 最后还是被pass了 原题是这样 一堆硬币有n个 都是朝下的 翻转n次 第一次翻转能被1整除的 第2次翻转能被2整除的 第三次翻转能被3整除的 这样直到第n次翻转能被n整除
  • 10月15号 360一面

    昨天去360参见一面 xff0c 作为小本一枚 xff0c 迅雷 xff0c 多玩 xff0c 360都是霸笔才得到的面试机会 xff0c 说实在的感觉360一面聊了40分钟 xff0c 整个过程比较轻松 面试官比我大个十岁左右 xff0c
  • kaggle网站原数据集Give Me Some Credit

    基于Give Me Some Credit数据集 xff0c 通过预测某人在未来两年内经历财务困境的可能性 xff0c 改进信用评分的先进水平 信用评分算法 xff0c 猜测违约的可能性 xff0c 是银行用来决定是否应该发放贷款的方法 这
  • linux下查看磁盘分区,文件系统,磁盘文件系统的命令

    http www linuxsir org bbs thread214738 html 一 df 命令 xff1b df 是来自于coreutils 软件包 xff0c 系统安装时 xff0c 就自带的 xff1b 我们通过这个命令可以查看
  • Redis源码-数据结构之Adlist双端链表

    Redis的Adlist实现了数据结构中的双端链表 xff0c 整个结构如下 xff1a 链表节点定义 xff1a typedef struct listNode struct listNode prev struct listNode n
  • Redis源码-事件库

    网上看了很多Redis事件库的解读 xff0c 自己也研究了好几遍 xff0c 还是记录下来 xff0c 虽然水平有限 xff0c 但是进步总会是有的 网络事件库封装了Epoll的操作 xff08 当然是指Linux下的多路复用了 xff0
  • Redis源码分析-内存数据结构intset

    这次研究了一下intset xff0c 研究的过程中 xff0c 一度看不下过去 xff0c 但是还是咬牙挺过来了 xff0c 看懂了也就是那么回事 xff0c 静下心来 xff0c 切莫浮躁 Redis为了追求高效 xff0c 在存储下做
  • 基于STM32实验:uC/OS-III操作系统移植并创建多任务系统实验

    本次实验内容 xff1a 将uC OS III操作系统移植到stm32F103C8T6上 xff0c 构建至少3个任务 xff08 task xff09 xff1a 分别以1s和3s周期对LED灯进行点亮 熄灭的控制 xff1b 另外一个t

随机推荐

  • 初级软件测试面试题汇总

    一 请描述如何划分缺陷与错误严重性和优先级别 xff1f 给软件缺陷与错误划分严重性和优先级的通用原则 xff1a xff08 1 xff09 表示软件缺陷所造成的危害和恶劣程度 xff08 2 xff09 优先级表示修复缺陷的重要程度和次
  • selenium 安装

    火狐浏览器版本 xff1a 35 0 1 1 官网安装 Version 2 9 1 1 通过官网安装插件 xff1a https addons mozilla org en GB firefox addon selenium ide ver
  • 构造函数与析构函数执行顺序

    构造函数与析构函数执行顺序 代码 xff1a include lt iostream gt using namespace std class ABCD public ABCD int a int b int c this gt a 61
  • java.lang.NoSuchMethodException异常

    在Struts2中 xff0c 有时候会出现java lang NoSuchMethodException异常 xff0c 有可能是三种情况导致的运行异常 xff1a 第一种 xff1a Action 类的方法被定义成 private 类型
  • java.lang.IllegalArgumentException异常解决

    在maven项目中测试代码的时候 xff0c 碰到java lang IllegalArgumentException 异常 xff1a 严重 Servlet service for servlet e3 manager in contex
  • 在idea中创建一个普通工程

    第一步 xff1a File gt new gt Project 第二步 xff1a 点击next 点击 finish 即可 xff01 xff01 xff01 运行结果
  • java:获取当月最后一天

    设置时间格式 SimpleDateFormat format 61 new SimpleDateFormat 34 yyyy MM dd 34 获得实体类 Calendar ca 61 Calendar getInstance 设置最后一天
  • idea自动生成UUID和解决办法

    正常情况下 xff0c 鼠标点击类名 xff0c Alt 43 Insert键就会出现生成UUID选项 xff0c 即 xff1a 有时候Alt 43 Insert没有UUID选项 xff0c 解决办法 第一种情况 xff1a Settin
  • 页面<div>位置调整

    调整页面 lt div gt 样式 给 lt div gt lt select gt 分别起名字 xff1a div2 xff0c s1 lt div gt 代码 xff1a lt div class 61 34 div2 34 style
  • 马士兵_JAVA自学之路(为那些目标模糊的码农们)

    转载自 xff1a https blog csdn net anlidengshiwei article details 42264301 JAVA自学之路 一 学会选择 为了就业 xff0c 不少同学参加各种各样的培训 决心做软件的 xf
  • 在深度学习中Softmax交叉熵损失函数的公式求导

    以下部分基本介绍转载于点击打开链接 在深度学习NN中的output层通常是一个分类输出 xff0c 对于多分类问题我们可以采用k 二元分类器来实现 xff0c 这里我们介绍softmax softmax回归中 xff0c 我们解决的是多分类
  • 1-基于ArUco码的标记与检测

    1 简介 姿态估计 xff08 Pose estimation xff09 在 计算机视觉领域扮演着十分重要的角色 xff1a 机器人导航 增强现实以及其它 这一过程的基础是找到现实世界和图像投影之间的对应点 这通常是很困难的一步 xff0
  • 4-基于ArUco相机姿态评估

    1 简介 基于ArUco评估相机姿态 xff0c 可以使用OPENCV的外部库 xff08 opencv contrib xff09 中的aruco模块 xff0c 可以参考安装目录 xff08 库目录 xff09 xff1a opencv
  • MySQL--40道基础概念选择题及答案

    一 单选题 xff08 题数 xff1a 40 xff0c 共 40 0 分 xff09 1 在计算机系统中能够实现对数据库资源进行统一管理和控制的是 xff08 A xff09 A DBMS B DBA C DBS D DBAS 2 数据
  • 抽象类方法——子类定义getDescription方法返回对一个人的简单描述

    Person与子类的关系图 每一个 人都有一些诸如名字这样的属性 xff0c 学生与雇员都有名字属性 xff0c 因此可以将getName方法放在位于继承关系较高层的通用超类 xff08 父类 xff09 中 xff0c 现在增加一个get
  • Exynos4412 Uboot 移植(一)—— Uboot 编译流程分析

    Uboot 所用版本 u boot 2013 01 u boot 2013 01 中有上千文件 xff0c 要想了解对于某款开发板 xff0c 使用哪些文件 哪些文件首先执行 可执行文件占用内存的情况 xff0c 最好的方法就是阅读它的Ma
  • Linux USB 驱动开发(五)—— USB驱动程序开发过程简单总结

    设备驱动程序是操作系统内核和机器硬件之间的接口 xff0c 由一组函数和一些私有数据组成 xff0c 是应用程序和硬件设备之间的桥梁 在应用程序看来 xff0c 硬件设备只是一个设备文件 xff0c 应用程序可以像操作普通文件一样对硬件设备
  • 路由器开发(一)—— 路由器硬件结构及软件体系

    一 路由器的硬件构成 路由器主要由以下几个部分组成 xff1a 输入 输出接口部分 包转发或交换结构部分 xff08 switching fabric xff09 路由计算或处理部分 如图所示 图1 路由器的基本组成 输入端口是物理链路和输
  • Linux 设备驱动开发思想 —— 驱动分层与驱动分离

    前面我们学习I2C USB SD驱动时 xff0c 有没有发现一个共性 xff0c 就是在驱动开发时 xff0c 每个驱动都分层三部分 xff0c 由上到下分别是 xff1a 1 XXX 设备驱动 2 XXX 核心层 3 XXX 主机控制器
  • C++ 学习基础篇(一)—— C++与C 的区别

    编程的学习学无止境 xff0c 只掌握一门语言是远远不够的 xff0c 现在我们开始C 43 43 的学习之路 xff0c 下面先看下C 43 43 与C 的区别 一 C 43 43 概述 1 发展历史 1980年 xff0c Bjarne