编写一个简单的扩展示例
使用php内置工具生成自定义扩展的框架。该工具在ext文件夹下面
./ext_skel --extname=wully
Ext_skel的主要参数(这里面只列举了最常用的两个)
--extname=module module is the name of your extension【必须有的】
生成模块的名称,会自动在ext内建立extname文件夹,最常用的
--proto=file file contains prototypes of functions to create【可选】
函数原型定义文件
官方说明地址:
http://www.php.net/manual/en/internals2.buildsys.skeleton.php
修正ext/wully/config.m4
修正后,运行:
./buildconf --force
./configure --enable-wully
Make & make install 及可以安装成功
可以通过运行 ext/wully/wully.php来校验是否安装成功
扩展的内部实现
依旧以生成的wully模块为例,这个模块的提供的外部函数在
const zend_function_entry wully_functions[]这里面陈列(wully.c),我们可以看到示例中定义的函数为
PHP_FE(confirm_wully_compiled, NULL)
这个模块的外部函数和模块在php请求各个阶段的调用逻辑是通过
wully_module_entry(zend_module_entry)结构传给php框架的
php扩展函数的实现逻辑
每一个扩展函数都是通过PHP_FUNCTION来实现的,例如我们的示例,通过PHP_FUNCTION(confirm_wully_compiled) 来实现的
内部如何获取php函数的参数
生成的源码函数为:
zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s", &arg, &arg_len)
这个是一个参数,且参数类型为字符串的情况
调用的原型为:
zend_parse_parameters(int num_args TSRMLS_DC, char *type_spec, …);
有两个参数我们要特别注意下:
第一个参数为该函数获取的参数个数
第三个参数为php函数传过来的参数类型,下表是一个对应关系图
后面的参数则是参数的值,及参数的大小,第二个参数对应于线程安全,这里面先不关心
参数类型对应表
类型指定符 | 对应的C类型 | 描述 |
l | long | 符号整数 |
d | double | 浮点数 |
s | char *, int | 二进制字符串,长度 |
b | zend_bool | 逻辑型(1或0) |
r | zval * | 资源(文件指针,数据库连接等) |
a | zval * | 联合数组 |
o | zval * | 任何类型的对象 |
O | zval * | 指定类型的对象。需要提供目标对象的类类型 |
z | zval * | 无任何操作的zval |
再看一个三个参数,一个为字符串,两个为整数
zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "sll", &arg, &arg_len,&n,&m)
PHP_FUNCTION展开模式:
#define PHP_FUNCTION ZEND_FUNCTION
#define ZEND_FUNCTION(name) ZEND_NAMED_FUNCTION(ZEND_FN(name))
#define ZEND_FN(name) zif_##name
#define ZEND_NAMED_FUNCTION(name) void name(INTERNAL_FUNCTION_PARAMETERS)
#define INTERNAL_FUNCTION_PARAMETERS int ht, zval *return_value, zval **return_value_ptr, zval *this_ptr, int return_value_used TSRMLS_DC
提供外部函数列表的展开模式,即PHP_FE
#define PHP_FE ZEND_FE
#define ZEND_FE(name, arg_info) ZEND_FENTRY(name, ZEND_FN(name), arg_info, 0)
#define ZEND_FENTRY(zend_name, name, arg_info, flags) { #zend_name, name, arg_info, (zend_uint) (sizeof(arg_info)/sizeof(struct _zend_arg_info)-1), flags },
按照php处理请求的各个阶段我们分为 mint rint script rshutdown mshutdown,我们自定义的外部扩展对请求的各个阶段也可以进行相应的干预。
各个阶段的干预接口为:
MINIT阶段:模块初始化阶段
对应PHP_MINIT(wully)的实现
在sapi的start阶段,调用zend_startup_module
ZEND_API int zend_startup_module(zend_module_entry *module)执行两个动作注册模块,启动模块
注册模块
zend_register_internal_module(module TSRMLS_CC)
把module放入 module_registry 中,其中 module_registry 为一个全局的hash tab
启动模块
zend_startup_module_ex(module TSRMLS_CC) == SUCCESS
检测依赖是否就绪,然后运行该module的 PHP_MINIT函数
MINIT宏展开模式:
#define PHP_MINIT ZEND_MODULE_STARTUP_N
#define ZEND_MINIT ZEND_MODULE_STARTUP_N
#define ZEND_MODULE_STARTUP_N(module) zm_startup_##module
RINIT阶段:请求初始化
对应PHP_RINIT(wully)的实现
int php_request_startup(TSRMLS_D)
void zend_activate_modules(TSRMLS_D)
int module_registry_request_startup(zend_module_entry *module TSRMLS_DC)
module->request_startup_func(module->type, module->module_number TSRMLS_CC)
RINIT宏展开模式
#define PHP_RINIT ZEND_MODULE_ACTIVATE_N
#define ZEND_RINIT ZEND_MODULE_ACTIVATE_N
#define ZEND_MODULE_ACTIVATE_N(module) zm_activate_##module
script阶段
我们的PHP_FUNCTION执行在script阶段,他们作为php语言编译和解释的一部分,纳入到opcode的执行中
PHP_RSHUTDOWN阶段:请求关闭
对应PHP_RSHUTDOWN(wully))的实现
void php_request_shutdown(void *dummy)
void zend_deactivate_modules(TSRMLS_D)
int module_registry_cleanup(zend_module_entry *module TSRMLS_DC)
module->request_shutdown_func(module->type, module->module_number TSRMLS_CC)
PHP_RSHUTDOWN 宏展开模式
#define PHP_RSHUTDOWN ZEND_MODULE_DEACTIVATE_N
#define ZEND_MODULE_DEACTIVATE_N(module) zm_deactivate_##module
PHP_MSHUTDOWN阶段:模块关闭
对应PHP_MSHUTDOWN(wully))的实现
在zend_startup中注册模块销毁时的析构函数
int zend_startup(zend_utility_functions *utility_functions, char **extensions TSRMLS_DC)
zend_hash_init_ex(&module_registry, 50, NULL, ZEND_MODULE_DTOR, 1, 0);
#define ZEND_MODULE_DTOR (void (*)(void *)) module_destructor
void module_destructor(zend_module_entry *module)
module->module_shutdown_func(module->type, module->module_number TSRMLS_CC);
PHP_MSHUTDOWN 宏展开模式
#define PHP_MSHUTDOWN ZEND_MODULE_SHUTDOWN_N
#define ZEND_MODULE_SHUTDOWN_N(module) zm_shutdown_##module
PHP_MINFO跟前面的宏意义有所不同,主要用于在phpinfo中显示模块的信息
#define PHP_MINFO ZEND_MODULE_INFO_N
#define ZEND_MINFO ZEND_MODULE_INFO_N

