C语言:春节回家过年,我发现只有我没有对象!

Wesley13
• 阅读 479

【回复“1024”,送你一个特别推送】

C语言:春节回家过年,我发现只有我没有对象!

文章转自刘欣大佬的「码农翻身」,文章实在太好了,忍不住分享给大家。

聚会

C语言春节回家过年,遇到了不少小伙伴:Java , Python, JavaScript,Ruby......

大家在大城市发展得都不错,回到老家,聚到一起吃饭, 谈天说地,都是喜气洋洋。

尤其是Python和JavaScript,更是成了明星,一个吹嘘说自己是人工智能的必备,另外一个炫耀说自己是世界上最流行的语言,不信有某某语言流行度排行榜为证, 还有GitHub上的众多项目云云。

老练的Java则是一直拿TIOBE排行榜说事儿:“我已经连续10多年排行第一了,高处不胜寒啊!”

提到TIOBE,Python更是得意:“我今年还被选为TIOBE的年度编程语言呢!”

虽然常年排名TIOBE第二,C语言有点黯然神伤,人类用自己写的程序可真不少,可都是处于底层,在系统级编程,什么操作系统,数据库,编译器...... 与应用层比起来,没那么光鲜亮丽。

现在很多人培训了Python, Java 就说自己会编程了, 不懂指针,不懂内存,不懂底层的基本原理, 那能算会编程吗?

C语言开始愤愤不平,闷头吃菜,似乎要把这股郁闷之气发泄到美味佳肴上去。

觥筹交错之间,Java 搂住C的肩膀,亲切地说:“兄弟,你有对象了吗?”

这下可捅了马蜂窝,大家的眼光齐刷刷地聚集到C语言的身上。

C嚅嗫了半天:“没...... 没有。”

“哈哈哈...... 我们都有对象,你这么大了还没对象?!”  Python笑道。

“是啊,一个没有对象的编程语言还有什么前途?”JavaScript补刀,他原来没有class的概念,是通过“原型”实现的OOP,最近几年才在语法层面引入class关键字。

“我虽然没有对象,但是有指针啊,功能非常强大。”

“指针?你说的是那容易出错的指针吗? 现在有谁用指针啊?”  JavaScript说道。

“不会用指针,就不是真正的程序员!”  C语言涨红了脸。

餐桌的气氛变得有些尴尬,捅了篓子的Java招呼着说:“来来来,继续喝酒。”

好不容易熬到聚餐结束,C语言回到了自己的家,家里冷冷清清,自己的“亲爹”丹尼斯·里奇(Dennis Ritchie),有史以来最伟大的程序员之一, 已经于2011年10月不幸去世。

桌子上摆着的一本《C程序设计语言》,那是丹尼斯·里奇唯一的遗著, 拿起这本书,C不由悲从心来。

C语言:春节回家过年,我发现只有我没有对象!

串门

C语言突然想起来对门的 Ken Thompson,那是Dennis Ritchie的“好基友”,他们俩一起创造了伟大的Unix操作系统,获得了计算机界的最高奖:图灵奖。

要不问问Ken? 为什么不让我有对象?不让我面向对象编程!

C来到Ken Thompson的门口,按了门铃,门开了,C语言一眼就看到Ken Thompson正在和Go玩得不亦乐乎,心中更是凄苦,Go才是人家的亲儿子,我算老几, 转身便要离去。

Ken 却从后面叫住了他:“小C啊,快进来,和你的兄弟Go玩一会儿。”

看到C满脸沮丧,Ken也大为吃惊:“大过年的,怎么回事?”

C不满地说:“当年你们为什么不让我有对象?”

“对象,什么对象? 奥,你是说面向对象编程吧!其实吧你亲爹把你设计出来,主要是做系统级编程的,要的是贴近硬件,要的是效率,要那复杂玩意儿干啥?中看不中用,再说了,你和Go一样,不是有struct吗? ”  Ken 转向Go,挤了挤眼睛。

“是啊是啊,struct很好用的!”  Go马上附和。

“但是struct也实现不了OOP啊, Python,JavaScript他们都嘲笑我! ”

“那你说说,什么是OOP?”Ken问道。

“嗯,就是封装、继承、多态吧? ” C回答到。

“好,我来给你掰扯掰扯,用C语言怎么实现封装、继承还有多态!”

封装

Ken Thompson 真不愧是老司机,唰唰唰迅速就写成了一段代码。

他说道:“我们先来说说封装,这封装就是把信息给隐藏起来,你先看看这段代码。”

shape.h

C语言:春节回家过年,我发现只有我没有对象!

shape.c

C语言:春节回家过年,我发现只有我没有对象!

main.c

C语言:春节回家过年,我发现只有我没有对象!

这里定义了一个叫做Shape的结构体,外界只能通过相关的函数来对这个Shape进行操作,例如创建(Shape_create), 移动(Shape_move), 还有获取位置(Shape_getX)等,不能直接访问Shape的内部数据结构。

