乘着还没有添加商业功能之前,先给大家把福利分享了

希望有需要的朋友能够用的上

这个框架是在用windows平台,GO做的http/https服务,调用dll现有的库接口实现特定功能的大框架

//dll

//现有库不能直接使用,必须要有一个代理库,这个必要性,我想大家都清楚

//XSKFCSPEnroll是我司商业库,这个就不能让诸位浏览了,大家都能理解

//Enroll库是代理库

//我们以获得调用的商业库版本为例来阐述

//Enroll.h

#ifndef _ENROLL_
#define _ENROLL_

# ifdef ENROLL
# define Export _declspec(dllexport)
# else
# define Export _declspec(dllimport)
# endif

extern "C" {

Export int getHandle(void** pEnrollHandle, void** pLibHandle);
Export int releaseHandle(void** pEnrollHandle, void** pLibHandle);

Export int freeMem(void** pEnrollHandle, void** pLibHandle, void** p);

Export int getVersion(void** pEnrollHandle, void** pLibHandle, char** ver);

}

#endif

//Enroll.cpp

#include "stdafx.h"
#define ENROLL

#include "Enroll.h"
#include "app_util.h"

Export int getHandle(void** pEnrollHandle, void** pLibHandle)
{
  HMODULE handle = LoadLibrary("XSKFCSPEnroll.dll");
  if (handle)
  {
    typedef void*(*F)();
    F f = (F)GetProcAddress(handle, "getLibraryHandle");
    if (!f)
    {
      FreeLibrary(handle);
    }
    else
    {
      void* libhandle = f();
      if (!libhandle)
      {
        FreeLibrary(handle);
      }
      else
      {
        *pEnrollHandle = handle;
        *pLibHandle = libhandle;
        return 1;
      }
    }
  }

  return 0;
}

Export int releaseHandle(void** pEnrollHandle, void** pLibHandle)
{
  if (pEnrollHandle && pLibHandle)
  {
    typedef void(*F)(void*);
    F f = (F)GetProcAddress((HMODULE)*pEnrollHandle, "releaseLibraryHandle");
    if (!f)
    {
      FreeLibrary((HMODULE)*pEnrollHandle);
    }
    else
    {
      f(*pLibHandle);
      FreeLibrary((HMODULE)*pEnrollHandle);
      return 1;
    }
  }

  return 0;
}

Export int freeMem(void** pEnrollHandle, void** pLibHandle, void** p)
{
  if (p&&(*p))
  {
    free(*p);
    return 1;
  }

  return 0;
}

Export int getVersion(void** pEnrollHandle, void** pLibHandle, char** ver)
{
  if (pEnrollHandle && pLibHandle)
  {
    typedef HRESULT (*F)(void*, BSTR*);
    F f = (F)GetProcAddress((HMODULE)*pEnrollHandle, "get_Version");
    if (f)
    {
      CComBSTR ret;
      if (S_OK==f(*pLibHandle, &ret))
      {
        CMemBlock<char> c_ret = CUtf8Unicode::f8UStringConvert(ret);
        *ver = (char*)malloc(c_ret.GetSize());
        if (*ver)
        {
          memcpy(*ver, c_ret, c_ret.GetSize());
          return 1;
        }
      }
    }
  }

  return 0;
}

//Enroll_Server.go

// 支持http/https

//遗留问题:服务如何退出,现在我还不知道

package main

import (
  "fmt"
  "os"
  "net/http"
  "Enroll_lib"
  //"encoding/json"
)

var w32_handle Enroll_lib.Win32_handle

func init() {
  fmt.Println("init")
  var pw32 *Enroll_lib.Win32_handle = &w32_handle
  res := pw32.LoadLib()
  if !res {
    fmt.Println("Load library failure")
  }
  res = pw32.GetHandle()
  if !res {
    fmt.Println("GetHandle failure")
  }
  //demo
  fmt.Println("Version: ", pw32.GetVersion())
}

func entry(res http.ResponseWriter, req *http.Request) {
  fmt.Fprintf(res, "Hello World!\n")
}

func main() {
  Port := "8086"
  IsHttp := true
  arg_num := len(os.Args)
  if 2<=arg_num {
    Port = os.Args[1]
  }
  if 3<=arg_num {
    if os.Args[2]=="true" {
      IsHttp = true
    } else {
      IsHttp = false
    }
  }
  fmt.Printf("server is http %t\n", IsHttp)
  fmt.Println("server listens at ", Port)

  http.HandleFunc("/", entry)

  var err error
  if IsHttp {
    err = http.ListenAndServe(":"+Port, nil)
  } else {
    err = http.ListenAndServeTLS(":"+Port, "server.crt", "server.key", nil)
  }
  if err != nil {
    fmt.Println("Server failure /// ", err)
  }

  w32_handle.FreeLib();
  fmt.Println("quit");
}

