对于用C++做NX二次开发,随着NX的版本变更,二开所用的函数也会有相应的更新调整,生成的DLL不能跨版本运行,报错率极高,甚至不能加载。折腾了好久找到三种方法:

1.纯使用C++的代码开发或者尽可能使用UF函数(方法不大靠普,能不能跨版使有点碰运气的成份)

2.同一功能对应不同的NX版本编译不同版本的DLL,对应不同的版本建立不同的按钮。(亲测是能正常使用,但是一个功能多个按钮,不专业)

3.同一个功能多个版本共用一个按钮,建立一个主DLL通过识别NX主版本,再根据NX的版本调用该功能对应版本的DLL(看起来专业多了,目前NX8.5、NX10、NX12 NX1946 做了测试,正常使用。不知道二开高手是怎么做,有更好的方法请指点下)

法1 和法 2 都有明显的缺陷,不适用,下面说明下法3是如何实现的。(自个网上找资料折腾出来的方法,不知道对不对,目前功能实现起来是没有问题)

1)首先把功能对应NX版本编译不同的DLL

2)新建工程

3)工程创建以下函数

//获取运行程序的路径
HMODULE GetSelfModuleHandle()
{

MEMORY_BASIC_INFORMATION mbi;
return ((::VirtualQuery(GetSelfModuleHandle, &mbi, sizeof(mbi)) != 0)
? (HMODULE)mbi.AllocationBase : NULL);
}
CString GetProgramDir() //获取运行程序的路径
{
char exeFullPath[MAX_PATH]; // 全路径
string strPath = "";
GetModuleFileName(GetSelfModuleHandle(), exeFullPath, MAX_PATH);
strPath = (string)exeFullPath; // 获取程序路径
int pos = strPath.find_last_of('\\', strPath.length());
CString path;
path = strPath.substr(0, pos).c_str();
return path;
}

string MyClass::GetPath() //输出程序路径
{
CString exe_path = GetProgramDir();
string ProgramDirPath = exe_path;

return string(ProgramDirPath);//返回路径
}

string MyClass::YiNingToolPath(string DLLDir) //分割程序路径获取工具目录
{
try
{
//反向找位置,分割字符串(只读取文件夹路径)
string strPath = DLLDir;
string strDir;
int nPos = strPath.find_last_of('\\');
if (string::npos != nPos)
{
strDir = strPath.substr(0, nPos);
}

return string(strDir);//返回文件夹路径
}
catch (exception& ex)
{
//---- Enter your exception handling code here -----
MyClass::theUI->NXMessageBox()->Show("分割程序路径获取工具目录", NXOpen::NXMessageBox::DialogTypeError, ex.what());
}

}

void MyClass::UFUN_API_Call_DLL(char* dllPath)
{
//调系统命令
typedef void(*load_ufusr_f_p_t)(char* param, int* retcode, int paramLen);
load_ufusr_f_p_t load_ufusr_ptr = NULL;
int irc = UF_load_library(dllPath, "ufusr", (UF_load_f_p_t*)&load_ufusr_ptr);

//调用DLL
if (load_ufusr_ptr != NULL)
{
int retcode;
load_ufusr_ptr(dllPath, &retcode, 1);
}

if (irc != 0)
{
uc1601("failed load", 1);
UF_unload_library(dllPath); // 注意该语句只能在失败时被调用,否则UG会退出
return;
}
}

