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的强大功能。

Figure 1.1 Draw Test Harness with Tcl/Tk

二、Tcl/Tk应用 OpenCascade Draw Test Harness

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

Figure 2.1 User defined command in Draw Test Harness

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,分别用来生成一个随机数和相等判断:

  1. /*
  2. * Copyright (c) 2014 eryar All Rights Reserved.
  3. *
  4. * File : Main.cpp
  5. * Author : eryar@163.com
  6. * Date : 2014-01-09 18:58
  7. * Version : 1.0v
  8. *
  9. * Description : Create new command for Tcl in C. Refer to
  10. * 1. Tcl and Tk Toolkit
  11. * 2. Practical Programming in Tcl and Tk
  12. *
  13. * Key Words : Tcl/Tk, C Interface, New Command
  14. *
  15. */
  16.  
  17. #include <tcl.h>
  18. #include <stdlib.h>
  19. #include <string.h>
  20.  
  21. #pragma comment(lib, "tcl85.lib")
  22.  
  23. /*
  24. * @breif Definitions for application-specific command procedures.
  25. */
  26. int RandomCmd(ClientData clientData, Tcl_Interp* interp, int objc, Tcl_Obj *CONST objv[])
  27. {
  28. if (objc != )
  29. {
  30. Tcl_WrongNumArgs(interp, , objv, "?range");
  31. return TCL_ERROR;
  32. }
  33.  
  34. int limit = ;
  35. Tcl_Obj* result = NULL;
  36.  
  37. Tcl_GetIntFromObj(interp, objv[], &limit);
  38.  
  39. result = Tcl_NewIntObj(rand() % limit);
  40.  
  41. Tcl_SetObjResult(interp, result);
  42.  
  43. return TCL_OK;
  44. }
  45.  
  46. int EqualCmd(ClientData clientData, Tcl_Interp* interp, int objc, Tcl_Obj *CONST objv[])
  47. {
  48. if (objc != )
  49. {
  50. Tcl_WrongNumArgs(interp, , objv, "string1 string2");
  51. return TCL_ERROR;
  52. }
  53.  
  54. Tcl_Obj* result = NULL;
  55.  
  56. char* arg1 = Tcl_GetString(objv[]);
  57. char* arg2 = Tcl_GetString(objv[]);
  58.  
  59. if (strcmp(arg1, arg2) == )
  60. {
  61. result = Tcl_NewBooleanObj();
  62. }
  63. else
  64. {
  65. result = Tcl_NewBooleanObj();
  66. }
  67.  
  68. Tcl_SetObjResult(interp, result);
  69.  
  70. return TCL_OK;
  71. }
  72.  
  73. /*
  74. * @breif Tcl_AppInit is called from Tcl_Main after the Tcl interpreter has been created,
  75. * and before the script file or interactive command loop is entered.
  76. */
  77. int Tcl_AppInit(Tcl_Interp* interp)
  78. {
  79. // Initialize packages Tcl_Init sets up the Tcl library facility.
  80. if (Tcl_Init(interp) == TCL_ERROR)
  81. {
  82. return TCL_ERROR;
  83. }
  84.  
  85. // Register application-specific commands.
  86. Tcl_CreateObjCommand(interp, "randomcmd", RandomCmd, NULL, NULL);
  87. Tcl_CreateObjCommand(interp, "equalcmd", EqualCmd, NULL, NULL);
  88.  
  89. return TCL_OK;
  90. }
  91.  
  92. int main(int argc, char* argv[])
  93. {
  94. Tcl_Main(argc, argv, Tcl_AppInit);
  95.  
  96. return ;
  97. }

程序效果如下图所示:

Figure 3.1 User defined command in Tcl

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

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

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中这种自定义类型是不支持的:

Figure 4.1 User Defined Object in PDMS PML

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

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/