//libs

//自开发的Go库

package Enroll_lib

import(
  "fmt"
  "syscall"
  "C"
  "unsafe"
)

type Win32_handle struct {
  handle syscall.Handle
  c_enroll_handle uintptr
  c_lib_handle uintptr
  err error
}

func (h *Win32_handle) LoadLib() bool {
  h.handle, h.err = syscall.LoadLibrary("Enroll.dll")
  if h.err!=nil {
    fmt.Println("Enroll.dll not found")
    return false
  }

  return true
}

func (h *Win32_handle) FreeLib() {
  syscall.FreeLibrary(h.handle)
}

func (h *Win32_handle) GetHandle() bool{
  getHandle, err := syscall.GetProcAddress(h.handle, "getHandle")
  if err!=nil {
    fmt.Println("getHandle not found")
    return false
  }
  r,_,retstr := syscall.Syscall(uintptr(getHandle), 3, uintptr(unsafe.Pointer(&h.c_enroll_handle)),
                uintptr(unsafe.Pointer(&h.c_lib_handle)),
                0)
  fmt.Println(retstr);
  if r==0 {
    fmt.Println("syscall failuer at getHandle")
    return false
  }

  return true
}

func (h *Win32_handle) ReleaseHandle() {
  releaseHandle, err := syscall.GetProcAddress(h.handle, "ReleaseHandle")
  if err!=nil {
    fmt.Println("releaseHandle not found")
  }
  r,_,retstr := syscall.Syscall(uintptr(releaseHandle), 3, uintptr(unsafe.Pointer(&h.c_enroll_handle)),
                uintptr(unsafe.Pointer(&h.c_lib_handle)),
                   0)
  fmt.Println(retstr)
  if r==0 {
    fmt.Println("syscall failuer at releaseHandle")
  }
}

func (h *Win32_handle) FreeMem(p *C.char) {
  freeMem, err := syscall.GetProcAddress(h.handle, "freeMem")
  if err!=nil {
    fmt.Println("freeMem not found")
  }
  r,_,retstr := syscall.Syscall(uintptr(freeMem), 3, uintptr(unsafe.Pointer(&h.c_enroll_handle)),
                uintptr(unsafe.Pointer(&h.c_lib_handle)),
                uintptr(unsafe.Pointer(&p)))
  fmt.Println(retstr)
  if r==0 {
    fmt.Println("syscall failuer at freeMem")
  }
}

func (h *Win32_handle) GetVersion() string {
  getVersion, err := syscall.GetProcAddress(h.handle, "getVersion")
  if err!=nil {
    fmt.Println("getVersionm not found")
  }
  var p *C.char
  r,_,retstr := syscall.Syscall(uintptr(getVersion), 3, uintptr(unsafe.Pointer(&h.c_enroll_handle)),
                uintptr(unsafe.Pointer(&h.c_lib_handle)),
                uintptr(unsafe.Pointer(&p)))
  fmt.Println(retstr)
  if r==0 {
    fmt.Println("syscall failure at getVersion")
  }

  defer h.FreeMem(p)

  return C.GoString(p)
}

//现场

//结果-https

Finally:

大家是不是迫不及待的想GO了,如果是,那我很欣慰!

本系列Go讲解暂时得告一段落啦

其实,我真心希望,Go&C能够一统江湖,让其它那些乱七八糟的语言自然衰老去dead

哈哈哈哈哈,上边这句你就当笑话听吧!权当我没说

不听才怪!哈哈哈

劝告:

1. 永远不要说自己的结论和推论的正确性

2. 伯乐只有一个,可惜不是你

3. 客观,唯物,在同事实践中,你才能知道一个人到底行不行

最后,祝大家Go(滚)的顺溜!哈哈哈

