Create New Commands in Tcl

Create New Commands in Tcl

eryar@163.com

摘要Abstract:Tcl/Tk脚本可以很容易实现用户自定义的命令,方便的创建图形化的用户界面GUI,所以Tcl和Tk的应用领域几乎覆盖了图形和工程应用的全部范围,包括计算机辅助设计、软件开发、测试、仪器控制、科学可视化及多媒体方面。本文主要详解如何在C程序中使用Tcl来创建自定义的命令,并理解OpenCascade的Draw Test Harness的实现。 

关键字Key Words:OpenCascade, Tcl/Tk, Draw Test Harness, Software Customization 

一、引言 Introduction

Tcl是”Tool Command Language”,和Python、Perl、Lua等一样也是一种脚本语言。利用Tcl可以很容易实现自定义的命令,利用Tk可以很方便地创建出跨平台的用户界面UI。因Tcl/Tk功能强大,跨平台便于移植,且是开源免费的,所以在OpenCascade中就利用这个库来实现了测试程序Draw Test Harness,并且也利用Tcl实现了自动化测试。 

因为Tcl是解析语言,所以Tcl可用来做为程序中的二次开发语言。因为修改Tcl的脚本不需要重新编译链接,只需要重新加载下,加快开发速度。 

通过在Tcl中实现自定义的命令,来理解OpenCascade中Draw Test Harness的实现,并去感受Tcl的强大功能。 

wps_clip_image-13557

Figure 1.1 Draw Test Harness with Tcl/Tk 

二、Tcl/Tk应用 OpenCascade Draw Test Harness

在Draw Test Harness中输入自定义的命令就可以对相应的建模造型程序进行检验。理解其实现方法,也可以加深对OpenCascade的其他模块的理解。如下图所示为自定义命令: 

wps_clip_image-14583

Figure 2.1 User defined command in Draw Test Harness 

wps_clip_image-21502

Figure 2.2 Command result in Test3d view 

可以结合《OpenCascade Test Harness User’s Guide》来试试Draw Test Harness,如果将命令做在自己的程序中,就可以实现脚本建模啦。 

Draw Test Harness中的命令都是用Tcl/Tk来实现的,下面通过一个简单的示例来说明如何在Tcl中实现自定义的命令。学会Tcl脚本应该也是掌握了一种强大的脚本工具,可以为自己的程序实现二次开发功能。 

三、程序示例 Example Code

在Tcl中实现自定义命令很方便,只需要按Tcl的格式定义一个命令函数。基于对象的命令函数的声明如下:

typedef  int  (Tcl_ObjCmdProc) _ANSI_ARGS_((ClientData clientData,  Tcl_Interp  * interp,  int  objc,  struct  Tcl_Obj  *  CONST  *  objv)); 

为了能在Tcl中调用一个命令函数,必须先调用Tcl_CreateObjCommand注册它,格式如下所示:

EXTERN Tcl_Command Tcl_CreateObjCommand (Tcl_Interp  *  interp,   CONST  char   *  cmdName, Tcl_ObjCmdProc  *  proc,   ClientData clientData,   Tcl_CmdDeleteProc  *  deleteProc); 

这就是把Tcl中的字符串与实现它的C函数关联起来“魔术”。一个完整的程序如下所示,程序实现了两个自定义的命令randomcmd和equalcmd,分别用来生成一个随机数和相等判断: 

/*
*    Copyright (c) 2014 eryar All Rights Reserved.
*
*           File : Main.cpp
*         Author : eryar@163.com
*           Date : 2014-01-09 18:58
*        Version : 1.0v
*
*    Description : Create new command for Tcl in C. Refer to
*                  1. Tcl and Tk Toolkit
*                  2. Practical Programming in Tcl and Tk
*
*      Key Words : Tcl/Tk, C Interface, New Command
*                  
*/

#include <tcl.h>
#include <stdlib.h>
#include <string.h>

#pragma comment(lib, "tcl85.lib")