Create New Commands in Tcl的更多相关文章

  1. OpenCascade Tcl vs. ACIS Scheme

    OpenCascade Tcl vs. ACIS Scheme eryar@163.com 摘要Abstract:本文通过OpenCascade的Tcl/Tk和ACIS的Scheme的对比来说明脚本语 ...

  2. [Cypress] Create a Single Custom Cypress Command from Multiple Commands

    Cypress provides a straightforward API that allows you to define custom commands. In this lesson, we ...

  3. ModelSim之tcl自动化仿真

    摘要: ModelSim的tcl最大的优势就在于它可以让整个仿真自动运行,免除每次进行各种用户界面控制操作的麻烦.用tcl就可以自动完成建库.映射库到物理目录.编译源代码.启动仿真器.运行仿真等一系列 ...

  4. 学习 java命令

    依稀记得自己第一次编译*.java文件,第一次运行*.class文件.但是六七年过去了,现在运行java写的程序更多的是用tomcat这种web容器.最近有个小需求,写一个监控zookeeper集群的 ...

  5. PostgreSQL pg_dump pg_dumpall and restore

    pg_dump dumps a database as a text file or to other formats. Usage: pg_dump [OPTION]... [DBNAME] Gen ...

  6. postgresql大批量数据导入方法

    一直没有好好关注这个功能,昨天看了一下,数据库插入有瓶颈,今天研究了一下: 主要有以下方案: 1.使用copy从文件导入: copy table_001(a, b, "f", d, ...

  7. ocp 1Z0-047 131-276题解析

    131. Which view would you use to display the column names and DEFAULT valuesfor a table?A. DBA_TABLE ...

  8. ocp 1Z0-051 23-70题解析

    23. Examine thestructure proposed for the TRANSACTIONS table: name Null Type TRANS_ID NOT NULLNUMBER ...

  9. argparse - 命令行选项与参数解析(转)

    argparse - 命令行选项与参数解析(译)Mar 30, 2013 原文:argparse – Command line option and argument parsing 译者:young ...

随机推荐

  1. src与 href 的一些区别

    src用于替换当前元素,href用于在当前文档和引用资源之间确立联系. src 是 source 的缩写,指向外部资源的位置,指向的内容将会嵌入到文档中当前标签所在位置:在请求 src 资源时会将其指 ...

  2. activity管理

    public class MainApp extends Application { private static ArrayList<WeakReference<Activity> ...

  3. Android开机启动Activity或者Service方法

    本文出自 “Bill_Hoo专栏” 博客,请务必保留此出处http://billhoo.blog.51cto.com/2337751/761230 这段时间在做Android的基础开发,现在有一需求是 ...

  4. 也学习Java/JVM/GC(四)

    GC日志分析 程序代码: public class JvmTest { public static void main(String[] args) { int m = 1024 * 1024; by ...

  5. Python成长笔记 - 基础篇 (六)python模块

    本节大纲: 模块介绍 time &datetime模块 random os sys shutil json & picle shelve xml处理 yaml处理 configpars ...

  6. mysql大数据表改表结构方案

    有一个表有上千W数据, 用什么方法给这个表加一个字段最快?1. alert2. 建一个表和第一个表一样,只是多了要加的字段,然后用多个INSERT INTO SELECT语句limit写入3. 就是导 ...

  7. CSS 禁止浏览器滚动条的方法(转)

    1.完全隐藏 在<boby>里加入scroll="no",可隐藏滚动条: <boby scroll="no"> 这个我用的时候完全没效果 ...

  8. JAVA RMI例子

    RMI 是java语言的一个RPC框架,本文给出基础例子如下: 1.实现接口: public interface ICalc extends Remote { public int add(int p ...

  9. Nim教程【七】

    这是国内第一个关于Nim的系列教程 先说废话 很开心,在今天凌晨快一点多的时候拿到了 nim-lang.com:nim-lang.cn:nim-lang.net 这三个域名,到不是为了投资,准备用ni ...

  10. jQuery 绑定事件到动态创建的元素上

    在进入主题之前,我们先来看一个前台页面经常用到的功能:点击页面输入框时自动选择其中文本. 很容易想到利用输入框的focus事件,当输入框获得焦点时,再调用jQuery的select()方法. Okay ...