windows go dll 框架的更多相关文章

  1. Ligg.EasyWinApp-000: 一款Windows应用编程框架介绍

        本框架(解决方案)是一个Windows应用编程框架和UI库,通过该框架,不需任何代码,通过XML配置文件,搭建任意复杂的Windows应用界面,以类似Execel公式的方式实现基本的过程控制( ...

  2. 使用C语言编写windows服务一般框架

    原文:使用C语言编写windows服务一般框架 编写windows服务和编写windows应用程序一样,有一些回调函数必须填写且向windows 服务管理器(service manager)进行注册, ...

  3. 【C#/WPF】如何查看System.Windows.Interactivity.dll中EventTrigger的EventNames属性有哪些

    WPF项目中,从Nuget搜索并下载System.Windows.Interactivity.dll,安装到项目中,并在XAML界面引入. <UserControl xmlns:i=" ...

  4. Python调用windows下DLL详解

    Python调用windows下DLL详解 - ctypes库的使用 2014年09月05日 16:05:44 阅读数:6942 在python中某些时候需要C做效率上的补充,在实际应用中,需要做部分 ...

  5. 【Windows下DLL查找顺序 】

    一.写作初衷 在Windows下单个DLL可能存在多个不同的版本,若不特别指定DLL的绝对路径或使用其他手段指定,在应用程序加载DLL时可能会查找到错误的版本,进而引出各种莫名其妙的问题.本文主要考虑 ...

  6. Windows中DLL文件的意义及其作用

    Windows中DLL文件的意义及其作用 DLL是Dynamic Link Library的缩写,意为动态链接库.DLL文件即动态链接库文件,是一种可执行文件,它允许程序共享执行特殊任务所必需的代码和 ...

  7. ArgumentException: The Assembly Mono.WebBrowser is referenced by System.Windows.Forms ('Assets/Plugins/System.Windows.Forms.dll'). But the dll is not allowed to be included or could not be found.

    最近有个项目要用到System.Windows.Forms.dll,在Unity编辑器里用着还好好的,但是一导出就给我报错,让我十分不爽. 于是请教百度,搜出了五花八门的答案,没一个能解决我的问题的, ...

  8. Directx11 教程(2) 基本的windows应用程序框架(2)

    原文:Directx11 教程(2) 基本的windows应用程序框架(2)      在本教程中,我们把前面一个教程的代码,进行封装.把初始化函数,Run函数,窗口回调函数,ShutdownWind ...

  9. Directx11 教程(1) 基本的windows应用程序框架(1)

    原文:Directx11 教程(1) 基本的windows应用程序框架(1)        在vs2010中,建立一个新的win32工程,名字是: myTutorialD3D11, 注意:同时勾选Cr ...

随机推荐

  1. db2pd工具

    内容 概览 简介 使用 db2pd 工具 监控的例子 db2pd 工具 用于监控 DB2 实例和数据库的新的 DB2 UDB 工具 简介 DB2 UDB V8.2 带来了一种新工具称为 db2pd,用 ...

  2. tomcat启动项目报错:The specified JRE installation does not exist

    在Build Path里设置好jre和各Library的顺序,代码无报错,启动时弹框,里面的信息是:The specified JRE installation does not exist. 后来想 ...

  3. .NET Core开发日志——Linux版本的SQL Server

    SQL Server 2017版本已经可以在Linux系统上安装,但我在尝试.NET Core跨平台开发的时候使用的是Mac系统,所以这里记录了在Mac上安装SQL Server的过程. 最新的SQL ...

  4. ESP8266 nodemcu

    主要资料来源于一下几个网站 1.nodemcu官网:此处有几个示例和github(用处不大) 2.用户说明:http://nodemcu.readthedocs.io/en/master/  (非常重 ...

  5. springmvc整合spring+mybatis出现的404或者报找不到这个类的时候。

    1.文件配置没有问题的时候看这个看看.输出目录改成这个.要有这里所有的目录才行

  6. falsk 与 django 过滤器的使用与区别

    1,flask中内置的过滤器模板中常用方法: {#过滤器调用方式{{变量|过滤器名称}} #} <!-- safe过滤器,可以禁用转义 --> {{'<strong>hello ...

  7. fiddler抓包常用功能详解

    一.基础部分: 1.设置代理ip及端口,tools --> telerik fiddler options --> connections -->勾选 “ Allow romote ...

  8. 添加linux系统调用的两种方式

    原文:https://blog.csdn.net/sdulibh/article/details/51889279 向linux内核添加系统调用,一是通过编译内核添加,二是通过内核模块的方式添加: 一 ...

  9. 内核poll机制

    内核版本:linux2.6.22.6 硬件平台:JZ2440 驱动源码 poll_key_int_drv.c : #include <linux/module.h> #include &l ...

  10. ES6 --- JS异步编程的几种解决方法及其优缺点

    导言: 我们都知道 JS 是单线程的,这也正是异步编程对于 JS  很重要的原因,因为它无法忍受耗时太长的操作.正因如此有一系列的实现异步的方法. 方法一  setTimeout 常用于:定时器,动画 ...