Jadual Kandungan
[转]自己写PHP扩展之创建一个类,php一个类
Rumah php教程 php手册 [转]自己写PHP扩展之创建一个类,php一个类

[转]自己写PHP扩展之创建一个类,php一个类

Jun 13, 2016 am 08:40 AM
panggilan fungsi

[转]自己写PHP扩展之创建一个类,php一个类

原文:http://www.imsiren.com/archives/572

比如我们要创建一个类..PHP代码如下

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 class Person {     public $name;     public $age;     public function __construct() {         echo "construct is running! ";     }     public function __destruct() {         echo " destruct is running!";     }     public function getproperty($key) {         echo $this->$key;     }     public function setproperty($key,$val) {         $this->$key = $val;     } }

用PHP来做,很简单..
那么用PHP扩展来写该怎么做?
OK.
1.在php_siren.h里面声明类

1 2 3 4 PHP_METHOD(Person,__construct); PHP_METHOD(Person,__destruct); PHP_METHOD(Person,setproperty); PHP_METHOD(Person,getproperty);

PHP_METHOD宏.
PHP_METHOD 等于ZEND_METHOD
这个宏接受两个参数,第一个是类名,第二个是类的方法

1 2 3 4 #define ZEND_METHOD(classname, name)    ZEND_NAMED_FUNCTION(ZEND_MN(classname##_##name)) #define INTERNAL_FUNCTION_PARAMETERS int ht, zval *return_value, zval **return_value_ptr, zval *this_ptr, int return_v    alue_used TSRMLS_DC //最后等于 void name(int ht, zval *return_value, zval **return_value_ptr, zval *this_ptr, int return_v    alue_used TSRMLS_DC )

这个宏是用来声明我们的方法…
2.设置接收的参数
我们的方法如果需要接受参数.那么就要执行

1 2 3 ZEND_BEGIN_ARG_INFO_EX(arg_person_info,0,0,2)         ZEND_ARG_INFO(0,name) ZEND_END_ARG_INFO()

详细讲这几个宏之前先看看zend_arg_info