void MyClass::GetNXRev()//获取NX主要版本
{
try
{
UF_initialize();
theSession->ListingWindow()->Open();//打开信息窗口

//获取NX主要版本
UF_get_release(&NXrelease); //获取NX主要版本
//theSession->ListingWindow()->WriteLine(NXrelease);

UF_terminate();

}
catch (exception& ex)
{
//---- Enter your exception handling code here -----
MyClass::theUI->NXMessageBox()->Show("帮助", NXOpen::NXMessageBox::DialogTypeError, "程序错误,请检查代码");
}
}
void MyClass::CallToolingDLL(string CallDllName) //对应版本调用DLL
{
try
{
UF_initialize();
theSession->ListingWindow()->Open();//打开信息窗口

string CurrentDllPath = GetPath();//输出程序路径
string CurrentToolPath = YiNingToolPath(CurrentDllPath); //分割程序路径获取工具目录
//theSession->ListingWindow()->WriteLine("当前DLL路径:" + CurrentDllPath);
//theSession->ListingWindow()->WriteLine("当前外挂路径:" + CurrentToolPath);

string NX85DllPath = CurrentDllPath + "\\NX85\\" + CallDllName + "_NX85.dll";
string NX10DllPath = CurrentDllPath + "\\NX10\\" + CallDllName + "_NX10.dll";
string NX12DllPath = CurrentDllPath + "\\NX12\\" + CallDllName + "_NX12.dll";
string NX1946DllPath = CurrentDllPath + "\\NX1946\\" + CallDllName + "_NX1946.dll";

char YiNing_tools_NX85[256];
char YiNing_tools_NX10[256];
char YiNing_tools_NX12[256];
char YiNing_tools_NX1946[256];
sprintf(YiNing_tools_NX85, "%s", NX85DllPath.c_str());
sprintf(YiNing_tools_NX10, "%s", NX10DllPath.c_str());
sprintf(YiNing_tools_NX12, "%s", NX12DllPath.c_str());
sprintf(YiNing_tools_NX1946, "%s", NX1946DllPath.c_str());

GetNXRev();//获取NX主版本

//获取NX主要版本
if (strcmp(NXrelease, "NX V8.5") == 0)
{
UFUN_API_Call_DLL(YiNing_tools_NX85);
//theSession->ListingWindow()->WriteLine(YiNing_tools_NX85);
}
else if (strcmp(NXrelease, "NX V10.0") == 0)
{
UFUN_API_Call_DLL(YiNing_tools_NX10);
//theSession->ListingWindow()->WriteLine(YiNing_tools_NX10);
}
else if (strcmp(NXrelease, "NX V12.0") == 0)
{
UFUN_API_Call_DLL(YiNing_tools_NX12);
//theSession->ListingWindow()->WriteLine(YiNing_tools_NX12);
}
else if (strcmp(NXrelease, "NX V1926") == 0)
{
UFUN_API_Call_DLL(YiNing_tools_NX1946);
//theSession->ListingWindow()->WriteLine(YiNing_tools_NX1946);
}

UF_terminate();

}
catch (exception& ex)
{
//---- Enter your exception handling code here -----
MyClass::theUI->NXMessageBox()->Show("对应版本调用", NXOpen::NXMessageBox::DialogTypeError, "程序错误,请检查代码");
}
}

3)入口函数徙通过字符串对比识别菜单按钮调用DLL

4)编译生成DLL:YiNingToolsSelectMen.dll

5)外挂菜单挂上生成的DLL

6)效果图如下