/*
* @breif Definitions for application-specific command procedures.
*/
int RandomCmd(ClientData clientData, Tcl_Interp* interp, int objc, Tcl_Obj *CONST objv[])
{
    if (objc != 2)
    {
        Tcl_WrongNumArgs(interp, 1, objv, "?range");
        return TCL_ERROR;
    }

    int limit = 0;
    Tcl_Obj* result = NULL;

    Tcl_GetIntFromObj(interp, objv[1], &limit);

    result = Tcl_NewIntObj(rand() % limit);

    Tcl_SetObjResult(interp, result);

    return TCL_OK;
}


int EqualCmd(ClientData clientData, Tcl_Interp* interp, int objc, Tcl_Obj *CONST objv[])
{
    if (objc != 3)
    {
        Tcl_WrongNumArgs(interp, 1, objv, "string1 string2");
        return TCL_ERROR;
    }

    Tcl_Obj* result = NULL;

    char* arg1 = Tcl_GetString(objv[1]);
    char* arg2 = Tcl_GetString(objv[2]);

    if (strcmp(arg1, arg2) == 0)
    {
        result = Tcl_NewBooleanObj(1);
    }
    else
    {
        result = Tcl_NewBooleanObj(0);
    }

    Tcl_SetObjResult(interp, result);

    return TCL_OK;
}


/*
* @breif Tcl_AppInit is called from Tcl_Main after the Tcl interpreter has been created,
*        and before the script file or interactive command loop is entered.
*/
int Tcl_AppInit(Tcl_Interp* interp)
{
    // Initialize packages Tcl_Init sets up the Tcl library facility.
    if (Tcl_Init(interp) == TCL_ERROR)
    {
        return TCL_ERROR;
    }

    // Register application-specific commands.
    Tcl_CreateObjCommand(interp, "randomcmd", RandomCmd, NULL, NULL);
    Tcl_CreateObjCommand(interp, "equalcmd", EqualCmd, NULL, NULL);

    return TCL_OK;
}


int main(int argc, char* argv[])
{
    Tcl_Main(argc, argv, Tcl_AppInit);

    return 0;
}

程序效果如下图所示: 

wps_clip_image-23159

Figure 3.1 User defined command in Tcl 

结合上面的代码来理解Draw Test Harness中自定义命令的实现可以事半功倍。当然也可以在C程序中使用Tk来定义用户界面UI。详细信息请参考References中罗列的参考资料。 

由上可知,使用Tcl/Tk可以使自己的程序实现命令行的功能,如AutoCAD、PDMS中都有这种功能,用户通过输入命令来实现一定的功能,很方便,如下图所示为PDMS中的命令窗口: 

wps_clip_image-6737

Figure 3.2 Command Window of AVEVA Plant/PDMS 

四、结论 Conclusion

通过在程序中使用Tcl实现自定义的命令,理解Tcl中C接口的用法。在此基础上来理解OpenCascade中Draw Test Harness的实现要更容易。 

通过学习Tcl/Tk可知,Tcl可以作为程序的二次开发语言,类似PDMS中的PML。Tcl中自定义命令方便,还可对表达式进行解析并求值,功能相当强大。缺点就是在Tcl中面向对象的功能要差点儿,如在Tcl脚本中自定义一个对象,像在PML中可以这样来自定义对象,而在Tcl中这种自定义类型是不支持的: 

wps_clip_image-4895

Figure 4.1 User Defined Object in PDMS PML 

好像Tcl也有对面向对象的加强库TclOO,基本用法如图所示: 

wps_clip_image-10514

Figure 4.2 Basic Usage of TclOO 

五、参考资料 References

1. Tcl and the Tk Toolkit 

2. Practical Programming in Tcl and Tk 

3. Tcl/Tk A Developer’s Guide 

4. http://sourceforge.net/projects/tcl/

5. http://www.tcl.tk/

 