1 2 3 4 5 6 7 8 9 10 11 typedef struct _zend_arg_info {         const char *name; //参数名称         zend_uint name_len;//长度         const char *class_name;  //所属类名         zend_uint class_name_len;  //类名长度         zend_bool array_type_hint;         zend_bool allow_null; //允许为空         zend_bool pass_by_reference;  //引用传值         zend_bool return_reference;   //引用返回         int required_num_args;   //参数个数 } zend_arg_info;

ZEND_BEGIN_ARG_INFO_EX定义在Zend/zend_API.h

1 2 3 #define ZEND_BEGIN_ARG_INFO_EX(name, pass_rest_by_reference, return_reference, required_num_args)       \         static const zend_arg_info name[] = {                                                                                                                                           \                 { NULL, 0, NULL, 0, 0, 0, pass_rest_by_reference, return_reference, required_num_args },

很明显 声明一个zend_arg_info的数组name,然后初始化结构体的值
ZEND_ARG_INFO(0,name)的定义如下

1 #define ZEND_ARG_INFO(pass_by_ref, name)  { #name, sizeof(#name)-1, NULL, 0, 0, 0, pass_by_ref, 0, 0 },

这三个宏 执行代码 等于

1 2 3 static const zend_arg_info name[] = {                                                                                                                                                    { NULL, 0, NULL, 0, 0, 0, pass_rest_by_reference, return_reference, required_num_args }, { #name, sizeof(#name)-1, NULL, 0, 0, 0, pass_by_ref, 0, 0 }, };

3.创建zend_function_entry结构数组

1 2 3 4 5 6 7 const zend_function_entry person_functions[]={         PHP_ME(Person,__construct,NULL,ZEND_ACC_PUBLIC|ZEND_ACC_CTOR)         PHP_ME(Person,__destruct,NULL,ZEND_ACC_PUBLIC|ZEND_ACC_DTOR)         PHP_ME(Person,getproperty,arg_person_info,ZEND_ACC_PUBLIC)         PHP_ME(Person,setproperty,arg_person_info,ZEND_ACC_PUBLIC)         PHP_FE_END };

zend_function_entry定义如下

1 2 3 4 5 6 7 typedef struct _zend_function_entry {         const char *fname; //函数名称         void (*handler)(INTERNAL_FUNCTION_PARAMETERS);         const struct _zend_arg_info *arg_info;//参数         zend_uint num_args;//参数个数         zend_uint flags;//标示PUBLIC ?PRIVATE ?PROTECTED } zend_function_entry;

PHP_ME宏接收四个参数
1 类名,
2 方法名,
3 zend_arg_info 的参数列表,

ZEND_ACC_PUBLIC ZEND_ACC_PRIVATE ZEND_ACC_PROTECTED是我们类里面的三个访问权限
ZEND_ACC_CTOR标示构造函数
ZEND_ACC_DTOR标示析构函数
4.修改PHP_MINIT_FUNCTION
前面我们说过 PHP_MINIT_FUNCTION是在模块启动的时候执行的函数
首先创建一个全局指针 zend_class_entry *person_ce;
在PHP_MINIT_FUNCTION加入如下代码

1 2 3 4 zend_class_entry person; INIT_CLASS_ENTRY(person,"Person",person_functions); person_ce=zend_register_internal_class_ex(&person,NULL,NULL TSRMLS_CC); zend_declare_property_null(person_ce,ZEND_STRL("name"),ZEND_ACC_PUBLIC TSRMLS_CC);

1行创建一个zend_class_entry对象person.
zend_class_entry这个结构体前面也讲过 PHP内核研究之类的实现
2行初始化zend_class_entry 它执行了如下代码

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 {                                                                                                                       \         int _len = class_name_len;                                                              \         class_container.name = zend_strndup(class_name, _len);  \         class_container.name_length = _len;                                             \         class_container.builtin_functions = functions;                  \         class_container.constructor = NULL;                                             \         class_container.destructor = NULL;                                              \         class_container.clone = NULL;                                                   \         class_container.serialize = NULL;                                               \         class_container.unserialize = NULL;                                             \         class_container.create_object = NULL;                                   \         class_container.interface_gets_implemented = NULL;              \         class_container.get_static_method = NULL;                               \         class_container.__call = handle_fcall;                                  \         class_container.__callstatic = NULL;                                    \         class_container.__tostring = NULL;                                              \         class_container.__get = handle_propget;                                 \         class_container.__set = handle_propset;                                 \         class_container.__unset = handle_propunset;                             \         class_container.__isset = handle_propisset;                             \         class_container.serialize_func = NULL;                                  \         class_container.unserialize_func = NULL;                                \         class_container.serialize = NULL;                                               \         class_container.unserialize = NULL;                                             \         class_container.parent = NULL;                                                  \         class_container.num_interfaces = 0;                                             \         class_container.interfaces = NULL;                                              \         class_container.get_iterator = NULL;                                    \         class_container.iterator_funcs.funcs = NULL;                    \         class_container.module = NULL;                                                  \ }

可以对应文章>> PHP内核研究之类的实现来分析
zend_declare_property_null(person_ce,ZEND_STRL(“name”),ZEND_ACC_PUBLIC TSRMLS_CC);
创建一个值为NULL的属性
第一个参数是类名,第二个参数是 属性名称,第三个参数是属性名的长度,因为ZEND_STRL宏定义了长度,所以这里不用再传递长度.
第四个参数是属性的访问权限.
还有其他几个函数用来创建不同类型的属性

1 2 3 4 5 6 7 zend_declare_property_bool zend_declare_property_double zend_declare_property_ex zend_declare_property_long zend_declare_property_null zend_declare_property_string zend_declare_property_stringl

5.创建 php_siren.h头文件中的方法体

1 2 3 4 5 6 7 8 9 10 11 12 PHP_METHOD(Person,__construct){         php_printf("construct is running
"
); } PHP_METHOD(Person,__destruct){         php_printf("destruct is running
"
); } PHP_METHOD(Person,setproperty){   } PHP_METHOD(Person,getproperty){   }

6.最后make&& make install
编译我们的扩展,
重新启动apache.
$p=new Person();
?>
我们就能在浏览器里看到输出的内容

construct is running
destruct is running

这样 ..我们用扩展创建的一个基本类就完成了.

Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn

Alat AI Hot

Undresser.AI Undress

Undresser.AI Undress

Apl berkuasa AI untuk mencipta foto bogel yang realistik

AI Clothes Remover

AI Clothes Remover

Alat AI dalam talian untuk mengeluarkan pakaian daripada foto.

Undress AI Tool

Undress AI Tool

Gambar buka pakaian secara percuma

Clothoff.io

Clothoff.io

Penyingkiran pakaian AI

AI Hentai Generator

AI Hentai Generator

Menjana ai hentai secara percuma.

Artikel Panas

R.E.P.O. Kristal tenaga dijelaskan dan apa yang mereka lakukan (kristal kuning)
4 minggu yang lalu By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. Tetapan grafik terbaik
4 minggu yang lalu By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. Cara Memperbaiki Audio Jika anda tidak dapat mendengar sesiapa
4 minggu yang lalu By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. Arahan sembang dan cara menggunakannya
4 minggu yang lalu By 尊渡假赌尊渡假赌尊渡假赌

Alat panas

Notepad++7.3.1

Notepad++7.3.1

Editor kod yang mudah digunakan dan percuma

SublimeText3 versi Cina

SublimeText3 versi Cina

Versi Cina, sangat mudah digunakan

Hantar Studio 13.0.1

Hantar Studio 13.0.1

Persekitaran pembangunan bersepadu PHP yang berkuasa

Dreamweaver CS6

Dreamweaver CS6

Alat pembangunan web visual

SublimeText3 versi Mac

SublimeText3 versi Mac

Perisian penyuntingan kod peringkat Tuhan (SublimeText3)

Penalaan prestasi panggilan fungsi C++: kesan nilai lulus dan pulangan parameter Penalaan prestasi panggilan fungsi C++: kesan nilai lulus dan pulangan parameter May 04, 2024 pm 12:57 PM

Pengoptimuman prestasi panggilan fungsi C++ merangkumi dua aspek: strategi lulus parameter dan pengoptimuman jenis nilai pulangan. Dari segi lulus parameter, nilai lulus sesuai untuk objek kecil dan parameter tidak boleh diubah, manakala rujukan atau penunjuk lulus sesuai untuk objek besar dan parameter boleh diubah suai, dan penunjuk lulus adalah yang terpantas. Dari segi pengoptimuman nilai pulangan, nilai kecil boleh dikembalikan secara langsung, dan objek besar harus mengembalikan rujukan atau petunjuk. Memilih strategi yang sesuai boleh meningkatkan prestasi panggilan fungsi.

Bagaimana untuk memanggil fungsi dalam modul yang berbeza dalam C++? Bagaimana untuk memanggil fungsi dalam modul yang berbeza dalam C++? Apr 12, 2024 pm 03:54 PM

Memanggil fungsi merentas modul dalam C++: Isytihar fungsi: Isytiharkan fungsi yang akan dipanggil dalam fail pengepala modul sasaran. Laksanakan fungsi: Laksanakan fungsi dalam fail sumber. Memautkan modul: Gunakan pemaut untuk memautkan bersama modul yang mengandungi pengisytiharan dan pelaksanaan fungsi. Panggil fungsi: Sertakan fail pengepala modul sasaran dalam modul yang perlu dipanggil, dan kemudian panggil fungsi tersebut.

Teknologi refleksi panggilan fungsi C++: lulus parameter dan akses dinamik nilai pulangan Teknologi refleksi panggilan fungsi C++: lulus parameter dan akses dinamik nilai pulangan May 05, 2024 am 09:48 AM

Teknologi pantulan panggilan fungsi C++ membolehkan secara dinamik mendapatkan parameter fungsi dan maklumat nilai pulangan pada masa jalan. Gunakan ungkapan typeid(decltype(...)) dan decltype(...) untuk mendapatkan parameter dan maklumat jenis nilai kembali. Melalui refleksi, kami boleh memanggil fungsi secara dinamik dan memilih fungsi tertentu berdasarkan input masa jalan, membolehkan kod fleksibel dan berskala.

Terokai pelbagai cara untuk memanggil fungsi PHP Terokai pelbagai cara untuk memanggil fungsi PHP Apr 16, 2024 pm 02:03 PM

Terdapat lima cara untuk memanggil fungsi PHP: panggilan terus, panggilan melalui pembolehubah, fungsi tanpa nama, penunjuk fungsi dan refleksi. Dengan memilih kaedah yang paling sesuai dengan keadaan, anda boleh mengoptimumkan prestasi dan meningkatkan kesederhanaan kod.

Penjelasan terperinci tentang mekanisme panggilan fungsi C++ Penjelasan terperinci tentang mekanisme panggilan fungsi C++ Apr 11, 2024 pm 02:12 PM

Mekanisme panggilan fungsi dalam C++ melibatkan menghantar argumen kepada fungsi dan melaksanakan kodnya, mengembalikan hasilnya jika wujud. Terdapat dua cara untuk lulus parameter: lulus mengikut nilai (pengubahsuaian dibuat di dalam fungsi) dan lulus melalui rujukan (pengubahsuaian ditunjukkan dalam pemanggil). Dalam penghantaran nilai, pengubahsuaian nilai dalam fungsi tidak menjejaskan nilai asal (seperti printValue), manakala pengubahsuaian dalam hantaran rujukan mempengaruhi nilai asal (seperti printReference).

Ujian unit panggilan fungsi C++: pengesahan ketepatan parameter lulus dan nilai pulangan Ujian unit panggilan fungsi C++: pengesahan ketepatan parameter lulus dan nilai pulangan May 01, 2024 pm 02:54 PM

Apabila mengesahkan panggilan fungsi C++ dalam ujian unit, anda perlu mengesahkan dua perkara berikut: Lulus parameter: Gunakan penegasan untuk menyemak sama ada parameter sebenar sepadan dengan nilai yang dijangkakan. Nilai pulangan: Gunakan penegasan untuk menyemak sama ada nilai pulangan sebenar adalah sama dengan nilai yang dijangkakan.

Makro prapemproses panggilan fungsi C++: penggunaan lanjutan parameter lulus dan nilai pulangan Makro prapemproses panggilan fungsi C++: penggunaan lanjutan parameter lulus dan nilai pulangan May 04, 2024 pm 04:33 PM

Dalam C++, makro prapemproses boleh digunakan untuk memanggil fungsi, melibatkan langkah berikut: Lulus parameter: Parameter makro disertakan dalam kurungan dan dipisahkan dengan koma. Nilai pulangan: Gunakan parameter makro untuk menentukan nilai yang akan dikembalikan dan berikannya kepada pembolehubah. Kes praktikal: Dengan menggunakan pengoptimuman makro untuk mencari fungsi indeks nilai maksimum dalam tatasusunan, bilangan pengiraan dikurangkan dan kecekapan dipertingkatkan.

Penjelasan terperinci tentang panggilan fungsi C++: analisis mendalam mekanisme lulus parameter Penjelasan terperinci tentang panggilan fungsi C++: analisis mendalam mekanisme lulus parameter May 04, 2024 am 10:48 AM

Terdapat tiga mekanisme lulus parameter untuk panggilan fungsi C++: panggilan mengikut nilai (menyalin nilai parameter), panggilan melalui rujukan (melalui rujukan parameter, yang boleh mengubah suai pembolehubah asal), dan lulus penunjuk (melepasi penunjuk parameter). Mekanisme pemilihan perlu mempertimbangkan saiz parameter, sama ada pembolehubah asal perlu diubah suai, dan kecekapan.

See all articles