/
vtest.cc
99 lines (92 loc) · 1.66 KB
/
vtest.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
#include <stdio.h>
class abstract;
typedef const char *(abstract::*M_ref)(int);
typedef const char *(*work_pfn)(abstract *a);
// forward decls
const char *do_work(abstract *a);
const char *do_work2(abstract *a);
class abstract
{
public:
abstract()
: m_worker(do_work),
bit_1(0),
bit_2(1)
{}
virtual ~abstract() = default;
virtual void dump(FILE *fp)
{
fprintf(fp, "anstract");
}
virtual const char *get_name()
{
return "abstract";
}
virtual const char *set_field(int i)
{
bit_1 = i & 1;
return get_name();
}
virtual const char *set_field2(int i)
{
bit_1 ^= i & 1;
return get_name();
}
virtual M_ref get_ref()
{
return &abstract::set_field;
}
virtual M_ref get_ref2()
{
return bit_1 ? &abstract::set_field : &abstract::set_field2;
}
// fields
int bit_1: 1;
int bit_2: 2;
work_pfn m_worker;
};
class derived: public abstract
{
public:
derived(): abstract()
{
m_worker = do_work2;
bit_1 = 1;
}
virtual ~derived() = default;
virtual void dump(FILE *fp)
{
fprintf(fp, "derived");
}
virtual const char *get_name()
{
return "derived";
}
virtual const char *set_field(int i)
{
bit_2 = i & 3;
return get_name();
}
};
const char *do_work(abstract *a)
{
a->dump(stdout);
return a->get_name();
}
const char *do_work2(abstract *a)
{
a->bit_1 = 0;
return a->get_name();
}
int main(int argc, char **argv)
{
abstract *a;
if ( argc > 1 )
a = new derived();
else
a = new abstract();
a->m_worker(a);
auto iref = a->get_ref2();
printf("result %s\n", (a->*iref)(argc));
delete a;
}