执行php文件
func Test_exec(t *testing.T) {
engine.Initialize()
ctx := &engine.Context{
Output: os.Stdout,
}
err := engine.RequestStartup(ctx)
if err != nil {
fmt.Println(err)
}
defer engine.RequestShutdown(ctx)
err = ctx.Exec("/tmp/index.php")
if err != nil {
fmt.Println(err)
}
}
其中 /tmp/index.php 的内容为
<?php
echo("hello\n");
Eval,返回值
func Test_eval(t *testing.T) {
engine.Initialize()
ctx := &engine.Context{}
err := engine.RequestStartup(ctx)
if err != nil {
fmt.Println(err)
}
defer engine.RequestShutdown(ctx)
val, err := ctx.Eval("return 'hello';")
if err != nil {
fmt.Println(err)
}
defer engine.DestroyValue(val)
if engine.ToString(val) != "hello" {
t.FailNow()
}
}
返回的value的生命周期所有权是golang程序,所以我们要负责DestroyValue
设置全局变量来传参
func Test_argument(t *testing.T) {
engine.Initialize()
ctx := &engine.Context{}
err := engine.RequestStartup(ctx)
if err != nil {
fmt.Println(err)
}
defer engine.RequestShutdown(ctx)
err = ctx.Bind("greeting", "hello")
if err != nil {
fmt.Println(err)
}
val, err := ctx.Eval("return $greeting;")
if err != nil {
fmt.Println(err)
}
defer engine.DestroyValue(val)
if engine.ToString(val) != "hello" {
t.FailNow()
}
}
传递进去的参数的生命周期是php控制的,在request shutdown的时候内存会被释放。
PHP 回调 Golang
type greetingProvider struct {
greeting string
}
func (provider *greetingProvider) GetGreeting() string {
return provider.greeting
}
func newGreetingProvider(args []interface{}) interface{} {
return &greetingProvider{
greeting: args[0].(string),
}
}
func Test_callback(t *testing.T) {
engine.Initialize()
ctx := &engine.Context{}
err := engine.RequestStartup(ctx)
if err != nil {
fmt.Println(err)
}
defer engine.RequestShutdown(ctx)
err = engine.Define("GreetingProvider", newGreetingProvider)
if err != nil {
fmt.Println(err)
}
val, err := ctx.Eval(`
$greetingProvider = new GreetingProvider('hello');
return $greetingProvider->GetGreeting();`)
if err != nil {
fmt.Println(err)
}
defer engine.DestroyValue(val)
if engine.ToString(val) != "hello" {
t.FailNow()
}
}
PHP 错误日志
func Test_log(t *testing.T) {
engine.PHP_INI_PATH_OVERRIDE = "/tmp/php.ini"
engine.Initialize()
ctx := &engine.Context{
Log: os.Stderr,
}
err := engine.RequestStartup(ctx)
if err != nil {
fmt.Println(err)
}
defer engine.RequestShutdown(ctx)
_, err = ctx.Eval("error_log('hello', 4); trigger_error('sent from golang', E_USER_ERROR);")
if err != nil {
fmt.Println(err)
}
}
其中 /tmp/php.ini 的内容为
error_reporting = E_ALL error_log = "/tmp/php-error.log"
错误会被输出到 /tmp/php-error.log。直接调用error_log会同时再输出一份到stderr
HTTP 输入输出
func Test_http(t *testing.T) {
engine.Initialize()
recorder := httptest.NewRecorder()
ctx := &engine.Context{
Request: httptest.NewRequest("GET", "/hello", nil),
ResponseWriter: recorder,
}
err := engine.RequestStartup(ctx)
if err != nil {
fmt.Println(err)
}
defer engine.RequestShutdown(ctx)
_, err = ctx.Eval("echo($_SERVER['REQUEST_URI']);")
if err != nil {
fmt.Println(err)
}
body, err := ioutil.ReadAll(recorder.Result().Body)
if err != nil {
fmt.Println(err)
}
if string(body) != "/hello" {
t.FailNow()
}
}
所有的PHP超级全局变量都会被初始化为传递进去的Request的值,包括
$_SERVER $_GET $_POST $_FILE $_COOKIE $_ENV
echo的内容,http code和http header会被写回到传入的ResponseWriter
fastcgi_finish_request
PHP-FPM 很常用的一个功能是 fastcgi_finish_request ,用于在php里做一些异步完成的事情。这个特殊的全局函数必须支持
func Test_fastcgi_finish_reqeust(t *testing.T) {
engine.Initialize()
buffer := &bytes.Buffer{}
ctx := &engine.Context{
Output: buffer,
}
err := engine.RequestStartup(ctx)
if err != nil {
fmt.Println(err)
}
defer engine.RequestShutdown(ctx)
ctx.Eval("ob_start(); echo ('hello');")
if buffer.String() != "" {
t.FailNow()
}
ctx.Eval("fastcgi_finish_request();")
if buffer.String() != "hello" {
t.FailNow()
}
}
实际的作用就是把output提前输出到 ResposneWriter 里去,让调用方知道结果。对于当前进程的执行其实是没有影响的,只是影响了output。
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!
# golang
# 调用
# php7
# php7实例
# php7详解及示例代码
# golang、python、php、c++、c、java、Nodejs性能对比
# Golang加密解密之RSA(附带php)
# PHP中SERIALIZE和JSON的序列化与反序列化操作区别分析
# PHP的serialize序列化数据以及JSON格式化数据分析
# php中unserialize返回false的解决方法
# 浅谈php函数serialize()与unserialize()的使用方法
# 非常好用的两个PHP函数 serialize()和unserialize()
# 详解php中serialize()和unserialize()函数
# 浅谈php serialize()与unserialize()的用法
# golang实现php里的serialize()和unserialize()序列和反序列方法详解
# 全局变量
# 希望能
# 里去
# 谢谢大家
# 回调
# 时再
# 返回值
# 直接调用
# FailNow
# ToString
# return
# val
# Test_eval
# echo
# Eval
# DestroyValue
# 来传参
# struct
# greetingProvider
相关文章:
如何通过wdcp面板快速创建网站?
高防服务器:AI智能防御DDoS攻击与数据安全保障
建站之星代理费用多少?最新价格详情介绍
C++用Dijkstra(迪杰斯特拉)算法求最短路径
代刷网站制作软件,别人代刷火车票靠谱吗?
山东云建站价格为何差异显著?
高端网站建设与定制开发一站式解决方案 中企动力
哈尔滨网站建设策划,哈尔滨电工证查询网站?
建站之星后台密码遗忘如何找回?
javascript中对象的定义、使用以及对象和原型链操作小结
胶州企业网站制作公司,青岛石头网络科技有限公司怎么样?
道歉网站制作流程,世纪佳缘致歉小吴事件,相亲网站身份信息伪造该如何稽查?
如何在景安服务器上快速搭建个人网站?
制作门户网站的参考文献在哪,小说网站怎么建立?
高配服务器限时抢购:企业级配置与回收服务一站式优惠方案
内部网站制作流程,如何建立公司内部网站?
制作网站的过程怎么写,用凡科建站如何制作自己的网站?
如何在西部数码注册域名并快速搭建网站?
浙江网站制作公司有哪些,浙江栢塑信息技术有限公司定制网站做的怎么样?
宁波免费建站如何选择可靠模板与平台?
如何快速选择适合个人网站的云服务器配置?
如何选择高效稳定的ISP建站解决方案?
制作充值网站的软件,做人力招聘为什么要自己交端口钱?
c# Task.Yield 的作用是什么 它和Task.Delay(1)有区别吗
如何在沈阳梯子盘古建站优化SEO排名与功能模块?
如何快速搭建虚拟主机网站?新手必看指南
建站ABC备案流程中有哪些关键注意事项?
百度网页制作网站有哪些,谁能告诉我百度网站是怎么联系?
网站制作专业公司有哪些,如何制作一个企业网站,建设网站的基本步骤有哪些?
建站之星logo尺寸如何设置最合适?
如何自己制作一个网站链接,如何制作一个企业网站,建设网站的基本步骤有哪些?
已有域名和空间,如何快速搭建网站?
企业宣传片制作网站有哪些,传媒公司怎么找企业宣传片项目?
保定网站制作方案定制,保定招聘的渠道有哪些?找工作的人一般都去哪里看招聘信息?
建站主机是否等同于虚拟主机?
建站之星安装路径如何正确选择及配置?
厦门模型网站设计制作公司,厦门航空飞机模型掉色怎么办?
重庆网站制作公司哪家好,重庆中考招生办官方网站?
简单实现Android验证码
专业企业网站设计制作公司,如何理解商贸企业的统一配送和分销网络建设?
零基础网站服务器架设实战:轻量应用与域名解析配置指南
宁波自助建站系统如何快速打造专业企业网站?
网站设计制作公司地址,网站建设比较好的公司都有哪些?
无锡制作网站公司有哪些,无锡优八网络科技有限公司介绍?
微信小程序 input输入框控件详解及实例(多种示例)
C++如何编写函数模板?(泛型编程入门)
如何选择网络建站服务器?高效建站必看指南
高端建站如何打造兼具美学与转化的品牌官网?
制作ppt免费网站有哪些,有哪些比较好的ppt模板下载网站?
招贴海报怎么做,什么是海报招贴?
*请认真填写需求信息,我们会在24小时内与您取得联系。