NX二次开发 版本通用方法的更多相关文章

  1. 【NX二次开发】NX内部函数,查找内部函数的方法

    [NX二次开发]NX内部函数,libufunx.dll文件中的内部函数 [NX二次开发]NX内部函数,libugui.dll文件中的内部函数 [NX二次开发]NX内部函数,libuifw.dll文件中 ...

  2. NX二次开发-基于MFC界面的NX对Excel读写操作(OLE方式(COM组件))

    NX二次开发API里没有对EXCAL读写操作的相关函数,市面上有很多种方法去实现,比如UFUN调KF,ODBC,OLE(COM组件)等等.这里我是用的OLE(COM组件)方式去做的,这种在VC上创建的 ...

  3. NX二次开发-UDO用户自定义对象(UFUN)【持续完善】

    每当提起UDO总是会让我想起大专毕业那会失业找工作,后来有个宝贵机会去了软件公司上班,拿到了我人生中的第一个NX二次开发项目,一个关于测量汽车前后左右摄像头的项目.当时那个项目就用到了UDO,对于只看 ...

  4. C# NX二次开发环境搭建

    在网上看到一篇C#二次开发环境搭建的文章:NX二次开发-使用NXOPEN C#手工搭建开发环境配置 ,写得非常好.我按照文章操作,过程中遇到几个问题,把问题分享给大家,希望对各位有帮助. 注意三点: ...

  5. NX二次开发-C#使用DllImport调用libufun.dll里的UF函数(反编译.net.dll)调用loop等UF函数

    在写这篇文章的时候,我正在头晕,因为下班坐车回家,有些晕车了.头疼的要死.也吃不下去饭. 版本:NX11+VS2013 最近这一年已经由C++过度到C#,改用C#做应用程序开发和NX二次开发. C#在 ...

  6. NX二次开发-创建NX9 NXOpenCPP Wizard开发向导模板

    这篇文章,我在CSDN,在唐工论坛都发过.http://www.nxopen.cn/forum.php?mod=viewthread&tid=2039&highlight=NX9 博客 ...

  7. NX二次开发-基于NX开发向导模板的NX对Excel读写操作(OLE方式(COM组件))

    在看这个博客前,请读者先去完整看完:NX二次开发-基于MFC界面的NX对Excel读写操作(OLE方式(COM组件))https://ufun-nxopen.blog.csdn.net/article ...

  8. NX二次开发-BlockUI对话框嵌套MFC对话框制作进度条

    半年前在一些QQ群看到有大神NX二次开发做出了进度条,那个时候我还不会弄,也不知道怎么弄得,后来断断续续得研究了一下,直到今天我把它做出来了.内心还是很喜悦的!回想自己这两年当初从没公司肯给我做NX二 ...

  9. NX二次开发-UFUN计算两点距离UF_VEC3_distance

    NX11+VS2013 #include <uf.h> #include <uf_curve.h> #include <uf_vec.h> UF_initializ ...

  10. NX二次开发-NX+VS写代码设断点调试技巧

    在做NX二次开发的时候写完代码,编译可以通过,但是执行的时候却没有反应,或者得到的结果不对,说明肯定有地方传值出错了.我在查找代码错误的时候有几种方法:1.uc1601打印函数输入和输出的值看对不对. ...

随机推荐

  1. File、Files、Path、Paths

    一.Path.Paths 和 File.Files // Paths 工具类,用于获取 Path 实例 Path path = Paths.get("files/Data.txt" ...

  2. 【SSO单点系列】(1):CAS4.0 之环境的搭建

    [SSO单点系列](1):CAS4.0 环境的搭建 一.概述 今天开始写CAS相关的第一篇文章,这篇文章主要是关于CAS环境的搭配,提供给刚刚接触CAS的一个入门指南,并演示一个CAS的最简单的实例 ...

  3. 拓展django-haystack全文检索的样式和搜索频率限制

    一.样式: django-haystack在utils模块中封装了HighHighlighter用于配置搜索结果的样式展示.想要更改结果的样式,可以写个子类重写相应的方法达到效果 1.关键字高亮: H ...

  4. UF_OBJ_delete_array_of_objects函数vector转数组用法

        1 UF_initialize(); 2 std::vector<tag_t>tool_tag; 3 tag_t ObjectTag = NULL_TAG; 4 int Type, ...

  5. S家lic

    1,用1patch里对应的工具patch 2019的s家的工具2,用ocad里的lmgrd和snpslmd替换s家的scl里的3,用scl_keygen产生license,再用fix.bat,添加si ...

  6. Object.assign合并多个对象

    语法: Object.assign(target, ...sources) target, 目标对象 sources, 源对象 assign函数用来合并多个对象. 该方法会修改第一个对象,函数在最后r ...

  7. 关于JDK1.8 java HashMap的tableSizeFor的解析:一个数最近2的幂次数方法

    简介 一个数的最近2的幂次数,是java hashmap初始化方法指定容量里面对容量进行处理采用的方法 1.位运算符号介绍 符号 描述 运算规则 & 与 两个位都为1时,结果才为1 | 或 两 ...

  8. python与c++区别

    1 都是面向对象编程,但是python是脚本语言,无需main函数 2 python不需要引用库,前不要定义类型,后不要: 列表可以用-x,表示倒数第几个,不能用a++ a = [1, 2, 3, 4 ...

  9. react实现某个元素节点截图(html2canvas)

    1.安装所需插件:html2canvas: 2.引入插件: import html2canvas from 'html2canvas'; 3.使用: const creatImg = () => ...

  10. 打包pom文件

    <?xml version="1.0"?><project xsi:schemaLocation="http://maven.apache.org/PO ...