转载于:https://www.cnblogs.com/opencascade/p/3512954.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
## 1 自定义命令 (1)编写自定义命令 > 1 编写自定义命令内容 > 2 注册指定命令 ```tcl /* *set up numbers sort command */ #include <stdio.h> #include <stdlib.h> #include <string.h> #include <tcl.h> // numbers sort int NumSort(ClientData ClientData, Tcl_Interp *Interp, int objc, Tcl_Obj *const objv[]) { double arr[objc], tmp; char *pstr, *pstr2; char buf[100]; int cnt = 1, cnt2 = 0, i = 0; //1. string convert number,numbers sort for(cnt = 1; cnt < objc; ++cnt) { //1.1 string convert number tmp = atof(Tcl_GetString(objv[cnt])); //1.2 numbers sort if(cnt == 1) arr[0] = tmp; else { for(cnt2 = cnt-1; cnt2 > 0; --cnt2) { if(tmp >= arr[cnt2 -1]) { arr[cnt2] = tmp; break; } else arr[cnt2] = arr[cnt2 -1]; } } } //2. numbers convert string and return for(cnt2 = 0; cnt2 < objc -1;++cnt2) { Tcl_AppendResult(Interp,gcvt(arr[cnt2], sizeof(arr[cnt2]), buf), " ",NULL); } return 0; } int Numsort_Init(Tcl_Interp *Interp) { //login command Tcl_CreateObjCommand (Interp, "numsort", NumSort, 0, 0); return TCL_OK; } ``` > 注解: (1)初始化函数Init的名字必须是文件名首字母大写+"_Init",上文c文件名是numsort.c, 故初始化函数名为Numsort_Init; ## 2 编译生成动态库 (1) 格式: gcc -fPIC -shared ____.c -o lib____.so ``` $ gcc -fPIC -shared numsort.c -o libnumsort.so ``` > 注解: (1) ____.c 代表要进行编译的C文件名 (2)lib___是lib+C文件名,生成动态库的库名为lib__.so ## 3 加载动态库 (1) 格式: load libname ```tcl % load libnumsort.so ``` ## 4 调用自定义命令 (1) commondname args ```tcl % numsort 1 1.1 2 2.2 22 3.3 90 2.5 11.11 1 1.1 2 2.2 2.5 3.3 11.11 22 90 ``` ## 1 自定义命令 (1)编写自定义命令 > 1 编写自定义命令内容 > 2 注册指定命令 ```tcl /* *set up numbers sort command */ #include <stdio.h> #include <stdlib.h> #include <string.h> #include <tcl.h> // numbers sort int NumSort(ClientData ClientData, Tcl_Interp *Interp, int objc, Tcl_Obj *const objv[]) { double arr[objc], tmp; char *pstr, *pstr2; char buf[100]; int cnt = 1, cnt2 = 0, i = 0; //1. string convert number,numbers sort for(cnt = 1; cnt < objc; ++cnt) { //1.1 string convert number tmp = atof(Tcl_GetString(objv[cnt])); //1.2 numbers sort if(cnt == 1) arr[0] = tmp; else { for(cnt2 = cnt-1; cnt2 > 0; --cnt2) { if(tmp >= arr[cnt2 -1]) { arr[cnt2] = tmp; break; } else arr[cnt2] = arr[cnt2 -1]; } } } //2. numbers convert string and return for(cnt2 = 0; cnt2 < objc -1;++cnt2) { Tcl_AppendResult(Interp,gcvt(arr[cnt2], sizeof(arr[cnt2]), buf), " ",NULL); } return 0; } int Numsort_Init(Tcl_Interp *Interp) { //login command Tcl_CreateObjCommand (Interp, "numsort", NumSort, 0, 0); return TCL_OK; } ``` > 注解: (1)初始化函数Init的名字必须是文件名首字母大写+"_Init",上文c文件名是numsort.c, 故初始化函数名为Numsort_Init; ## 2 编译生成动态库 (1) 格式: gcc -fPIC -shared ____.c -o lib____.so ``` $ gcc -fPIC -shared numsort.c -o libnumsort.so ``` > 注解: (1) ____.c 代表要进行编译的C文件名 (2)lib___是lib+C文件名,生成动态库的库名为lib__.so ## 3 加载动态库 (1) 格式: load libname ```tcl % load libnumsort.so ``` ## 4 调用自定义命令 (1) commondname args ```tcl % numsort 1 1.1 2 2.2 22 3.3 90 2.5 11.11 1 1.1 2 2.2 2.5 3.3 11.11 22 90 ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值