<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>潘锦的空间 &#187; PHP扩展开发</title>
	<atom:link href="https://www.phppan.com/tag/php%e6%89%a9%e5%b1%95%e5%bc%80%e5%8f%91/feed/" rel="self" type="application/rss+xml" />
	<link>https://www.phppan.com</link>
	<description>SaaS SaaS架构 团队管理 技术管理 技术架构 PHP 内核 扩展 项目管理</description>
	<lastBuildDate>Sat, 23 May 2026 03:04:48 +0000</lastBuildDate>
	<language>zh-CN</language>
		<sy:updatePeriod>hourly</sy:updatePeriod>
		<sy:updateFrequency>1</sy:updateFrequency>
	<generator>https://wordpress.org/?v=3.9.40</generator>
	<item>
		<title>PHP扩展开发：简单类实现</title>
		<link>https://www.phppan.com/2010/11/php-extension-1-simple-class/</link>
		<comments>https://www.phppan.com/2010/11/php-extension-1-simple-class/#comments</comments>
		<pubDate>Wed, 03 Nov 2010 01:52:16 +0000</pubDate>
		<dc:creator><![CDATA[admin]]></dc:creator>
				<category><![CDATA[PHP]]></category>
		<category><![CDATA[PHP扩展]]></category>
		<category><![CDATA[PHP扩展开发]]></category>

		<guid isPermaLink="false">http://www.phppan.com/?p=1079</guid>
		<description><![CDATA[PHP扩展开发：简单类实现 话说之前有写过如何在vs2008下开发PHP扩展，今天我们来实现一个简单类。 对于 [&#8230;]]]></description>
				<content:encoded><![CDATA[<p>PHP扩展开发：简单类实现<br />
话说之前有写过如何在vs2008下开发PHP扩展，今天我们来实现一个简单类。<br />
对于一个类，一般包括类定义，声明成员变量，声明成员函数，定义类常量，<br />
对于类继承，重载，接口实现等不在这里说明。<br />
首先我们看下这个简单类的PHP实现：</p>

<div class="wp_syntax"><table><tr><td class="line_numbers"><pre>1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
</pre></td><td class="code"><pre class="php" style="font-family:monospace;"><span style="color: #000000; font-weight: bold;">&lt;?php</span>
<span style="color: #009933; font-style: italic;">/**
 * 简单类示例的PHP实现  2010-11-03 sz
 * @author phppan.p#gmail.com  http://www.phppan.com                               
 * 哥学社成员（http://www.blog-brother.com/）
 * @package test
 */</span>
<span style="color: #000000; font-weight: bold;">class</span> Phppan <span style="color: #009900;">&#123;</span>
&nbsp;
    <span style="color: #000000; font-weight: bold;">private</span> <span style="color: #000088;">$_name</span><span style="color: #339933;">;</span>
&nbsp;
    <span style="color: #000000; font-weight: bold;">CONST</span> URL <span style="color: #339933;">=</span> <span style="color: #0000ff;">'http://www.phppan.com'</span><span style="color: #339933;">;</span>
&nbsp;
    <span style="color: #000000; font-weight: bold;">public</span> <span style="color: #000000; font-weight: bold;">function</span> __construct<span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span> <span style="color: #009900;">&#123;</span>
    <span style="color: #009900;">&#125;</span>
&nbsp;
    <span style="color: #000000; font-weight: bold;">public</span> <span style="color: #000000; font-weight: bold;">function</span> getName<span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span> <span style="color: #009900;">&#123;</span>
        <span style="color: #b1b100;">return</span> <span style="color: #000088;">$this</span><span style="color: #339933;">-&gt;</span>_name<span style="color: #339933;">;</span>
    <span style="color: #009900;">&#125;</span>
&nbsp;
    <span style="color: #000000; font-weight: bold;">public</span> <span style="color: #000000; font-weight: bold;">function</span> setName<span style="color: #009900;">&#40;</span><span style="color: #000088;">$name</span><span style="color: #009900;">&#41;</span> <span style="color: #009900;">&#123;</span>
        <span style="color: #000088;">$this</span><span style="color: #339933;">-&gt;</span>_name <span style="color: #339933;">=</span> <span style="color: #000088;">$name</span><span style="color: #339933;">;</span>
    <span style="color: #009900;">&#125;</span>
<span style="color: #009900;">&#125;</span></pre></td></tr></table></div>

<p>我们同样使用在之前创建的martin扩展<br />
下面说明其实现过程：<br />
1、声明类的成员函数<br />
在php_martin.h文件中声明类的成员函数，如下所示代码：</p>

<div class="wp_syntax"><table><tr><td class="line_numbers"><pre>1
2
3
4
5
</pre></td><td class="code"><pre class="c" style="font-family:monospace;">&nbsp;
PHP_METHOD<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> __construct<span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>                                      
PHP_METHOD<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> __destruct<span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
PHP_METHOD<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> getName<span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
PHP_METHOD<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> setName<span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span></pre></td></tr></table></div>

<p>2、定义类的成员函数<br />
在martin.c文件中给出这4个函数的实现</p>

<div class="wp_syntax"><table><tr><td class="line_numbers"><pre>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
</pre></td><td class="code"><pre class="c" style="font-family:monospace;">&nbsp;
&nbsp;
<span style="color: #808080; font-style: italic;">/** {{{ 
*/</span>
PHP_METHOD<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> __construct<span style="color: #009900;">&#41;</span> <span style="color: #009900;">&#123;</span>
&nbsp;
<span style="color: #009900;">&#125;</span>
<span style="color: #808080; font-style: italic;">/* }}} */</span>
&nbsp;
<span style="color: #808080; font-style: italic;">/** {{{ 
*/</span>
PHP_METHOD<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> __destruct<span style="color: #009900;">&#41;</span> <span style="color: #009900;">&#123;</span>
<span style="color: #009900;">&#125;</span>
<span style="color: #808080; font-style: italic;">/* }}} */</span>
&nbsp;
<span style="color: #808080; font-style: italic;">/** {{{ 
*/</span>
PHP_METHOD<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> getName<span style="color: #009900;">&#41;</span> <span style="color: #009900;">&#123;</span>
	zval <span style="color: #339933;">*</span>self<span style="color: #339933;">,</span> <span style="color: #339933;">*</span>name<span style="color: #339933;">;</span>
&nbsp;
	self <span style="color: #339933;">=</span> getThis<span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
&nbsp;
	name <span style="color: #339933;">=</span> zend_read_property<span style="color: #009900;">&#40;</span>Z_OBJCE_P<span style="color: #009900;">&#40;</span>self<span style="color: #009900;">&#41;</span><span style="color: #339933;">,</span> self<span style="color: #339933;">,</span> ZEND_STRL<span style="color: #009900;">&#40;</span><span style="color: #ff0000;">&quot;_name&quot;</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">,</span> <span style="color: #0000dd;">0</span> TSRMLS_CC<span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
&nbsp;
	RETURN_STRING<span style="color: #009900;">&#40;</span>Z_STRVAL_P<span style="color: #009900;">&#40;</span>name<span style="color: #009900;">&#41;</span><span style="color: #339933;">,</span> <span style="color: #0000dd;">0</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
<span style="color: #009900;">&#125;</span>
<span style="color: #808080; font-style: italic;">/* }}} */</span>
&nbsp;
<span style="color: #808080; font-style: italic;">/** {{{ 
*/</span>
PHP_METHOD<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> setName<span style="color: #009900;">&#41;</span> <span style="color: #009900;">&#123;</span>
	<span style="color: #993333;">char</span> <span style="color: #339933;">*</span>arg <span style="color: #339933;">=</span> NULL<span style="color: #339933;">;</span>
	<span style="color: #993333;">int</span> arg_len<span style="color: #339933;">;</span>
	zval <span style="color: #339933;">*</span>value<span style="color: #339933;">,</span> <span style="color: #339933;">*</span>self<span style="color: #339933;">;</span>
&nbsp;
	<span style="color: #b1b100;">if</span> <span style="color: #009900;">&#40;</span>zend_parse_parameters<span style="color: #009900;">&#40;</span>ZEND_NUM_ARGS<span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span> TSRMLS_CC<span style="color: #339933;">,</span> <span style="color: #ff0000;">&quot;s&quot;</span><span style="color: #339933;">,</span> <span style="color: #339933;">&amp;</span>arg<span style="color: #339933;">,</span> <span style="color: #339933;">&amp;</span>arg_len<span style="color: #009900;">&#41;</span> <span style="color: #339933;">==</span> FAILURE<span style="color: #009900;">&#41;</span> <span style="color: #009900;">&#123;</span>
		WRONG_PARAM_COUNT<span style="color: #339933;">;</span>
	<span style="color: #009900;">&#125;</span>
&nbsp;
	self <span style="color: #339933;">=</span> getThis<span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
&nbsp;
	MAKE_STD_ZVAL<span style="color: #009900;">&#40;</span>value<span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
	ZVAL_STRING<span style="color: #009900;">&#40;</span>value<span style="color: #339933;">,</span> arg<span style="color: #339933;">,</span> arg_len<span style="color: #339933;">,</span> <span style="color: #0000dd;">0</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
&nbsp;
	SEPARATE_ZVAL_TO_MAKE_IS_REF<span style="color: #009900;">&#40;</span><span style="color: #339933;">&amp;</span>value<span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
	zend_update_property<span style="color: #009900;">&#40;</span>Z_OBJCE_P<span style="color: #009900;">&#40;</span>self<span style="color: #009900;">&#41;</span><span style="color: #339933;">,</span> self<span style="color: #339933;">,</span> ZEND_STRL<span style="color: #009900;">&#40;</span><span style="color: #ff0000;">&quot;_name&quot;</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">,</span> value TSRMLS_CC<span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
&nbsp;
	RETURN_TRUE<span style="color: #339933;">;</span>
<span style="color: #009900;">&#125;</span>
<span style="color: #808080; font-style: italic;">/* }}} */</span></pre></td></tr></table></div>

<p>3、注册这些函数</p>

<div class="wp_syntax"><table><tr><td class="line_numbers"><pre>1
2
3
4
5
6
7
8
9
10
11
12
13
14
</pre></td><td class="code"><pre class="c" style="font-family:monospace;">&nbsp;
<span style="color: #808080; font-style: italic;">/* {{{ martin_functions[]
 *
 * Every user visible function must have an entry in martin_functions[].
 */</span>
<span style="color: #993333;">const</span> zend_function_entry martin_functions<span style="color: #009900;">&#91;</span><span style="color: #009900;">&#93;</span> <span style="color: #339933;">=</span> <span style="color: #009900;">&#123;</span>
	PHP_FE<span style="color: #009900;">&#40;</span>martin_echo<span style="color: #339933;">,</span>	NULL<span style="color: #009900;">&#41;</span>		
	PHP_ME<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> __construct<span style="color: #339933;">,</span> 	NULL<span style="color: #339933;">,</span> ZEND_ACC_PUBLIC<span style="color: #339933;">|</span>ZEND_ACC_CTOR<span style="color: #009900;">&#41;</span> 
	PHP_ME<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> __destruct<span style="color: #339933;">,</span>  	NULL<span style="color: #339933;">,</span> ZEND_ACC_PUBLIC<span style="color: #339933;">|</span>ZEND_ACC_DTOR<span style="color: #009900;">&#41;</span> 
	PHP_ME<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> getName<span style="color: #339933;">,</span> 	 	 	NULL<span style="color: #339933;">,</span> ZEND_ACC_PUBLIC<span style="color: #009900;">&#41;</span> 
	PHP_ME<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> setName<span style="color: #339933;">,</span> 	 	 	setName_args<span style="color: #339933;">,</span> ZEND_ACC_PUBLIC<span style="color: #009900;">&#41;</span> 
	<span style="color: #009900;">&#123;</span>NULL<span style="color: #339933;">,</span> NULL<span style="color: #339933;">,</span> NULL<span style="color: #009900;">&#125;</span>	<span style="color: #808080; font-style: italic;">/* Must be the last line in martin_functions[] */</span>
<span style="color: #009900;">&#125;</span><span style="color: #339933;">;</span>
<span style="color: #808080; font-style: italic;">/* }}} */</span></pre></td></tr></table></div>

<p>4、在扩展模块初始化时初始化类，并声明成员变量和常量</p>

<div class="wp_syntax"><table><tr><td class="line_numbers"><pre>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
</pre></td><td class="code"><pre class="c" style="font-family:monospace;">zend_class_entry <span style="color: #339933;">*</span>phppan_ce<span style="color: #339933;">;</span>
&nbsp;
<span style="color: #808080; font-style: italic;">/* 类方法的参数 */</span>
ZEND_BEGIN_ARG_INFO<span style="color: #009900;">&#40;</span>setName_args<span style="color: #339933;">,</span> <span style="color: #0000dd;">0</span><span style="color: #009900;">&#41;</span>
	ZEND_ARG_INFO<span style="color: #009900;">&#40;</span><span style="color: #0000dd;">0</span><span style="color: #339933;">,</span> name<span style="color: #009900;">&#41;</span>
ZEND_END_ARG_INFO<span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span>
&nbsp;
<span style="color: #808080; font-style: italic;">/* {{{ PHP_MINIT_FUNCTION
 */</span>
PHP_MINIT_FUNCTION<span style="color: #009900;">&#40;</span>martin<span style="color: #009900;">&#41;</span>
<span style="color: #009900;">&#123;</span>
	<span style="color: #808080; font-style: italic;">/* If you have INI entries, uncomment these lines 
	REGISTER_INI_ENTRIES();
	*/</span>
	zend_class_entry phppan<span style="color: #339933;">;</span>
&nbsp;
	INIT_CLASS_ENTRY<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> <span style="color: #ff0000;">&quot;Phppan&quot;</span><span style="color: #339933;">,</span> martin_functions<span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
	phppan_ce <span style="color: #339933;">=</span> zend_register_internal_class_ex<span style="color: #009900;">&#40;</span><span style="color: #339933;">&amp;</span>phppan<span style="color: #339933;">,</span> NULL<span style="color: #339933;">,</span> NULL TSRMLS_CC<span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
&nbsp;
&nbsp;
	<span style="color: #808080; font-style: italic;">/* 声明常量URL */</span>
	zend_declare_class_constant_stringl<span style="color: #009900;">&#40;</span>phppan_ce<span style="color: #339933;">,</span> ZEND_STRL<span style="color: #009900;">&#40;</span><span style="color: #ff0000;">&quot;URL&quot;</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">,</span> ZEND_STRL<span style="color: #009900;">&#40;</span><span style="color: #ff0000;">&quot;http://www.phppan.com&quot;</span><span style="color: #009900;">&#41;</span> TSRMLS_CC<span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
&nbsp;
	<span style="color: #808080; font-style: italic;">/* 声明私有成员变量 _name  */</span>
	zend_declare_property_null<span style="color: #009900;">&#40;</span>phppan_ce<span style="color: #339933;">,</span> ZEND_STRL<span style="color: #009900;">&#40;</span><span style="color: #ff0000;">&quot;_name&quot;</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">,</span> ZEND_ACC_PRIVATE TSRMLS_CC<span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
&nbsp;
&nbsp;
	<span style="color: #b1b100;">return</span> SUCCESS<span style="color: #339933;">;</span>
<span style="color: #009900;">&#125;</span>
<span style="color: #808080; font-style: italic;">/* }}} */</span></pre></td></tr></table></div>

<p>以上为所有代码<br />
在这些代码里面有一些东西需要说明一下：<br />
1、类方法的定义与php的单个函数的定义一样，使用zend_function_entry结构数组，不同的是单个方法使用PHP_FE宏，<br />
类方法的定义使用PHP_ME宏，在h文件中使用ZEND_METHOD声明，普通的函数使用ZEND_FUNCTION声明。<br />
PHP_ME宏<br />
定义在php.h中<br />
#define PHP_ME          ZEND_ME<br />
#define ZEND_ME(classname, name, arg_info, flags)        ZEND_FENTRY(name, ZEND_MN(classname##_##name), arg_info, flags)<br />
2、在注册类时把该结构体作为参数交给相关的类注册方法即可<br />
ZEND_BEGIN_ARG_INFO(setName_args, 0)<br />
	ZEND_ARG_INFO(0, name)<br />
ZEND_END_ARG_INFO()<br />
3、取this变量使用getThis();<br />
4、使用zend_read_property读取类成员变量，返回的是zval 指针类型<br />
5、使用zend_update_property更新类成员变量。<br />
6、初始化类使用INIT_CLASS_ENTRY宏<br />
7、注册类使用zend_register_internal_class_ex函数</p>
<p>function registration failed duplicate name.问题的解决方法：<br />
这个由于我们在写这个简单类时偷了一下懒，将martin_functions作为模块的函数也将它作为了类的方法。<br />
解决方法是替换上面的martin_functions数组，增加phppan_functions，并且在注册时使用phppan_functions，在模块的functions字段使用martin_functions</p>

<div class="wp_syntax"><table><tr><td class="code"><pre class="c" style="font-family:monospace;"><span style="color: #808080; font-style: italic;">/* {{{ martin_functions[]
 *
 * Every user visible function must have an entry in martin_functions[].
 */</span>
<span style="color: #993333;">const</span> zend_function_entry martin_functions<span style="color: #009900;">&#91;</span><span style="color: #009900;">&#93;</span> <span style="color: #339933;">=</span> <span style="color: #009900;">&#123;</span>
    <span style="color: #009900;">&#123;</span>NULL<span style="color: #339933;">,</span> NULL<span style="color: #339933;">,</span> NULL<span style="color: #009900;">&#125;</span>  <span style="color: #808080; font-style: italic;">/* Must be the last line in martin_functions[] */</span>
<span style="color: #009900;">&#125;</span><span style="color: #339933;">;</span>
&nbsp;
<span style="color: #993333;">const</span> zend_function_entry phppan_functions<span style="color: #009900;">&#91;</span><span style="color: #009900;">&#93;</span> <span style="color: #339933;">=</span> <span style="color: #009900;">&#123;</span>
    PHP_ME<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> __construct<span style="color: #339933;">,</span>     NULL<span style="color: #339933;">,</span> ZEND_ACC_PUBLIC<span style="color: #339933;">|</span>ZEND_ACC_CTOR<span style="color: #009900;">&#41;</span>
    PHP_ME<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> __destruct<span style="color: #339933;">,</span>      NULL<span style="color: #339933;">,</span> ZEND_ACC_PUBLIC<span style="color: #339933;">|</span>ZEND_ACC_DTOR<span style="color: #009900;">&#41;</span>
    PHP_ME<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> getName<span style="color: #339933;">,</span>             NULL<span style="color: #339933;">,</span> ZEND_ACC_PUBLIC<span style="color: #009900;">&#41;</span>
    PHP_ME<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> setName<span style="color: #339933;">,</span>             setName_args<span style="color: #339933;">,</span> ZEND_ACC_PUBLIC<span style="color: #009900;">&#41;</span>
    <span style="color: #009900;">&#123;</span>NULL<span style="color: #339933;">,</span> NULL<span style="color: #339933;">,</span> NULL<span style="color: #009900;">&#125;</span>  <span style="color: #808080; font-style: italic;">/* Must be the last line in martin_functions[] */</span>
<span style="color: #009900;">&#125;</span><span style="color: #339933;">;</span>
<span style="color: #808080; font-style: italic;">/* }}} */</span>
&nbsp;
INIT_CLASS_ENTRY<span style="color: #009900;">&#40;</span>phppan<span style="color: #339933;">,</span> <span style="color: #ff0000;">&quot;Phppan&quot;</span><span style="color: #339933;">,</span> phppan_functions<span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span></pre></td></tr></table></div>

]]></content:encoded>
			<wfw:commentRss>https://www.phppan.com/2010/11/php-extension-1-simple-class/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss>