虽然这里没有class这样的关键字,数据结构和相关操作是分开写的,看起来不太完美, 但确实是实现了封装。

C语言:春节回家过年,我发现只有我没有对象!

C 看到Ken Thompson居然把那个指针的名称叫做self, 和Python的相同,不由得笑了起来:“我明白了,那继承该怎么做呢?”

继承

Ken Thompson不吭声,继续写代码。

大牛的风格看来都是类似的: 别瞎BB,给我上代码。

C语言:春节回家过年,我发现只有我没有对象!

这次定义了一个矩形(Rectangle)的结构体,其中嵌套了Shape,难道这就实现了继承?  C有点疑惑。

Go小子在旁边叫了起来:“我明白了,在内存中,他们是这样的。”

C语言:春节回家过年,我发现只有我没有对象!

通过这种组合的方式,也算是实现了继承吧。

C语言:春节回家过年,我发现只有我没有对象!

多态

这么轻松就实现了封装和继承,C语言感到很兴奋, 但是多态怎么实现呢?

这时候又传来了门铃声,Linus大神拎着一瓶酒进来,要找C小伙儿喝酒,看到这桌子上的代码,立刻就明白了怎么回事。

他说道:“别整那么多花里胡哨的东西,还多态,不就是函数指针嘛! 我给你举个例子。”

C语言:春节回家过年,我发现只有我没有对象!

“这个结构体包含了两个函数指针,一个用来计算图形的面积,另外一个把这个图形画出来。我们把这个结构体叫做虚函数表。”

“这有什么用啊?”

“在你的Shape中,添加一个指向该函数表的指针就行了。” Linus回答。

C语言:春节回家过年,我发现只有我没有对象!

C和Go都是一脸茫然。

“蠢材, 你们想想啊,当你创建一个子类对象的时候,比如Rectangle, 把那个虚函数指针vptr指向另外一组函数,会怎么样?”

两人还是不懂,Linus只好继续画图:

C语言:春节回家过年,我发现只有我没有对象!

现在C有点明白了, 无论是Rectangle对象,还是Square对象,在调用Shape_area方法的时候, 都需要通过vptr这个指针找到虚函数表中的area方法,对于Rectangle,找到的是Rectangel_area方法,对于Square,找到的是Square_area方法。

struct Rectangle *r = Rectangle_create(5,5,10,10);

Shape_area((struct Shape *) r);

“其实吧,你的兄弟C++的多态实现原理也是类似的!在运行时查找真正的函数去执行。”  Ken 总结到。

“对,这种函数指针的使用方法太常见了,在我的Linux操作系统中也会定义类似的东西” Linus接口道,

C语言:春节回家过年,我发现只有我没有对象!

“只要IO设备提供这几个函数的实际定义,就可以将File结构体的函数指针指向对应的实现,那就实现了用同一套接口操作不同的IO设备。”

C语言高兴起来:“哈哈,我就说我的指针很厉害吧,这些全是通过指针来实现的。”

“是啊,别听Java, Python, JavaScript他们瞎BB,你也有对象,也能进行面向对象的编程!”

C语言说道:“走,喝酒去!”

注: 本文的例子主要来源于https://www.state-machine.com/doc/AN\_OOP\_in\_C.pdf  我做了修改。

公众号对话框,回复关键字“1024”

免费领取30本经典编程书籍

- 长按识别关注 -

C语言:春节回家过年,我发现只有我没有对象!

技术,职场,产品,思维

行业观察

本文分享自微信公众号 - 非著名程序员(non-famous-coder)。
如有侵权,请联系 support@oschina.cn 删除。
本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一起分享。

点赞
收藏
评论区
推荐文章
blmius blmius
2年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Jacquelyn38 Jacquelyn38
2年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
Stella981 Stella981
2年前
KVM调整cpu和内存
一.修改kvm虚拟机的配置1、virsheditcentos7找到“memory”和“vcpu”标签,将<namecentos7</name<uuid2220a6d1a36a4fbb8523e078b3dfe795</uuid
Easter79 Easter79
2年前
Twitter的分布式自增ID算法snowflake (Java版)
概述分布式系统中,有一些需要使用全局唯一ID的场景,这种时候为了防止ID冲突可以使用36位的UUID,但是UUID有一些缺点,首先他相对比较长,另外UUID一般是无序的。有些时候我们希望能使用一种简单一些的ID,并且希望ID能够按照时间有序生成。而twitter的snowflake解决了这种需求,最初Twitter把存储系统从MySQL迁移
Wesley13 Wesley13
2年前
mysql设置时区
mysql设置时区mysql\_query("SETtime\_zone'8:00'")ordie('时区设置失败,请联系管理员!');中国在东8区所以加8方法二:selectcount(user\_id)asdevice,CONVERT\_TZ(FROM\_UNIXTIME(reg\_time),'08:00','0
Wesley13 Wesley13
2年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Stella981 Stella981
2年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable
Wesley13 Wesley13
2年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
Python进阶者 Python进阶者
3个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这