所以这就是交易:我已经在一个庞大的系统(PHP)上工作了几年,现在,我决定放弃部分繁重的golang脚本工作。
到目前为止,我将一些PHP脚本复制到go版本。然后,我能够对哪个选项更好进行基准测试(好吧,我知道go更快,但我需要curl或socket来实现通信,所以,我必须检查它是否仍然值得。)
其中一个脚本只生成一个随机代码,检查这个新代码是否已经在使用(在mysql db上),如果没有,记录新代码并返回它,如果已经在使用,只需递归调用该函数再次找到一个独家代码。非常简单。
我已经在php中使用了这个代码生成器,因此,在go中使用json params将其称为http / post。 使用linux终端,我称之为
curl -H [headers] -d [jsondata] [host]
我回来了一个非常简单的json
{"locator" : "XXXXXX"}
之后,我编写了一个简单的php脚本来调用脚本并检查每个脚本完成的时间,例如:
<?php
public function indexAction()
{
$phpTime = $endPHP = $startPHP =
$goTime = $endGO = $startGO = 0;
// my standard function already in use
ob_start();
$startPHP = microtime(true);
$MyCodeLocator = $this->container->get('MyCodeLocator')->Generate(22, 5);
$endPHP = microtime(true);
ob_end_clean();
sleep(1);
// Lets call using curl
$data = array("comon" => "22", "lenght" => "5");
$data_string = json_encode($data);
ob_start();
$startGO = microtime(true);
$ch = curl_init('http://localhost:8888/dev/fibootkt/MyCodeGenerator');
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, "POST");
curl_setopt($ch, CURLOPT_POSTFIELDS, $data_string);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($ch, CURLOPT_HTTPHEADER, array(
'Content-Type: application/json',
'Content-Length: ' . strlen($data_string))
);
$result = curl_exec($ch);
curl_close($ch);
$endGO = microtime(true);
ob_end_clean();
$result_rr = json_decode($result);
// tst just echo the variable in a nice way, second parameter means no kill execution
tst($result, 1); // HERE IS MY PROBLEM, please read below
tst($result_rr, 1); // IT SHOW NULL
sleep(1);
// just to have params for comparision, always try by shell script
ob_start();
$startShell = microtime(true);;
$exec = "curl -H \"Content-Type: application/json\" -d '{\"comon\":\"22\"}' http://localhost:8888/dev/fibootkt/MyCodeGenerator";
$output = shell_exec($exec);
$endShell = microtime(true);
ob_end_clean();
tst(json_decode($output),1); // here show a stdclass with correct value
tst($output,1); // here shows a json typed string ready to be converted
// and here it is just for show the execution time ...
$phpTime = $endPHP - $startPHP;
$goTime = $endGO - $startGO ;
$shellTime = $endShell - $startShell;
tst("php " . $phpTime, 1);
tst("curl ". $goTime, 1);
tst("shell " . $shellTime, 1);
我从GO获得结果: 通过Shell脚本:
{"locator" : "DPEWX22"}
因此,这个很容易解码为stdobj。
但是,使用curl,操作更快!所以,我想用它。 但是,卷曲请求响应类似于:
{"Value":"string","Type":{},"Offset":26,"Struct":"CodeLocatorParams","Field":"lenght"}
{"locator":"DPEWX22"}
当我尝试解码它时,我得到一个null结果!!!
CodeLocatorParams我使用的struct类型去获取post params,如下所示
所以,这是我的问题:为什么GO会回来这个?如何避免它。
我有另一个类似的go脚本,它不接受params并响应类似的json(但在这种情况下,是一个qrcode图像路径)并且它工作正常!
我的去功能:
type CodeLocatorParams struct {
Comon string `json:"comon"`
Lenght int `json:"lenght"`
}
func Generate(w http.ResponseWriter, r *http.Request) {
data, err := ioutil.ReadAll(io.LimitReader(r.Body, 1048576))
if err != nil {
fmt.Println(err)
panic(err)
}
if err := r.Body.Close(); err != nil {
panic(err)
}
// retrieve post data and set it to params which is CodeLocatorParams type
var params CodeLocatorParams
if err := json.Unmarshal(data, ¶ms ); err != nil {
w.Header().Set("Content-Type", "application/json; charset=UTF-8")
w.WriteHeader(422) // unprocessable entity
if err := json.NewEncoder(w).Encode(err); err != nil {
fmt.Println(err)
panic(err)
}
}
var result struct{
Locator string `json:"locator"`
}
// here actually comes the func that generates random code and return it as string, but for now, just set it static
result.Locator = "DPEWX22"
w.Header().Set("Content-Type", "application/json; charset=UTF-8")
json.NewEncoder(w).Encode(result)
}
答案 0 :(得分:1)
解析传入的JSON时出错。此错误将作为{"Value":"string","Type":{},"Offset":26,"Struct":"CodeLocatorParams","Field":"lenght"}
写入响应。处理程序继续执行并写入正常响应{"locator":"DPEWX22"}
。
以下是解决问题的方法: