一乐电子

 找回密码
 请使用微信账号登录和注册会员

QQ登录

只需一步,快速开始

微信扫码登录

手机号码,快捷登录

手机号码,快捷登录

搜索
查看: 3836|回复: 3

用 C 语言实现程序的多态性

[复制链接]
发表于 2016-6-19 22:35 | 显示全部楼层 |阅读模式
本帖最后由 kenson 于 2016-6-19 22:36 编辑 ! j8 B; h& w: z% A
  X: a( g' e! d3 {
多态 (polymorphism) 一词最初来源于希腊语 polumorphos,含义是具有多种形式或形态的情形。在程序设计领域,一个广泛认可的定义是“一种将不同的特殊行为和单个泛化记号相关联的能力”。然而在人们的直观感觉中,多态的含义大约等同于“同一个方法对于不同类型的输入参数均能做出正确的处理过程,并给出人们所期望获得的结果”,也许这正体现了人们对于多态性所能达到的效果所寄予的期望:使程序能够做到越来越智能化,越来越易于使用,越来越能够使设计者透过形形色色的表象看到代码所要触及到的问题本质。6 `( U: W5 L' y; @- P9 A4 Q: g
. d4 j/ x' _3 v. U1 ~% S& q: z0 W9 V0 n9 Z
作为读者的你或许对于面向对象编程已有着精深的见解,或许对于多态的方便与神奇你也有了深入的认识。这时候你讶异的开始质疑了:“多态,那是面向对象编程才有的技术,C 语言是面向过程的啊!”而我想说的是,C 语言作为一种编程语言,也许并不是为了面向对象编程而设计,但这并不意味着它不能实现面向对象编程所能实现的功能,就比如说,多态性。
3 m! J. E3 Q# o$ w$ M7 R0 D" W# `1 ~  t* ], c
在本文中我们使用一个简单的单链表作为例子,展示 C 语言是如何体现多态性的。
& ?. b  Y: y" \; T  m# p' K$ e
. @% f. ~3 |! A  B1 Z" Y: i$ |回页首1 J, `7 e! j4 l* E1 F2 n0 z

4 z# M8 j/ L8 S, C2 {  k2 I结构体:不得不说的故事+ Q4 ^; C# O8 Z  _, B+ V, r
0 u5 o2 m, Z" O; d& P' q
许多从写 C 代码开始,逐渐走向 C++ 的程序员都知道,其实 C++ 里面的 class,其前身正是 C 语言中的 structure。很多基于 C 语言背景介绍 C++ 的书籍,在介绍到 class 这一章的时候都会向读者清晰地展示,一个 C 语言里的 structure 是怎样逐渐变成一个典型的 C++ class 的,甚至最后得出结论:“structure 就是一个所有成员都公有的类”,当然了,class 还是 class,不能简单的把它看做一个复杂化了的 structure 而已。
; p6 U, e  W4 a3 _' Q3 Z/ G& ^3 B: V- `
下面我们来看看在 C 语言中定义一个简单的存储整型数据的单链表节点是怎么做的,当然是用结构体。大部分人会像我一样,在 linkList.h 文件里定义:0 f' S; q- G4 o* n# j

; i( P+ V) M1 h( n: {" ~+ ] typedef struct Node* linkList; $ V5 J0 m1 A2 Q6 V2 G5 `- N
struct Node                                       // 链表节点* ?/ |8 r( W6 D5 `# U* C
{ * s( n* I8 @# C
int data;                                 // 存储的整型数据/ q, p4 q5 m2 y) i
linkList next;                            // 指向下一个链表节点
) j  P5 g3 c, A7 V };
- h6 u. B$ v' K" m* c; Z链表有了,下面就是你想要实现的一些链表的功能,当然是定义成函数。我们只举几个常用功能:# |$ P8 f5 ]8 C/ w% Z
9 `1 M4 w7 E" F( }6 c
linkList initialLinklist();                               // 初始化链表+ ~' X; ~- O7 _, k
link newLinkList (int data);                        // 建立新节点
4 R3 Y1 E! O4 r7 `4 m void insertFirst(linkList h,int data);              // 在已有链表的表头进行插入节点操作
  R$ @: o) j4 C6 ~+ l. X void linkListOutput(linkList h);                           // 输出链表中数据到控制台3 o* h" f" U/ {0 O- t' n9 y
这些都是再自然不过的 C 语言的编程过程,然后我们就可以在 linkList.c 文件中实现上述两个函数,继而在 main.c 中调用它们了。5 A2 f0 ]9 w% L. ]) @1 r

