Go调用C API并设置回调函数

Go调用 C API并设置回调函数

一听这名字,就些许骚气,耗费大半天饱读度娘,试过一个个坑,终于哭尽肝来!

环境坑(编译器GoLand)

  • 工程文件必须在GOPATH环境变量目录下
    在这里插入图片描述
    否则,编译无输出,一位写Go的胸得说的。
  • 记住"Run kind"模式设置为“Package”模式
    在这里插入图片描述
    切记不要选"file"模式,因为“Export” Go方法,和Go中得C语言会单独编译为小文件模块,要不会出现不可以思议得编译错误,百思不得其姐。

Go编写回调

//export showMsg
func showMsg(msg *C.char,len int)  {
	fmt.Println("show msg in go:",C.GoString(msg))
}
//export showMsg

这个不是注释,写C/C++的人儿注意了,这是Go方法导出为C方法的金钥匙!

Go中写真正的C回调

在这里插入图片描述
以下这个不是注释,而且和import "C"不能有空行。

/*
.....
*/

这里的showMsg(s,len)就是上一步export的Go方法。

动态调用C动态库Dll,设置回调

func init() {
	dir, _ := filepath.Abs(filepath.Dir(os.Args[0]))
	dllfile := fmt.Sprintf(`%s\test.dll`, dir)
	lib := syscall.NewLazyDLL(dllfile)
	setcallback = lib.NewProc("setCallback")
}

func main() {
	fmt.Print("start\n")
	C.sayHello()
	_, _, err := setcallback.Call((uintptr)(unsafe.Pointer(C.showMsg_cgo)))
	if err != syscall.Errno(0) {
		fmt.Println(`调用出错,错误原因:%v`, err)
	}
	select
	{}
}

C/C++动态库导出函数和代码

#ifdef DLL1_EXPORTS
#define DLL1_API extern "C" __declspec(dllexport)
#else
#define DLL1_API __declspec(dllimport)
#endif

typedef  void ShowInfo(char *str, int len);

DLL1_API void setCallback(ShowInfo * cb);
#include "Dll1.h"
#include <thread>
#include <chrono>

using namespace std::chrono;

ShowInfo *  g_showInfo = nullptr;
int			g_cnt      = 0;

DLL1_API void setCallback(ShowInfo * cb)
{
	g_showInfo = cb;
	if (g_cnt == 0) {
		std::thread([&]() {
			while (1) {
				if (g_showInfo) {
					char buf[255] = { 0 };
					sprintf_s(buf, "say hello in C %d", ++g_cnt);
					g_showInfo(buf, strlen(buf));
				}
			}
		}).detach();
	}
}

结果截图

在这里插入图片描述

Go完整代码

  • main.go
package main
/*
void showMsg_cgo(char *s,int len);
*/
import "C"
import (
	"fmt"
	"os"
	"path/filepath"
	"syscall"
	"unsafe"
)
var (
	setcallback   *syscall.LazyProc // 初始化,path为库配置文件config.xml路径
)

//export showMsg
func showMsg(msg *C.char,len int)  {
	fmt.Println("show msg in go:",C.GoString(msg))
}

func init() {
	dir, _ := filepath.Abs(filepath.Dir(os.Args[0]))
	dllfile := fmt.Sprintf(`%s\test.dll`, dir)
	lib := syscall.NewLazyDLL(dllfile)
	setcallback = lib.NewProc("setCallback")
}

func main() {
	fmt.Print("start\n")
	C.sayHello()
	_, _, err := setcallback.Call((uintptr)(unsafe.Pointer(C.showMsg_cgo)))
	if err != syscall.Errno(0) {
		fmt.Println(`调用出错,错误原因:%v`, err)
	}
	select
	{}
}

  • bridge.go
package main
/*
#include <stdio.h>

void showMsg_cgo(char *s ,int len)
{
	showMsg(s,len);
}
*/
import "C"
这个错误信息表明,在尝试向数据库插入数据时,提供的日期值 `'12830-08-01 00:00:00'` 超出了 MySQL 中 `DATETIME` 或 `TIMESTAMP` 类型的有效范围。 ### 原因分析 1. **MySQL 的有效日期范围** - 对于 `DATE`, `DATETIME`, 和 `TIMESTAMP` 数据类型,有效的日期范围通常是: ``` 1000-01-01 到 9999-12-31 (对于 DATE/DATETIME) 1970-01-01 到 2038-01-19 (对于 TIMESTAMP) ``` 提供的日期 `'12830-08-01'` 远远超出了上述范围,因此无法存储到列 `end_date` 中。 2. **输入验证不足** 如果未对用户输入或程序生成的数据进行严格的校验,则可能导致这种无效日期进入数据库操作流程。 --- ### 解决方案 #### 方法一:修正数据源中的非法日期值 检查并修改导致该问题的具体记录。例如,如果你发现某个字段被误设为未来的极端年份(如12830),可以将其调整为合理的数值(比如当前时间或其他合法值)。示例 SQL 修改命令如下: ```sql UPDATE your_table_name SET end_date = '9999-12-31' WHERE id = problematic_id; ``` #### 方法二:在应用层添加数据验证逻辑 确保所有传递给数据库的操作都经过合法性检查,避免类似超出范围的情况发生。可以在代码中加入条件判断: ```python import datetime def validate_date(date_str): try: # 尝试解析字符串为标准格式,并限制最大日期不超过指定范围 dt_obj = datetime.datetime.strptime(date_str, '%Y-%m-%d %H:%M:%S') max_allowed_date = datetime.datetime(9999, 12, 31) # 最大允许日期设置 if dt_obj > max_allowed_date or dt_obj < datetime.datetime(1900, 1, 1): # 可选最小日期设置 raise ValueError("Date out of valid range") except Exception as e: print(f"Invalid date {date_str}: {e}") return False return True # 测试函数 test_dates = ["12830-08-01 00:00:00", "2024-05-20 12:30:00"] for d in test_dates: is_valid = validate_date(d) print(f"{d} -> {'Valid' if is_valid else 'Not Valid'}") ``` #### 方法三:更改表结构以适应更大范围的需求 如果业务场景确实需要处理非常遥远未来的时间点,考虑使用更大的容器替代默认的数据类型——将现有 DATETIME 改成 BIGINT 来直接保存 Unix 时间戳形式表示的大整数即可支持任意大小数字代表无限延伸时刻。 不过需注意这会牺牲掉一部分查询便利性和标准化规范优势! --- ### 总结建议 优先从源头解决问题,即保证正确的原始数据质量;其次增强系统健壮性的措施也很关键,包括但不限于前端界面提示、后端API接口约束以及SQL脚本层面的安全防护机制等多方面努力共同防范此类异常的发生概率降至最低限度之内。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值