PHP用于构建动态网站,其核心功能包括:1.生成动态内容,通过与数据库对接实时生成网页;2.处理用户交互和表单提交,验证输入并响应操作;3.管理会话和用户认证,提供个性化体验;4.优化性能和遵循最佳实践,提升网站效率和安全性。

PHP在数据库操作和服务器端逻辑处理中使用MySQLi和PDO扩展进行数据库交互,并通过会话管理等功能处理服务器端逻辑。1)使用MySQLi或PDO连接数据库,执行SQL查询。2)通过会话管理等功能处理HTTP请求和用户状态。3)使用事务确保数据库操作的原子性。4)防止SQL注入,使用异常处理和关闭连接来调试。5)通过索引和缓存优化性能,编写可读性高的代码并进行错误处理。

在PHP中使用预处理语句和PDO可以有效防范SQL注入攻击。1)使用PDO连接数据库并设置错误模式。2)通过prepare方法创建预处理语句,使用占位符和execute方法传递数据。3)处理查询结果并确保代码的安全性和性能。

PHP和Python各有优劣,选择取决于项目需求和个人偏好。1.PHP适合快速开发和维护大型Web应用。2.Python在数据科学和机器学习领域占据主导地位。

PHP在电子商务、内容管理系统和API开发中广泛应用。1)电子商务:用于购物车功能和支付处理。2)内容管理系统:用于动态内容生成和用户管理。3)API开发:用于RESTfulAPI开发和API安全性。通过性能优化和最佳实践,PHP应用的效率和可维护性得以提升。

PHP可以轻松创建互动网页内容。1)通过嵌入HTML动态生成内容,根据用户输入或数据库数据实时展示。2)处理表单提交并生成动态输出,确保使用htmlspecialchars防XSS。3)结合MySQL创建用户注册系统,使用password_hash和预处理语句增强安全性。掌握这些技巧将提升Web开发效率。

PHP和Python各有优势,选择依据项目需求。1.PHP适合web开发,尤其快速开发和维护网站。2.Python适用于数据科学、机器学习和人工智能,语法简洁,适合初学者。

PHP仍然具有活力,其在现代编程领域中依然占据重要地位。1)PHP的简单易学和强大社区支持使其在Web开发中广泛应用;2)其灵活性和稳定性使其在处理Web表单、数据库操作和文件处理等方面表现出色;3)PHP不断进化和优化,适用于初学者和经验丰富的开发者。


热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

SecLists
SecLists是最终安全测试人员的伙伴。它是一个包含各种类型列表的集合,这些列表在安全评估过程中经常使用,都在一个地方。SecLists通过方便地提供安全测试人员可能需要的所有列表,帮助提高安全测试的效率和生产力。列表类型包括用户名、密码、URL、模糊测试有效载荷、敏感数据模式、Web shell等等。测试人员只需将此存储库拉到新的测试机上,他就可以访问到所需的每种类型的列表。

Atom编辑器mac版下载
最流行的的开源编辑器

DVWA
Damn Vulnerable Web App (DVWA) 是一个PHP/MySQL的Web应用程序,非常容易受到攻击。它的主要目标是成为安全专业人员在合法环境中测试自己的技能和工具的辅助工具,帮助Web开发人员更好地理解保护Web应用程序的过程,并帮助教师/学生在课堂环境中教授/学习Web应用程序安全。DVWA的目标是通过简单直接的界面练习一些最常见的Web漏洞,难度各不相同。请注意,该软件中

mPDF
mPDF是一个PHP库,可以从UTF-8编码的HTML生成PDF文件。原作者Ian Back编写mPDF以从他的网站上“即时”输出PDF文件,并处理不同的语言。与原始脚本如HTML2FPDF相比,它的速度较慢,并且在使用Unicode字体时生成的文件较大,但支持CSS样式等,并进行了大量增强。支持几乎所有语言,包括RTL(阿拉伯语和希伯来语)和CJK(中日韩)。支持嵌套的块级元素(如P、DIV),

适用于 Eclipse 的 SAP NetWeaver 服务器适配器
将Eclipse与SAP NetWeaver应用服务器集成。