前 言
当前WINDODS 上的编程工具日新月异,推陈出新;然而因UNIX运用的局限性,其编程工
具十分贫乏。在UNIX上要做画面、菜单、查询、输入,编程者往往要从底层入手,重复劳动
,效率极低。笔者为了解决这个问题而尽点力,将本系统介绍给您,如您能读完本文,收获
一定不少。
编者:王光红
EMAIL: wgh166@hotmail.com
软件下载:
http://www.csdn.net/cnshare/soft/16/16735.shtm
http://www.ccrun.com/program/down.asp?id=10
一。系统介简
(一)实现功能∶
用于UNIX上做画面、菜单、查询、输入、修改;可操作数据库。
(二)设计思路∶
将人机对话,可分为基本的输入与输出两大类,在此前提下,为了不同的需要把域分作
多种类型;对涉及数据库的域指明有关的表名、字段名,以进行数据库的读写。将这些
不同功能的域有机的组合在一个FORM中,来实现一个功能模块。将FORM编号,按一定的
结构以二进制的形式存入文件,在使用时,系统按编号读入FORM的数据,按步实现预定
的功能。
(三)软件环境∶
1.操作系统SCO UNIX OpenServer 5.0、SCO UNIX OpenServer 3.0、SCO UNIX 3.0。
2.编程语言 C、INFORMIX/ESQL。
说明∶笔者的工作条件仅限于以上环境,如能提供其他的UNIX、SQL数据库(如ORACLE、
SYBASE等), 系统只需略加修改。
二。系统的文件与库
(一)集成开发器执行程序∶
/usr/wform/bin/mkform FORM的定制工具
/usr/wform/bin/netform FORM的定制工具(网络版)
/usr/wform/bin/net_server FORM的数据服务器
(二)配置文件∶
FORM的配置文件的路径由环境变量CONFIGPATH指定,缺省为/usr/wform/etc/form.cfg
文本文件
它规定UNIX编辑器、FORM参数文件的路径;菜单的格式;存储变量的多少;
特殊键的键值。
(三)FORM参数文件∶
FORM参数文件的路径在配置文件中声明。
二进制文件
存放Dialog结构FORM的数据,系统读此文件,按步执行。
mkform(netform)的画面的参数文件是 /usr/wform/etc/form.bin.sys
(四)帮助信息文件:
帮助信息文件的路径由环境变量MSGPATH指定,缺省为 /usr/wform/etc/form_msg.dat
文本文件
(五)系统说明文件:
系统说明文件 /usr/wform/etc/readme.frm
(六)程序库:
/usr/wform/lib/libsetform.a 在SCO UNIX OpenServer 5.0上生成。
/usr/wform/lib/libnetform.a 在SCO UNIX OpenServer 5.0上生成(网络版)。
(七)演示例程∶
1.两个FORM参数文件∶
/usr/wform/data/demo.form
/usr/wform/data/test.form
2.菜单式演示程序∶
/usr/wform/menu/menu
/usr/wform/menu/menu.ec
/usr/wform/menu/print.c
/usr/wform/menu/errhandle.c
/usr/wform/menu/makefile
/usr/wform/menu/mknet 链接libnetform.a的makefile
3.交易式演示程序∶
/usr/wform/trade/trade
/usr/wform/trade/trade.ec
/usr/wform/trade/makefile
/usr/wform/trade/mknet 链接libnetform.a的makefile
4.两个数据库∶
/usr/wform/demo.exp
/usr/wform/test.exp
说明:演示程序menu、trade使用的数据库是test,form参数文件是test.form。
三。系统的基本概念及用途
(一)FORM的结构:
typedef struct _item_st_ {
char flag; //数据类型(或标志)
char len; //数据长度
char seat_y; //纵坐标
short seat_x; //横坐标
char fun; //FORM中嵌入的函数指针编号
char field[41]; //复合域
}Item;
typedef struct tagOfDialog {
unsigned long form_id; //交易号
unsigned long link_form; //连接交易
char mode; //驱动方式
char key; //索引项
short num; //以下Item的个数
Item *f; //FORM中域的结构
}Dialog;
(二)基本概念及用途
1.[交易号] (Dialog.form_id)
FORM的编号,以此为索引。
2.[连接交易] (Dialog.link_form)
FORM结束后调用的另一个FORM的交易号,此值=0,则不调用。
3.[驱动方式] (Dialog.mode)
以确定FORM的性质。
M∶菜单
P∶打印
B∶浏览
E∶结束清域
D∶结束删除WINDOW
S∶屏幕滚动
C∶综合查询
4.[索引项] (Dialog.key)
指定查询条件的字段数, 其值是对应于[序],序号小于等于该值的字段为查询条
件。 仅对查询类FORM有效。
以下为FORM中域的结构(Dialog.f),域的数量可变。
5.[类型] (Dialog.Item->flag)
标志符 用 途 数据类型 SQLDateType
(1) 输入输出项
C、c 普通字符串 char char
H 汉字 char char
R 日期(对字符串进行日期合法性检查) char char
A 帐号(对字符串有复制功能) char char
P 密码(不回显) char char
G 选项 char char
F 帮助项 char char
D、d 日期(自1899.12.31起至今的天数) long char
M、m 金额 double money, float
N、n 数值 long intger, smallint
以上大写的字符是输入项、小写的字符是输出项。
可在复合区指明表名和字段名, 形如table.field
(2) 杂项
t 提示
f 菜单格式
B 值域检查(对上一项值进行检查)
b 数据库(指定远程数据库),形式database@host
l 标题
i 菜单条目
x 执行 SQL语句
- 连接符
| 划线符
# 画背景符
W
当无需指定表和字段时, 复合区前四位存入坐标, 后面的字符可作标题
类型为 'l', 坐标=0, 则作表头用。
6.[长度]
对输出输出项而言是数据长度;对杂项'|','#'是代表颜色。
字符串数据长度<=128。
金额的长度 <=16∶长度是小数点前和小数点后位数总数,小数点后定为二位。
例∶长度=14,
小数点前12位,小数点后2位。
金额的长度 >=20∶
个位是小数点后位数, 个位前是小数点前的位数。
例∶长度=125,
小数点前12位,小数点后5位。
金额的长度 >=17and<20非法.
7.[坐标]
对一般的FORM后二位是横坐标。前面是纵坐标;纵坐标<24。
对于浏览器的坐标有特别规定:
当坐标值大于2499∶ 后三位是横坐标,前面是纵坐标。
横坐标的最大值512.
例∶坐标值5150表示∶ 横坐标=150,纵坐标=5。
8.[嵌入函数]
FORM中嵌入的函数指针编号,当程序执行完该域后,调用一个特定的函数,函数
指针须初始化符值。
9.[复合域]
是多用途的域,可用作写标题、SQL语句、数据库表名、字段名等。
四。实例讲解(略)
五。本系统的函数及运用
(一) 系统的定义∶
宏及FORM结构 wform.h
变量定义 wvai.h
嵌套函数指针 fun.h
(二) 系统函数的调用:
1. 菜单的调用: int SetMenu(long id)
id是主菜单的编号。
返回值是反映程序执行到菜单第几层的第几项,祥见菜单制作.
2. FORM的调用: int SetupForm(long id, short wincode)
id是FORM的编号。
wincode 是窗口的编号(0-9).
如同时用多个窗口,窗口编号须不同,否则会覆盖。
返回值:
0 取消 (即按[ESCAPE]退出)
1 正常 (即按[ENTER] 退出)
>1 按热键或自定义键返回
对某些特殊键和自定义键,即退出FORM,返回值如下∶
键 名 返回码
借方键 DEBIT
贷方键 CREDIT
向前键 PREVIOES
向后键 NEXT
热键 HOTKEY
发送键 FINISHKEY
自定义键 键值+2000
3. 其他函数∶
(1) int MsgBox(char type, const char *fmt, .../* args */);
用途∶信息提示。信息长度小于80在屏幕底线显示,否则在当前光标下组成BOX.
type∶MSGERR(0) 错误 MSGPROM(1) 提示 MSGWAIT(2) 等待 MSGWARN(3) 警告
用法同printf函数。
(2) void SetFormPara(short mode, short para);
用途∶设FORM参数。
mode:
宏 数值 作 用
WINFRAME 0 para:1 窗口画框 0 窗口无框 缺省值 0
F_COLOR 1 设定前景色 缺省值COLOR_WHITE(白色)
B_COLOR 2 设定背景色 缺省值COLOR_BLACK(黑色)
具体色彩参数见 <tinfo.h>
DISPLAYMODE 3 para:1 预显示原值
0 不显示
缺省值 0
FIELDBOUND 4 para:1 对输入项划定界符 "[......]"
0 不划
缺省值 1
DISPLAYBOLD 5 para:1 显示的字符为高亮
缺省值 0
ENTERDEFAULT 6 para:1 输入项只按[ENTER],复制成原值。
缺省值 0
LINETYPE 7 para:1 双线
0 单线
缺省值 1
PATTERNFRONT 8 para:1 运行菜单时,输入的字符与菜单选项的
第一个字符相同,能选中并自动回车。
0 不能自动回车。
缺省值 1
(3) void TouchForm(short wincode)
用途∶重显被覆盖的FORM.
(4) int SqlMsg(short sqlcode)
用途∶能提供错误信息,反映程序执行到何处(那个文件的那一行)。
通常用于SQL执行错误返回。
(5) void DelDlgwin(short wincode)
用途∶删除FORM.
(6) int SkipDomain(short step)
用途∶跳过若干个域(step)。用于嵌入函数。
(7) void Rfmtdate(date i_date, char *fmt, char *result)
用途∶将INFORMIX中的date转成char.
例∶
1899.12.31至1999.12.4的天数是36132.
Rfmtdate(36132, "mm-dd-yyyy", result);
result="12-04-1998"
Rfmtdate(36132, "mm-dd-yy", result);
result="12-04-98"
Rfmtdate(36132, "日期: YYYY年MM月DD日", result);
result="日期: 一九九八年十二月四日".
(8) void Rfmtdouble(double money, char *fmt, char *result)
用途∶将double转成char.
例∶
double a=-9030236.035;
Rfmtdate(a, "---,---,--9.999", result);
result=" -9,030,236.035"
(9) char *DoubleToStr(double money, const char *fmt);
用途∶将double转成char返回(char *).
(10) void PrintForm(short wincode, short lin, short col, char *fmt, ...);
用途∶向FORM输送字符.
(11) int RunFunction(int (*fun)(), char *msg);
用途∶运行函数fun时,在屏幕底显示信息msg,隔一秒钟跳动字符 '>'。
例∶
见menu.ec的280行。
RunFunction(DeleteDetail, "正在清理数据");
如果函数DeleteDetail()不在menu.ec中,
必须声明extern int DeleteDEtail();
(12) int CallFunction(int (*fun)(), char *msg);
用途∶同RunFunction(),不跳动字符。
(13) char *PntName();
用途∶获得打印输出的设备名。
在主控台上得到"/dev/lp0",在终端上得到相应的终端号"/dev/tty???"
(14) void Draws(WINDOW *w, int b_y, int b_x, int e_y, int e_x)
用途∶在指定的窗口画线。
起点坐标(b_x, b_y) 终点坐标(e_x, e_y)
b_x=e_x 画垂直线。
b_y=e_y 画水平线。
b_x!=e_x b_y!=e_y 画矩形。
(15) int FormPath(char *path);
用途∶
调用该函数设置FORM的参数的路径, 确省路径在配置文件中设定。
(16) InsetFun(n, fun);
用途∶
函数指针初始化。
n 是函数指针的编号,fun是函数名.
(三) 关于嵌入函数的使用:
作用∶当FORM执行到某一域时调用的函数.
我们通过menu.ec这个程序来说明嵌入函数的运用。
****注意∶以下是test.form中的 110号form.
[命令] g [交易号] 110 [连接交易] 0 [驱动方式] E [索引项] 0
[序] [类型][长度][坐标][嵌入函数] [ 复 合 域 ]
0 C 3 1237 0 bank.exchno
1 c 20 1337 0 bank.officename
2 R 8 1537 1
3 t 0 0 0 日期须本日前一星期内
4 A 7 1737 0
5 M 10 1937 0
6 B 0 0 0
7 i 0 0 0 62.50
8 i 0 0 0 999999.99
9 C 1 2137 0
10 B 0 0 0
11 i 0 0 0 [1,4]
12 l 0 1226 2 受 理 行
13 l 0 1326 0 行 名∶
14 l 0 1526 0 凭证日期
15 l 0 1726 0 号 码
16 l 0 1926 0 金 额
17 l 0 2126 0 联 次
18 l 0 2224 0
第 2、12域有嵌入函数∶1、2是函数指针的编号,分别代表函数CheckDate()、Pattern().
函数指针初始化∶
InsetFun(1, CheckDate);
InsetFun(2, Pattern);
见menu.ec中第217、218行。
当程序执行完第 2域(即输入日期),系统就调用函数CheckDate(),以检查日期
是否本日至前一星期内,如日期正确函数返回 0,程序继续执行; 如日期错误函
数返回-1, 程序仍停止在第 2域。如函数返回-3程序退回到第 0域。
设SN为程序执行的序号.
ret=Fun[n]();
SN +=ret;
你可根据需要函数返回值,如返回值为2,就会跳过2个域。
当程序执行完第11域,系统就调用函数Pattern(),进行凭证配对。
(四)关于FORM的值∶
1.类型.
类 型 变量名 宿主变量名 FORM编辑器的代名
字符(char) Chars(n) $INFString[n] $Cn
整型(long) Number[n] $Number[n] $Nn
金额(double) Money[n] $Money[n] $Mn
日期(long) Date[n] $Date[n] $Dn
注∶
n是顺序号
Chars(n)是宏,C语言中Chars(n)、INFString[n]都可用;
SQL语言中只能用INFString[n],不能用宏Chars(n);
2.值的配置.
系统根据配置动态申请内存,按你的需要调整系统配置(或修改form.cfg),
确定值的个数.
3.值的顺序.
用下例来说明∶
[序] [类型][长度][坐标] 对应的变量 注 解
0 C 3 1237 Chars(0) 'C'字符型第 1 次出现
1 c 20 1337 Chars(1) 'c'字符型第 2 次出现
2 H 8 1537 Chars(2) 'H'字符型第 3 次出现
3 A 7 1737 Chars(3) 'A'字符型第 4 次出现
4 M 10 1937 Money[0] 'M'金额型第 1 次出现
5 B 0 0 控制类与变量无关
6 i 0 0 控制类与变量无关
7 i 0 0 控制类与变量无关
8 c 11 2120 Chars(4) 'c'字符型第 5 次出现
9 B 0 0 控制类与变量无关
10 i 0 0 控制类与变量无关
11 m 12 2126 Money[1] 'm'金额型第 2 次出现
12 F 1 2146 Chars(5) 'F'字符型第 6 次出现
13 i 0 0 控制类与变量无关
14 i 0 0 控制类与变量无关
15 N 6 2170 Number[0] 'N'长整型第 1 次出现
六。使用FORM的用户环境
用户可设置以下环境变量
(1)CONFIGPATH: FORM的配置文件的路径。缺省为 /usr/wform/etc/form.cfg
FORM的参数的路径在配置文件中设定。
用户的程序中可在启动FORM前,调用FormPath(char *path)设定路径。
(2)MSGPATH: 帮助项信息文件路径。缺省为 /usr/wform/etc/form_msg.dat
(3)NETCONFIGFILE:通讯的配置文件,适用于本系统网络版,客户及服务端都
需要,缺省为 /usr/wform/etc/sys.net
七。FORM生成器(mkform、netform)的使用
_.----..__.'
_ / -._
__ ``-._ ) | ||..\\\ .\ \`-
-'_ `-._ /( `-.\ / // | /-'' ``-._| \`-. .`. / | |\ \\ | | /_.---.__.| `_/``' `-..-'_/_/ /_/ |//\ _..--'/ /' -._ .'"(@ ` @`/-._ \ ( ' /\ `-.` `(| _' ;`-. ` `_ `.
/ / ``-._ _.'| _ ` ,' '_.-` .'
/ /-------- ``'-~-' `;`~'`` . '
'._ \\ ` \\ / . '
```` --- .. -. \\ \'
`\ \ \\ \ `. | || . || ┏━━━━━选择━━━━━┓
\ . || .' `-|| ┃ V 全屏幕编辑_______(0)┃
\ `||' ;|| ┃ T 交易式演示程序___(1)┃
;|| '// ┃ M 菜单式演示程序___(2)┃
_; || // [ ]┃ D 显示系统配置_____(3)┃
/.``||..__.-; ┃ A 调整系统配置_____(3)┃
.'` ``-._|_|.-| ┃ S 清理FORM_________(5)┃
/ ; ┃ P 改变FORM参数路径_(6)┃
; ; ┃ L 登记注册_________(7)┃
; ; ┃ E 退出_____________(8)┃
; ,' ┗━━━━━━━━━━━━┛
------------ 图10 --------------
(一)运行mkform(或netform)
mkform的用法∶mkform -d database -f filename
1.打开数据库∶ -d database 可省略(不访问数据库)。
2.指定FORM参数文件名∶-f filename 如省略则按系统配置文件中指定的路径。
mkform与netform的主要区别:mkform对远程数据库的访问是利用了informix本
身的功能,也就是由sqlexecd提供服务;netform访问数据库的服务程序(net_server)由
作者编写,在数据安全方面进行了控制,所以在运行netform的同时,必须启动net_server
服务程序。
(二)菜单功能
当你运行mkform时,就出现图10,下面就菜单的功能逐一介绍∶
1. V 全屏幕编辑
进入FORM的编辑、定制程序。(祥见以下关于FORM编辑的一章)。
2. T 交易式演示程序
演示以交易码驱动的程序。
3. M 交易式演示程序
演示以菜单驱动的程序。
4. D 显示系统配置(图11)
配置文件的路径由环境变量CONFIGPATH指定,缺省为 /usr/wform/etc/form.cfg
┏━━━━━━━━━━━ 系 统 配 置 ━━━━━━━━━━━┓
┃ 0. FORM参数路径: ../data/demo.form ┃
┃ 1. 编缉命令: /usr/bin/vi ┃
┃ ┃
┃ 2. 字符型(char)个数: 50 3. 数值型(long)个数: 10 ┃
┃ 4. 金额型(double)个数: 10 5. 日期型(date)个数: 8 ┃
┃ ┃
┃ 6. 菜单最大项数: 100 ┃
┃ 主菜单格式: ( 7. 2 8. 5 ) ┃
┃ 子菜单格式: ( 9. 12 10. 1 ) ┃
┃ ┃
┃ 11.双零键: 46 12.四零键: 47 ┃
┃ 13.五零键: 42 14.六零键: 45 ┃
┃ 15.借方键: 3 16.贷方键: 4 ┃
┃ 17.向前键: 91 18.向后键: 93 ┃
┃ 19.热 键: 26 20.发送键: 10 ┃
┃ ┃
┃ 选择: [ ] (Esc取销) ┃
┃ ┃
┃ ┃
┗━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┛
------------ 图11 --------------
5. A 调整系统配置
只有root用户才能调整系统配置。
编缉命令: 编辑FORM所调用的文本编辑程序。
主菜单格式: ( 7. 2 8. 5 )
该项配置说明主菜单格式是 2行, 5列。
选择第 7项修改行数,选择第 8项修改列数。
子菜单格式: ( 9. 12 10. 1 )
该项配置说明主菜单格式是12行, 1列。
选择第 9项修改行数,选择第10项修改列数。
6. S 清理FORM
对作废和删除的FORM定期清理。
7. P 改变FORM参数路径
FORM参数路径在mkform启动时或系统配置文件中指定,如果你需要其他的FORM
参数文件可由此选项改变。
8. L 登记注册。
9. E 退出系统。
(三)关于FORM编辑
在图10中的菜单选第 0项,就进入图12。
┏━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┓
┃ [命令] ┏━━━━━━━┓ [连接交易] 0 [驱动方式] [索引项] ┃
┃ [序] [类型┃a 进入___(00) ┃函数] [ 复 合 域 ]┃
┃ 0 ┃e 退出___(01) ┃ ┃
┃ 1 ┃g 取数___(02) ┃ ┃
┃ 2 ┃h 帮助___(03) ┃ ┃
┃ 3 ┃s 演示___(04) ┃ ┃
┃ 4 ┃u 恢复___(05) ┃ ┃
┃ 5 ┃d 设计___(06) ┃ ┃
┃ 5 ┃v 编辑___(07) ┃ ┃
┃ 6 ┃w 存盘___(08) ┃ ┃
┃ 7 ┃F 单选___(09) ┃ ┃
┃ 8 ┃D 删除___(10) ┃ ┃
┃ 9 ┗━━━━━━━┛ ┃
┃ 10 ┃
┃ 11 ┃
┃ 12 ┃
┃ 13 ┃
┃ 14 ┃
┃ 15 ┃
┃ 16 ┃
┃ 17 ┃
┃ 18 ┃
┃ 19 ┃
┗━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┛
------------ 图12 ------------
图12中的命令菜单功能∶
1. a 进入本系统的FORM编辑器,编辑的窗口每屏是20项,共 5屏;
当光标到最后一行,再按↓键,就可翻到下一屏。
2. e 退出本编辑器。
3. g 取数
输入交易号就可读到此交易的FORM参数。
4. h 帮助
显示文本/usr/wform/etc/readme.frm.
5. s 演示FORM
输入交易号就可此交易的FORM框架,演示输入、查询的过程。
6. u 恢复被误删除的FORM.
7. d 设计
利用UNIX编辑工具,对FORM进行初步的定制。具体的方法详见FORM的定制。
7. v 编缉
利用UNIX编辑工具,对FORM的参数进行设定。具体的方法详见FORM的编辑。
9. w 存盘
将编辑或修改完的FORM存入文件。原来同号的FORM被作废,故需定期清理
FORM。
10. F 单选
可单独选出一个指定的FORM参数,文件名"./f????", ?表示FORM号。
作用∶有些FORM的功能通用,可移到其他用户的FORM参数文件中。
移值方法∶ cat f??? >> other.form
11. D 删除
对弃用的FORM进行删除。
(四)FORM的定制:
1. 利用UNIX编辑工具,对FORM进行初步的定制。
(1) 选命令 v,输入交易号,(假设交易号是88)。系统就调用UNIX编辑工具
(如vi),生成文本/tmp/form88,在文本在写的内容如下∶
序号 文 本 内 容
1
2
3
4
5 记 帐
6
7
8 帐 号∶ $A12
9
10 凭证号∶ $N6
11
12 摘 要∶ $C10
13
14 金 额∶ $M12
------------ 图13 ------------
在'$'符后的字符是类型,类型后的数字是长度。
将该文本存盘退出后,即返回系统画面,系统根据文本内容及各个域的
坐标,在内存生成结构为Dialog的FORM,见图14.
┏━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┓
┃ [命令] g [交易号] 88 [连接交易] 0 [驱动方式] [索引项] 0 ┃
┃ [序] [类型][长度][坐标][嵌入函数] [ 复 合 域 ]┃
┃ 0 A 12 739 0 ┃
┃ 1 N 6 939 0 ┃
┃ 2 C 10 1139 0 ┃
┃ 3 M 12 1339 0 ┃
┃ 4 l 0 437 0 记 帐 ┃
┃ 5 l 0 730 0 帐 号∶ ┃
┃ 6 l 0 930 0 凭证号∶ ┃
┃ 7 l 0 1130 0 摘 要∶ ┃
┃ 8 l 0 1330 0 金 额∶ ┃
┃ 9 0 0 0 ┃
┃ 10 0 0 0 ┃
┃ 11 0 0 0 ┃
┃ 12 0 0 0 ┃
┃ 13 0 0 0 ┃
┃ 14 0 0 0 ┃
┃ 15 0 0 0 ┃
┃ 16 0 0 0 ┃
┃ 17 0 0 0 ┃
┃ 18 0 0 0 ┃
┃ 19 0 0 0 ┃
┗━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┛
------------ 图14 ------------
用命令 w存盘,再用命令 s进行演示,见图15。如果对FORM不满意,可通过对
文本修改或直接修改FORM,直至满意。
┏━━━━━━━━━━━━━━┓
┃ 记 帐 ┃
┃ ┃
┃ ┃
┃帐 号∶[ ] ┃
┃ ┃
┃凭证号∶[ ] ┃
┃ ┃
┃摘 要∶[ ] ┃
┃ ┃
┃金 额∶[ ] ┃
┗━━━━━━━━━━━━━━┛
------------ 图15 ------------
(五)FORM的编辑:
用UNIX编辑只是FORM的初步定制,它只能对标题、数据的输入、显示域进行定义。
而对[连接交易]、[驱动方式]、[索引项]、[嵌入函数]以及菜单、数据库的表名、 字段名等等,需本系统提供的编辑功能或直接用vi编辑。
方法一____系统提供的编辑功能:
(1) 用命令 a进入,用光标键移到需改动的域。
(2) 行的复制功能.
把A 行开始的N 条复制到B 行上,方法如下∶
将光标移至A 行按F2键,再按要复制的条数N,再将光标移到B行按 F2键。
连续按两次F2,可在原行下复制一行。
(3) 行的插入功能.
按[Insert]键,可增加一行空行。
(4) 行的删除功能.
将删除行的类型改为空格,存盘时,会剔除该行。
(5) 复合域的编辑功能。
F1键向右移,[Backspace]键向左移。
命令切换键F3或[Delete],x删除字符,i插入字符,功能同vi.
方法二___利用vi编辑功能:
(1) 用命令 v进入、再输入form号。
(2) 如form是新建的,系统生成一个带有说明和一定格式的文本(见图16),由您
填写编辑。
如form已存在,系统生成一个带有原form参数的文本,由您修改。
[交易号] 120 [连接交易] 0 [驱动方式] B [索引项] 3
[类型][长度][坐标][嵌入函数] [ 复 合 域 ]
~~提示:以下值的横坐标必须在对应字段的区域内(以'[]'为界),否则无效!空缺值为零~~
------------ 图16 ------------
九。演示程序
演示程序menu、trade的功能相同;共用数据库test、FORM参数文件test.form.
你可阅读menu.ec、trade.ec中是如何调用本系统函数的;
你还可参考./menu/makefile、mknet ./trade/makefile、mknet;
十。有关网络版
所谓网络版就是在原版的基础上, 将程序分为前台程序(界面控制程序)和后台程序 (数据服
务器), 前台把FORM中涉及数据数据库的FORM, 组织成SQL语句通过网络传给后台, 后台把
SQL 的着执行结果返回前台, 如图16所示:
┏━━━━━┓ SQL语句送后台 ┏━━━━━━━━━┓
┃ ┃ ────────────→ ┃ ┃
┃ 前 台 ┃ ........TCP/IP........... ┃ 后台(net_server) ┃
┃ client ┃ ←──────────── ┃ server ┃
┗━━━━━┛ 查询、更新结果返回前台 ┗━━━━━━━━━┛
------图16-----
查询的核心代码:
Select(int tag, char *command)
{
$char *sel, Str[128];
$short flag;
$date Date;
$long Number, b_count, j;
$long prec;
$long type;
$long scale;
$double money;
$char result[128];
int ret, offset=0;
int i=0;
char fmt[32], Tag;
if(DisplayFlag) printf("tag=%c\nsql=%s\n", tag, command);
RetPacket.txcode=SELECT;
if(tag=='M'){
//多项查询
sel=getfield(command, &offset);
if(strlen(sel)<2) goto EXIT1;
Tag=sel[0];
}
else
Tag=tag;
sel=getfield(command, &offset);
if(tag=='M') ChangeCmd(sel);
if(strlen(sel)<10) goto EXIT1;
if((ret=ComposeBrowse(10, sel))<0){
SqlErrHandle(SQLCODE, SELECT);
goto EXIT0;
}
RetPacket.txcode=SELECT;
$get descriptor 'browsdesc' :b_count=count;
if(SQLCODE){
SqlErrHandle(SQLCODE, SELECT);
goto EXIT0;
}
$fetch BROWSE using sql descriptor 'browsdesc';
if(SQLCODE){
strcpy(RetPacket.data, "SELECT ERROR|");
SqlErrHandle(SQLCODE, SELECT);
goto EXIT0;
}
for(j=1;j<=b_count;j++){
$get descriptor 'browsdesc' value $j
$prec=precision, /*money、decimal*/
$scale=scale, /*money、decimal*/
$type=type;
if(SQLCODE){
strcpy(RetPacket.data, "SELECT ERROR|");
SqlErrHandle(SQLCODE, SELECT);
goto EXIT0;
}
switch(type){
case SQLFLOAT:
case SQLSMFLOAT:
case SQLDECIMAL:
case SQLMONEY:
if(prec)
sprintf(fmt, "%s%d.%df\0", "%",prec+1, scale);
else
strcpy(fmt, "%f");
$get descriptor 'browsdesc' value $j
$money=data;
sprintf(result, fmt, money);
if(SQLCODE==DATAISNULL) result[0]=0;
break;
default:
$get descriptor 'browsdesc' value $j
$result=data;
}//switch
DelTailSpace(result);
if(SQLCODE==DATAISNULL) SQLCODE=0;
if(SQLCODE){
strcpy(RetPacket.data, "SELECT ERROR|");
SqlErrHandle(SQLCODE, SELECT);
goto EXIT0;
}
strcat(RetPacket.data, result);
strcat(RetPacket.data, "|");
}//for
if(DisplayFlag) puts(RetPacket.data);
EXIT0:
$close BROWSE;
$free BROWSE;
$deallocate descriptor 'browsdesc';
EXIT1:
return TRUE;
}
(一)网络版的作用:
增强安全性。INFORMIX能访问远程数据库,但须建立对等关系,会造成不安全。
本版无须建立对等关系,克服了这个弱点。
(二)网络版的用法:
1. 网络通讯使用TCP/IP协议。
2. 前、后台程序也可在同台机器上.
3. FORM的定制、函数及功能不变.
4. 链接的程序库: /usr/wform/lib/libnetform.a。
5. 通讯的配置文件: $HOME/etc/sys.net的内容:
(1)数据库名:对后台而设
[Database]
Database=mobile
(2)后台机器名:前、后台须一致。/etc/hosts中存在,也可以直接写IP地址。
[ServerHost]
ServerHost=save01
(3)端口号:前、后台须一致。
[NetForm Tcp Port]
NetFormTcpPort=8168
(4)socket超时报警时间(秒),确省值60
[Socket Read & Write Time Out(s)]
SocketTimeOut=35
(5)查询超时报警时间(秒),确省值60, 仅对前台而设。
[Wgetch Time Out(s)]
WgetchTimeOut=30
(6)跟踪标志,仅对前台而设,0屏蔽,1开放。跟踪文件:$HOME/run/trace
[Trace Flag]
TraceFlag=0
(7)connect超时报警时间(秒) 仅对前台而设。
[Company Server Connect Time Out(s)]
ConnectTimeOut=8
(8) fork()服务子进程同时存在的最大数, 仅对后台而设。
[Max User Number]
UserNum=10
(9)显示部分服务程序接受和发送的数据。0屏蔽,1开放
[Display Server Data Flag]
DisplayFlag=1
6. 地址簿文件: $HOME/etc/hosts.equ 设在后台
记录允许访问数据库的前台机器的IP地址,未记录的机器不能访问。
设置方法:
(1)文件中加入all.hosts,对所有的客户开放。
(2)文件中加入网段地址(如13.7.5),对该网段开放。
(3)文件中加入完整的客户端IP地址(如12.17.23.76),对该客户机开放。
7. 服务器的启动:net_server
8. 服务器的停止:net_server stop
9. 返回值∶ 当sqlcode>-10时是本系统定义,含义如下∶
-1 收数据包出错: 如超时。
-2 拒绝服务∶ 地址簿无本机IP地址
-4 服务进程超过限定的数目。