: f( {/ c' D5 s然而上面我们定义的链表还只能对整型数据进行操作。如果下次你要用到一个存储字符串类型的链表,就只好把上面的过程重新来过。也许你觉得这个在原有代码基础上做略微修改的过程并不复杂,可是也许我们会不断的增加对于链表这个数据结构的操作,而需要用链表来存储的数据类型也越来越多,这些都意味着海量的代码和繁琐的后期维护工作。当你有了上百个存储不同数据类型的链表结构,每当你要增加一个操作,或者修改某个操作的传入参数,工作量会变大到像一场灾难。+ f$ Y+ _7 T1 j

# U6 |% E, n# l& a1 d' Q但是我们可以改造上述代码,让它能够处理你所想要让它处理的任何数据类型:实行,字符型,乃至任何你自己定义的 structure 类型。
: X4 S, \- G6 i  ^. m7 S' M
/ V. {) t! i" A% W: ^) _( J回页首
8 u. Z) Q/ p0 s$ Z5 N) J8 f
6 A. }. h2 \5 {! w, h. gVoid*:万能的指针“挂钩”
$ {" U4 Z* \. u  N, t/ |; e; F' L" d9 ~! v/ s( G8 |$ R
几乎所有讲授 C 语言课程的老师都会告诉你:“指针是整个 C 语言的精髓所在。”而你也一直敬畏着指针,又爱又恨地使用着它。许多教材都告诉你,int * 叫做指向整型的指针,而 char * 是指向字符型的指针,等等不一而足。然而这里有一个另类的指针家族成员—— void *。不要按照通常的命名方式叫它做指向 void 类型的指针,它的正式的名字叫做:可以指向任意类型的指针。你一定注意到了“任意类型”这四个字,没错,实现多态,我们靠的就是它。# _5 p9 R, w( |$ @& d' E. d

; x. a& N0 h- B6 I/ ^9 U; f下面来改造我们的链表代码,在 linkList.h 里,如下:
5 Q9 B8 n( M' U2 }& t7 x' e
# ^  R# E5 y8 r& _* S2 Q* t/ N  ^ typedef struct Node* linkList; + c( \: F9 K8 Q& v  ~8 X5 ~5 {
struct Node                                       // 链表节点: |+ v- G" _  S2 L4 M
{ & {( W$ U- I( j! Q7 _
void *data;                               // 存储的数据指针' A! D' o5 U/ L- r8 c3 S& V1 ?
linkList next;                            // 指向下一个链表节点
3 C2 H9 @& h# ^0 G }; 5 a  A' A6 O. N6 N! X
6 e7 ]) z8 R3 \$ d/ e- f1 y: W: L
linkList initialLinklist();                             // 初始化链表
  s* H3 X" w9 p* Y link newLinkList (void *data);                    // 建立新节点" ]3 q/ n9 M3 i+ y7 c
void insertFirst(linkList h, void *data);         // 在已有链表的表头进行插入节点操作/ b# _( f' N. W! b; l) N$ R
void linkListOutput(linkList h);                         // 输出链表中数据到控制台# A. S$ P- {$ q; f
我们来看看现在这个链表和刚才那个只能存储整型数据的链表的区别。
2 V/ D3 r" ?1 M) m4 Y$ z- v  h3 Q
- T# y& s* G7 b8 S$ Y当你把 Node 结构体里面的成员定义为一个整型数据,就好像把这个链表节点打造成了一个大小形状固定的盒子,你定义一个链表节点,程序进行编译的时候编译器就为你打造一个这样的盒子:装一个 int 类型的数据,然后装一个 linkList 类型的指针。如果你想强行在这个盒子里装别的东西,编译器会告诉你,对不起,盒子的大小形状并不合适。所以你必须为了装各种各样类型的数据打造出不同的生产盒子的流水线,想要装哪种类型数据的盒子,就开启对应的流水线来生产。
$ f9 N8 g. ?/ t/ F
4 ~  z. i+ A5 x但是当你把结构体成员定义为 void *,一切都变得不同了。这时的链表节点不再像个大小形状固定的盒子,而更像一个挂钩,它可以挂上一个任意类型的数据。不管你需要存储什么类型的数据,你只要传递一个指针,把它存储到 Node 节点中去,就相当于把这个数据“挂”了上去,无论何时你都可以根据指针找到它。这时的链表仿佛变成了一排粘贴在墙上的衣帽钩,你可以挂一排大衣,可以挂一排帽子,可以挂一排围巾,甚至,你可以并排挂一件大衣一顶帽子一条围巾在墙上。void * 初露狰狞,多态离 C 语言并不遥远。, j7 L! k. t) v. Y4 l

7 u8 r$ R  S, |回页首
6 p8 O. X) T! f* J1 @
9 e" D' s+ b# V. u& H' n. n实现:你的多态你做主
$ G0 V6 ^0 V9 d8 F1 O$ ^7 V4 ~  J3 @1 w! x4 Z9 C
当你真正开始着手做这个工作的时候,你会发现把数据放入链表中的操作和普通的存放 int 类型的链表的实现并没有什么大的区别,很方便。但是当你要把已经存进去的数据读取出来的时候,就有一点麻烦了。对于 void * 类型的指针,编译器只知道它里面存储了一个地址,但是关于这个地址里的数据类型,编译器是没有任何概念的。毕竟我们不能指望编译器什么都知道,什么都能替你做好,所以存进去的数据的类型,作为程序员的我们必须清楚的知道,并且在取出这个数据的时候,用这一类型的指针来对 void * 做强制类型转换。% ~7 k1 N& W' f' e- n. [, v
* u, o. `0 `* k6 s0 U! q
为了方便的做到这一点,我采取的方法是在 Node 结构体中增加一个标识数据类型的域,并用一个枚举类型来存放这些数据类型。这时的 linkList.h 如下所示:
- n5 H, l$ _9 f2 G& c6 `
  L4 z. J2 b9 @8 P3 D) y, z/ [4 i# Z. U #ifndef LINKLIST_H $ I4 Z1 q' v3 C( e+ p
#define LINKLIST_H
$ q3 ]; N$ V" I0 r. t! P8 v4 n' |6 [4 W5 ?& |% l7 A
typedef struct Node* linkList;
% S# i; P5 D9 n- o6 @7 \1 I, \9 F( R! O( Q4 f5 h$ O
enum dataType , e9 S$ O; z! N3 X9 `
{ 5 d0 o$ m& x$ x
    INT, - B8 \% n$ R2 K1 Z
    DOUBLE,
3 \9 ^5 U# {% R" P    CHAR, 7 T4 Z! s" W8 G) k% o2 m
    STRING , J( S, T! `. C  u2 ^
};
8 Q* }# |0 k/ x$ y) ?# C4 @- S/ ~. G5 l
struct Node                                               // 链表节点
* ~- G8 U& A3 m( P, N {
# d( @+ l- e1 c7 T    void *data;                                       // 存储的数据指针: t- N1 Z" Q9 U% ?$ b8 `$ }
    int dataType;                                     // 存储数据类型
) \3 s  p/ b( L/ f5 e    linkList next;                                    // 指向下一个链表节点
2 e! Z8 y' A5 x% ?3 H };
8 U4 c# ^& B1 G% n- L
: x7 L& w  ?4 w1 d linkList initialLinklist();                               // 初始化链表
; k! P7 _$ k; }6 ~* f linkList newLinkList (void *data, int dataType);          // 建立新节点
( N9 k! }" u8 L& C void insertFirst(linkList h, void *data, int dataType);   // 在已有链表的表头进行插入节点操作' Z, L& _1 U+ C; M; f
void linkListOutput(linkList h);                          // 输出链表中数据到控制台
8 D+ v( ?8 p2 Q" m4 ^* q. E
9 |$ I4 N% `1 J$ j6 j #endif
& Y. m" u' |0 q1 r+ g初始化链表,代码如下:& e8 T3 A. g$ @4 g$ u4 `& f

- U1 E; i7 H3 ~( B: j- W. W linkList initialLinklist()
0 {- j( j& Y7 n { $ c, ~$ s' T! v, m8 J, k# X) U
linkList link = (linkList*)malloc(sizeof(*link));
( P9 ~# M, S2 ]4 E6 c  R. d" E8 b    link->data = NULL; ' ^8 T4 h, n& k+ k/ ?7 S  t
    link->dataType = -1;
7 }& H! A: J% ]& S5 y    link->next = NULL;
& R- P  [- P- }! L' ?1 |4 `" `* ?5 \9 g, @* B! ]5 v
    return link;
, I% e, J5 `7 x' P& V, A }
8 v4 `9 g6 z2 s# ]6 r4 P5 I建立新节点,代码如下:5 b/ M8 y; V1 y+ g. `
/ m  T% N6 a5 P0 k$ z& M; F9 V) W9 {
linkList newLinkList (void *data, int dataType) : o' u" Q) E1 a3 v5 I/ N7 d
{ 7 b# u) E$ T, [" w; R7 |7 M5 p
    linkList link = (linkList*)malloc(sizeof(*link));
$ T" d( q$ i  O    link->data = data;
! h' h& O% N) s* `8 A0 Y  u* B    link->dataType = dataType; ) X% X' f# d0 `
    link->next = NULL; 8 y4 Q6 p7 e9 q( |
$ X& k) t4 g' n0 b  C' O9 J' C
    return link; ( Z# j2 M. e" a! i, g
} ( S" s: J7 s5 \, R' z4 ^1 f
在已有链表的表头进行插入节点操作,代码如下:
) @& s5 O$ Y; o
  X* L& K. ^: S# Y( `! _ void insertFirst(linkList h, void *data, int dataType) % X6 u/ {# T: l, U$ O5 H0 K
{ 8 s6 l: E. _  v
    linkList l = newLinkList(data, dataType);
6 u+ F6 h% C0 H9 g    l->next = h->next; * R/ K1 X1 X1 n( }8 ~6 @5 f# Q. u
    h->next = l; ) P; d" n4 j" }, w
} 1 Y5 C4 ^2 u  M4 v
输出链表中数据到控制台,代码如下:1 w4 ^6 q+ F1 i& J% |  G
* k! ]3 m+ A$ e2 Q4 B! m
void linkListOutput(linkList h) # r& `+ T% e8 p4 R* T
{
! x  h7 k; Q6 L8 _. T  Q9 x    linkList p = h;
; M' L/ @0 n# e1 T' |; g
" X7 V  k0 G! g    p = p->next; 1 c+ d1 ^3 `' j/ Q7 i
    while(p != NULL) 0 Y0 _$ t  C+ i$ X$ Q
    { 3 m7 l" k  q7 Z( ~
      switch(p->dataType)
) S4 u1 v" y  v4 q    {
# A5 X2 i) Y# O( U. u6 t       case 0:
8 D! ^( P! |5 H0 }& Z( _       {
$ M9 ]5 ]& }0 V( i: `. L7 K4 n         printf("%4d", *(int*)(p->data));
. {  f1 d6 j( d9 K- K9 l9 m         break; 6 p6 y/ y8 k" X% A- z
       }
, W; y) A/ v+ B8 G5 _: d       case 1:
  S% L- P9 L6 R! s8 ?       { 0 ?6 |! K! m7 a3 r, e
         printf("%4f", *(double*)(p->data));
/ m$ v/ h! m; {0 [2 R         break;
2 @5 `0 m' H/ s6 Q8 [       }
1 n: ?, i( |+ F! L- ^9 \# ]       case 2:
8 s# R/ B4 h' A, L% D4 M       { : x7 j* Z: |8 R  g* r. q2 O" `
          printf("%4c", *(char*)(p->data)); ; {5 d. L* U' B
          break; ; E8 l+ Y/ Q; M& k% R, J5 w+ W
       } 4 c: J- @: }' z6 O: V; I! S5 i
       case 3: 3 \' o& L2 P6 ?; p; m! H
       { 0 {/ |- K. K+ M2 ^, m
          printf("%s ", (char*)(p->data));
( L  J( H- X' z0 c! r& U9 T2 ]          break; 6 J8 Z% T* ]$ b. r
       }
9 A0 B) L2 V3 H' k& i$ p$ O7 Z9 d7 Q    }
  \" H' U5 C# N' B    p = p->next;
  P7 Q" A: B$ k' t8 Q' V. F  } 0 ]7 P5 Q) \* q/ U6 ~& K# W' r  W, [
    printf("/n");
4 s* \9 Q$ y9 N6 d- } }
+ A$ ^# J7 M& i- @( H回页首
5 e% }  J: f1 `& f$ O
, {" e  _# }6 d8 x& o. A小结0 B: H% x  N7 P3 y# k6 V  o
' [. J6 ?- Y. u* R
通过上面这个链表的小例子,大家可能已经看到了 C 语言的灵活性。这段代码虽然短并且功能简单,但是已经实现了多态性。这篇文章的本意并不在于想要告诉大家用 C 实现多态的方法,而多态的含义也无疑是更加广泛的。这篇文章的初衷其实是基于这样一点认识:面向对象是一种程序设计思想,而 C 语言则是一种编程语言。也许它并不是专门为了面向对象编程而设计,但是这绝不意味着它不能实现面向对象的程序设计。当然以上所展示的这几个操作,如果是用别的编程语言,可能只要寥寥几行就可以完成,但是 C 语言想告诉我们的是:也许我不擅长,但是并不意味着我做不到。
发表于 2016-6-20 16:49 | 显示全部楼层
C语言也就因为有这灵活性,所以才可以开发linux内核这种大型项目吧。。。
回复

使用道具 举报

 楼主| 发表于 2016-6-22 12:05 | 显示全部楼层
补充一下多态的应用方法- n' B+ W: O8 A& J3 U0 [5 k3 u

* I/ {  ^$ i/ i: m- Y: b. C, g多态,对象根据所接收的消息而做出动作。同一消息为不同的对象接受时可产生完全不同的行动,这种现象称为多态性。利用多态性用户可发送一个通用的信息,而将所有的实现细节都留给接受消息的对象自行决定,如是,同一消息即可调用不同的方法。例如:RT-Thread系统中的设备:抽象设备具备接口统一的读写接口。串口是设备的一种,也应支持设备的读写。但串口的读写操作是串口所特有的,不应和其他设备操作完全相同,例如操作串口的操作不应应用于SD卡设备中。多态性的实现受到继承性的支持,利用类继承的层次关系,把具有通用功能的协议存放在类层次中尽可能高的地方,而将实现这一功能的不同方法置于较低层次,这样,在这些低层次上生成的对象就能给通用消息以不同的响应。RT-Thread对象模型采用结构封装中使用函数指针的形式达到面向对象中多态的效果。具体代码片段如下:/ d- {1 K5 m" c
#include <stdio.h>7 c% v; W9 }+ H; c# c" f
#include <stdlib.h>
1 ^) s5 l* W9 y1 p3 r1 i//抽象父类/ C! n& N' e$ ~$ v& v
struct parent_class
5 }# s( k# f3 p7 [{
+ G( C  X$ D- {7 Z& F& o1 d, o5 l7 B. [        int a;
6 ~8 _' F1 _  C) E  @1 `        void (*vfunc)(struct parent_class *,int a);
* q# C' _* O# F4 ~+ I) V2 y& f0 a};
9 d; F( M  {8 S. O) s9 x//抽象父类的方法
7 r+ j5 q+ C0 A, m+ Ovoid parent_class_vfunc(struct parent_class *parent_self_ptr,int a)
: I  U2 _. j! m( q% d& x3 O{8 _$ K% X  @' d2 ]+ k# o
        //调用对象自己本身的虚拟函数
8 E9 k  w5 _, o5 q( ?! R* P7 T# E        printf("parent_class_vfunc\n");  o8 s8 [( H( u2 h) q1 I
        parent_self_ptr->vfunc(parent_self_ptr,a);
4 z1 V( a$ l* a}8 v( q8 N$ t0 B' G& u
//子类继承parent_class
% ?2 a* @1 x9 ?4 @! E* Ystruct child_class1" s2 I( ?% C; f$ x* ]/ L4 L* ~
{
$ ?% V6 j0 n! H% {        struct parent_class parent;# w9 Q' ]! K5 `( f1 v, ?
        int b;7 O8 X: z4 ^6 p
};5 A/ _7 h+ N: R( A) w  o0 W
//子类虚拟函数实现
3 u3 T$ I  j6 O9 `' r- [void child1_class_vfunc(struct child_class1 *child1_self_ptr,int a)
/ Q% o- X$ J- F" d6 v{% w1 _/ ]  O5 d/ R  G( [
        printf("child1_class_vfunc\n");. E3 r4 k/ _& Z3 D# J
        child1_self_ptr->b = a + 10;
. \6 p. r+ [3 v: n}
, Y/ Y+ F7 f* Z- R//子类构造函数3 F3 C. z1 V9 }9 J$ w2 k! h
void child1_class_init(struct child_class1 *child1_self_ptr)
1 b! y9 h" a& O) I- h{7 y' l1 }4 N5 Y) P
        struct parent_class *parent;# U* D( s' @# i  v# X8 a2 p
        //获取子类的父类指针
. j8 r0 t# q' V( W5 C' q        parent = (struct parent_class *)child1_self_ptr;
" d  {8 j6 z/ G        //设置子类的虚函数  F9 C, b! F3 k2 S
        parent->vfunc = child1_class_vfunc;3 \0 d8 l0 D$ b* _9 K7 C
}  \5 X. M% F. ?5 H& W/ Z* p
//子类继承parent_class0 I  i/ X3 R5 C
struct child_class2
6 c# Y" t5 r* B3 D4 M6 f{4 x* U# V- l6 `6 K  P
        struct parent_class parent;% b: y9 ]$ O# p7 W( x1 h
        int b;6 y  U* f$ W/ [. M, t
};
% ?2 w( l" ~+ K* }8 B//子类虚拟函数实现  q. e1 K# T: [" F8 J
void child2_class_vfunc(struct child_class2 *child2_self_ptr,int a)
  e* v2 i% F/ F7 v1 M8 r0 Q{
) O% H" Q9 p5 Y; ]1 ]; \        
0 y8 `; \* f$ r1 ~$ g        printf("child2_class_vfunc\n");6 d' @- V  `* n. C2 ~& Y3 u2 R( F
        child2_self_ptr->b = a + 10;
# o: M0 s2 t. I* ?0 N/ a/ x9 \# H}
" k5 Y2 s7 ?9 a! w) n9 p//子类构造函数
/ g# d7 A; E0 H1 \  T. [void child2_class_init(struct child_class2 *child2_self_ptr)8 z2 n9 i& c* I2 V& g+ f
{
7 W& U+ l; W4 X0 _/ m2 s& p        struct parent_class *parent;
9 |$ g1 ?# B3 P        //获取子类的父类指针
/ s; j3 z. L) Y3 S) h" i& ^$ ~( G, p        parent = (struct parent_class *)child2_self_ptr;
: I3 }0 Q/ ^% s, q# p* q1 J        //设置子类的虚函数
6 v; t. @3 `0 {& i        parent->vfunc = child2_class_vfunc;
0 ^& b# q& T% A4 r}" z0 W8 q! j. i8 C* x
int main()0 m& P/ B. _2 T+ _
{
& Z( Y+ m+ V3 S* c        struct child_class1 child1,*child1_ptr;
4 t7 Y! [$ ?. X1 L% Z" |% ~! j/ ^7 s        struct child_class2 child2,*child2_ptr;; t2 |: U) F" w# h" H5 h+ u$ a# t
        struct parent_class *parent_ptr;0 Y) U" z, Y- c
        //child1初始化" H2 ]4 j( s3 G' S# q! N& a
        child1_ptr = &child1;. e3 H% z: H( B5 ?5 T; N
        parent_ptr = (struct parent_class *)child1_ptr;  B5 a0 p- p: O5 }" h
        child1_ptr->b = 10;
: H% v  l3 w; F- u9 o        child1_class_init(child1_ptr);
- b. J. z3 F$ v. W: s        parent_ptr->vfunc((struct parent_class *)child1_ptr,20);
1 E& x, {" E5 l/ [8 E2 u" Z' H4 d8 J( I: n: R
        //child2初始化
* e0 i, a5 u+ g' O       child2_ptr = &child2;
- [' {+ k% z8 A- d       parent_ptr = (struct parent_class *)child2_ptr;
, B% T5 ^+ b3 U       child2_ptr->b = 20;: E! \9 e+ |3 t+ @' d2 B
       child2_class_init(child2_ptr);
9 R" a7 ]" b) H9 |  ~6 r9 ~: d7 X       parent_ptr->vfunc((struct parent_class *)child2_ptr, 20);
, P7 B4 I1 y6 d$ D4 L/ l       return 0;7 s9 k- c5 d3 A; o% J
}; y# q& O9 X' a
执行结果:
( I7 {& j: D% }  ?0 B- uchild1_class_vfunc
- m$ R7 P; h6 ^; C7 H, d; D& kchild2_class_vfunc
回复

使用道具 举报

 楼主| 发表于 2016-6-22 12:11 | 显示全部楼层
zhixiaoyuhong 发表于 2016-6-20 16:49' p$ _9 \9 \) w. p% T5 s
C语言也就因为有这灵活性,所以才可以开发linux内核这种大型项目吧。。。
( X3 y6 _; {% r* |  c4 Z
C语言的确有它的灵活性,搞中小系统相当的不错,但不足就是它只是一个过程语言这就是它的缺点,所以想尽所有的办法来将这个缺点补上,所以就有现在这种BT的方法。
+ w, H' p2 a( m5 J
回复

使用道具 举报

本版积分规则

QQ|一淘宝店|手机版|商店|一乐电子 ( 粤ICP备09076165号 ) 公安备案粤公网安备 44522102000183号

GMT+8, 2026-1-11 16:59 , Processed in 0.043195 second(s), 21 queries , Gzip On.

Powered by Discuz! X3.5

© 2001-2025 Discuz! Team.

快速回复 返回顶部 返回列表