有时我们需要对易语言编辑框的文本,进行字数统计,方法有很多,但要做到实时统计,要求处理速度够快,(如,代码写在编辑框内容被改变事件下,有两种情况会造成该事件触发非常快,一是长按粘贴快捷键内容疯狂增加,二是长按退格键,内容频频减少)

本人测试这两种行为触发内容被改变的频率是20多30毫秒左右,就是说,如果统计字数的子程序执行一次超过这个时间,程序就会卡顿。

这里列出三个方法及测试,你可根据自己的需要选择。代码数据全部基于编辑框内容被改变事件下测试,并没有用时钟及点击按钮的方式。

工具/原料

  • 电脑
  • 易语言及相关支持库(易自带,为方便,可全部勾选)

方法/步骤

  1. 1

    一、土法实现

    1-1

    用易自带最基本的几个命令实现,相对实现简单,字数上限最好在4万以下,仅支持ANSI编码文本,不支持Unnicode编码。因为文本在ansi编码下,所有字符转换到全角都占用两个字节,取其长度除2就是字符数。

     

  2. 2

    1-2 土法代码如下:

    .版本 2.支持库 spec.支持库 iext

    .子程序 笨办法统计字数, , , 用易自带基本命令实叮何排现.局部变量 全字数, 文本型, , , 去除了空格、和回车的所有字符数。.局部变量 时间, 整数型

    时间 = 取整 (取启动时间 ())全字数 = 到文本 (取整 (取文本长度 (子文本替换 (到全角 (删全部空 (编辑框1.内容)), 到全角 (#换行符), “”, , , 假)) ÷ 2))调试输出 (“用时”, 取整 (取启动时间 () - 时间), “字数”, 全字数)状态条1.置文本 (1, 全字数)

  3. 3

    二、JScript脚本命令 调用,下面是效果图。

    2-1

    执行速度快了不少,用正则实现统计中文字数的统计。这个是单线程模式。当然也可以用易的脚本组件来调用。此法字数不要超过24万左右,否则同样会卡顿。

  4. 4

    2-2  JScript脚本命令   代码如下:

    .版本 2.支持库 spec.支持库 iext

    .子程序 取字符串长度.局部变量 JS脚本, 对象, , , JScript.局部变量 时间, 整数型.局部变量 全字数, 文本型.局部变量 中文字数, 文本型

    时间 = 取整 (取启动时间 ())JS脚本.创建 (“ScriptControl”, )JS脚本.写属性 (“Language”, “JScript”)JS脚本.数值方法 (“ExecuteStatement”, #JS命令)全字数 = JS脚本.通用方法 (“Run”, “求字数全”, 编辑框1.内容).取文本 ()中文字数 = JS脚本.通用方法 (“Run”, “求字数中文”, 编辑框1.内容).取文本 ()调试输出 (“用时”, 取整 (取启动时间 () - 时间), “字数”, 全字数)状态条1.置文本 (1, 全字数)状态条1.置文本 (3, 中文字数)

    需增加个  长文本常量 这里命名为:   #JS命令

    内容在下一步。

  5. 5

    2-3 JScript脚本败敏命令    长文本常量 : #JS命令     内容如下:

    function 求字数全(参1){参1=参1.replace(new RegExp(" ","gm"),"") //先删全部空格return 参1.replace(new RegExp("[\n\r]+","gm"),"").length; //再删换行求字数};function 求字数中文(参1)//删去所有非中文求字数。{return 参1.replace(new RegExp("[^\u4e00-\u9fa5]+","gm"),"").length;};function 删除指定文本(参1,参2) //参数2可以是表达式{return 参1.replace(new RegExp(参2,"gm"),"");};

  6. 6

    三、多线程js方式。

    3-1

    易在多线程下调用JS脚本,会没有任何效果,返回空文本或返回0。

    解决办法是增加两个dll能解决(CoUninitialize 和 CoInitialize)

    运行是正常了,但新问题又来了,多线程成单线程。

    百度是这样解释这对夫妻DLL(必须配对使用)的:

    CoInitialize告诉 Windows以单线程的方式创建com对象。

    经过测试也确实是这样,同时启动多个线执行简单代码,程序立刻死掉。

    最终,解决办法是,用类模块方式。

    插入两个DLL,代码如下:

    .版本 2

    .DLL命令 卸载COM, , "ole32.dll", "CoUninitialize", , 卸载COM 同加载COM配对使用

    .DLL命令 加载COM, , "ole32.dll", "CoInitialize", , 加载COM 告诉 Windows以单线程的方式创建com对暗态象,解决多线程内JS脚本代码无效问题    .参数 pvReserved, 整数型, , 值为0

  7. 7

    3-2

    选择易主菜单-插入-类模块 ,点击,并把下面的代码粘贴进刚刚新建的类模块

    注:粘贴前删除两个默认子程序   _初始化  和  _销毁   防止重复。粘贴完修改一下类名。

    代码如下:

    .版本 2

    .程序集 JS脚本类模块.程序集变量 JS脚本, 对象

    .子程序 _初始化, , , 当基于本类的对象被创建后,此方法会被自动调用,

    加载COM (0)JS脚本.创建 (“ScriptControl”, )JS脚本.写属性 (“Language”, “JScript”)

    .子程序 _销毁, , , 当基于本类的对象被销毁前,此方法会被自动调用

    JS脚本.清除 ()卸载COM ()

    .子程序 脚本语言, 文本型, 公开, 设置或获取脚本引擎解释的语言,可供选择的属性值:JScript  VBScript.参数 语言名, 文本型, 可空, 如果,提供的参数文本,既不等于 JScript 也不等于 VBScript 会设置为 JScript

    .判断开始 (是否为空 (语言名) = 真)    返回 (JS脚本.读文本属性 (“Language”, )).默认

        .如果真 (语言名 ≠ “JScript” 且 语言名 ≠ “VBScript”)        语言名 = “JScript”    .如果真结束    返回 (到文本 (JS脚本.写属性 (“Language”, 语言名))).判断结束

    .子程序 超时, 整数型, , Timeout,设置或返回时间(毫秒),此时间后用户可选择中止脚本代码的执行或允许代码继续执行。.参数 欲写入属性值, 整数型, 可空, -1 表示直到执行完才返回。

    .判断开始 (是否为空 (欲写入属性值) = 真)    返回 (JS脚本.读数值属性 (“Timeout”, )).默认    返回 (到整数 (JS脚本.写属性 (“Timeout”, 欲写入属性值))).判断结束

    .子程序 错误信息, 对象, 公开, Error 如果脚本执行出错返回出错说明。

    返回 (JS脚本.读对象型属性 (“Error”, ))

    .子程序 脚本执行, 整数型, 公开, ExecuteStatement,返回0为执行正常,返回非0为出错,错误信息可以从“错误码信息”属性中取得。本命令为初级对象成员命令。.参数 Statement, 文本型, , 包含要执行的脚本代码,如为空则默认为执行上次的脚本代码。如果脚本代码中包括函数或过程,执行此方法后可以用“运行”方法来单独执行。

    返回 (JS脚本.数值方法 (“ExecuteStatement”, Statement))

    .子程序 脚本运行, 文本型, 公开, Run,如有返回值返回相应文本,否则返回空文本。本命令为初级对象成员命令。命令参数表中最后一个参数可以被重复添加。.参数 ProcedureName, 文本型, , 所要运行的过程或函数名。.参数 P1, 文本型, 可空, 可以被扩展。.参数 P2, 文本型, 可空.参数 P3, 文本型, 可空.参数 P4, 文本型, 可空.参数 P5, 文本型, 可空.参数 P6, 文本型, 可空.参数 P7, 文本型, 可空.参数 P8, 文本型, 可空

    .判断开始 (是否为空 (P1) = 真)    返回 (JS脚本.通用方法 (“Run”, ProcedureName).取文本 ()).判断 (是否为空 (P2) = 真)    返回 (JS脚本.通用方法 (“Run”, ProcedureName, P1).取文本 ()).判断 (是否为空 (P3) = 真)    返回 (JS脚本.通用方法 (“Run”, ProcedureName, P1, P2).取文本 ()).判断 (是否为空 (P4) = 真)    返回 (JS脚本.通用方法 (“Run”, ProcedureName, P1, P2, P3).取文本 ()).判断 (是否为空 (P5) = 真)    返回 (JS脚本.通用方法 (“Run”, ProcedureName, P1, P2, P3, P4).取文本 ()).判断 (是否为空 (P6) = 真)    返回 (JS脚本.通用方法 (“Run”, ProcedureName, P1, P2, P3, P4, P5).取文本 ()).判断 (是否为空 (P7) = 真)    返回 (JS脚本.通用方法 (“Run”, ProcedureName, P1, P2, P3, P4, P5, P6).取文本 ()).判断 (是否为空 (P8) = 真)    返回 (JS脚本.通用方法 (“Run”, ProcedureName, P1, P2, P3, P4, P5, P6, P7).取文本 ()).默认    返回 (JS脚本.通用方法 (“Run”, ProcedureName, P1, P2, P3, P4, P5, P6, P7, P8).取文本 ()).判断结束

     

  8. 8

    3-3

    创建一个进入许可证,用于多线程向状态条写入字数数据。

    创建多线程统计字数子程序,代码如下:

    .版本 2.支持库 spec.支持库 EThread.支持库 iext

    .子程序 多线程统计字数.局部变量 时间, 整数型.局部变量 文本, 文本型.局部变量 全字数, 文本型, , , 去除了空格、和回车的所有字符数。.局部变量 中文字数, 文本型, , , 只统计纯中文(标点符号也不统计在内).局部变量 脚本, JS脚本类模块

    时间 = 取启动时间 ()脚本.脚本执行 (#JS命令)全字数 = 脚本.脚本运行 (“求字数全”, 编辑框1.内容)中文字数 = 脚本.脚本运行 (“求字数中文”, 编辑框1.内容)调试输出 (“用时”, 取整 (取启动时间 () - 时间), “字数”, 全字数)进入许可区 (许可证一字数显示操作)  ' 许可证作用:防止多线程在同一时刻,写组件属性(等同于变量)。状态条1.置文本 (1, 全字数)状态条1.置文本 (3, 中文字数)退出许可区 (许可证一字数显示操作)

  9. 9

    在编辑框1.内容被改变  事件下写下代码:

    启动线程 (&多线程统计字数, , )  ' 这里并没有提供线程句柄参数,目的是让易内部处理线程句柄。

     

    至此,大功告成。

    END

方法/步骤2

  1. 1

    最近有朋友反应,写出一模一样的程序,却不统计字数,由于我的原程序没了,我试着还原,还确实费了些功夫。

    原因是从经验页面复制到易语言窗口,程序会出现错误,其实就是文本格式以及换行的问题。

    我尝试了一下,复制易代码到记事本,然后再从记事本复制粘贴到经验里边就不会出现问题。下边我把整个代码用这种方法按几个部分粘贴在下面,需要的朋友可以直接复制到易语言。

    把这部分列为第4部分,前面的也不改动了,此部分方便快速建立完整的测试程序。

    4-1  界面 这部分无法复制。

    如下图

    1个编辑框  是否允许多行 设置为真  其它根据自己喜好

    3个单选框  用于测试3个方法。

    1个状态条 添加6个项目,并将0、2、4项目按下图设置好文字。状态条用于显示测试数据,不建议用调试输出,字数太多的情况下,调试输出会占用过多资源。

  2. 2

    4-2 DLL 

    插入一条新的DLL, 如 DLL命令1 ,选中它,然后把下面文本粘贴上去。

    .版本 2

    .DLL命令 卸载COM, , "ole32.dll", "CoUninitialize", , 卸载COM 同加载COM配对使用

    .DLL命令 加载COM, , "ole32.dll", "CoInitialize", , 加载COM 告诉 Windows以单线程的方式创建com对象,解决多线程内JS脚本代码无效问题   

        .参数 pvReserved, 整数型, , 值为0

  3. 3

    4-3  常量  注意事项:不要添加成普通常量。

    1.进入常量界面,单击右键。并从右键菜单中选择  新长文本常量,创建一个长文本常量。并将常量名改名: JS命令

    2.双击 JS命令 常量值部分(此时显示为: <文本长度: 0>)进入:请输入文本窗口。

    3.把下边内容粘贴到 请输入文本窗口   确认后自动退出请输入文本窗口。

    function 求字数全(参1)

    {

    参1=参1.replace(new RegExp(" ","gm"),"") //先删全部空格

    return 参1.replace(new RegExp("[\n\r]+","gm"),"").length; //再删换行求字数

    };

    function 求字数中文(参1)//删去所有非中文求字数。

    {

    return 参1.replace(new RegExp("[^\u4e00-\u9fa5]+","gm"),"").length;

    };

    function 删除指定文本(参1,参2) //参数2可以是表达式

    {

    return 参1.replace(new RegExp(参2,"gm"),"");

    };

    4.此时,JS命令 常量值部分显示为: <文本长度: 393>

    参考下图

  4. 4

    4-4 类模块

    1.点击易语言插入菜单,选择  类模块 如:  类1

    2.把类1下的两个默认添加的2个方法,_初始化 和 _销毁删除,只留下一个类头部。

    3.把类1名称改为:  JS脚本类模块

    4.由于粘贴不会生成类成员,需要自行添加1个类成员

    点击 JS脚本类模块名称位置,然后回车,在私有成员名输入:  JS脚本

    类型输入  对象。

    4.把下面代码粘贴在   JS脚本类模块  下面 【参考下图:】

    .版本 2

    .程序集 JS脚本类模块

    .子程序 _初始化, , , 当基于本类的对象被创建后,此方法会被自动调用,

    加载COM (0)

    JS脚本.创建 (“ScriptControl”, )

    JS脚本.写属性 (“Language”, “JScript”)

    .子程序 _销毁, , , 当基于本类的对象被销毁前,此方法会被自动调用

    JS脚本.清除 ()

    卸载COM ()

    .子程序 脚本语言, 文本型, 公开, 设置或获取脚本引擎解释的语言,可供选择的属性值:JScript  VBScript

    .参数 语言名, 文本型, 可空, 如果,提供的参数文本,既不等于 JScript 也不等于 VBScript 会设置为 JScript

    .判断开始 (是否为空 (语言名) = 真)

        返回 (JS脚本.读文本属性 (“Language”, ))

    .默认

        .如果真 (语言名 ≠ “JScript” 且 语言名 ≠ “VBScript”)

            语言名 = “JScript”

        .如果真结束

        返回 (到文本 (JS脚本.写属性 (“Language”, 语言名)))

    .判断结束

    .子程序 超时, 整数型, , Timeout,设置或返回时间(毫秒),此时间后用户可选择中止脚本代码的执行或允许代码继续执行。

    .参数 欲写入属性值, 整数型, 可空,  -1 表示直到执行完才返回。

    .判断开始 (是否为空 (欲写入属性值) = 真)

        返回 (JS脚本.读数值属性 (“Timeout”, ))

    .默认

        返回 (到整数 (JS脚本.写属性 (“Timeout”, 欲写入属性值)))

    .判断结束

    .子程序 错误信息, 对象, 公开, Error 如果脚本执行出错返回出错说明。

    返回 (JS脚本.读对象型属性 (“Error”, ))

    .子程序 脚本执行, 整数型, 公开, ExecuteStatement,返回0为执行正常,返回非0为出错,错误信息可以从“错误码信息”属性中取得。本命令为初级对象成员命令。 

    .参数 Statement, 文本型, , 包含要执行的脚本代码,如为空则默认为执行上次的脚本代码。如果脚本代码中包括函数或过程,执行此方法后可以用“运行”方法来单独执行。

    返回 (JS脚本.数值方法 (“ExecuteStatement”, Statement))

    .子程序 脚本运行, 文本型, 公开, Run,如有返回值返回相应文本,否则返回空文本。本命令为初级对象成员命令。命令参数表中最后一个参数可以被重复添加。

    .参数 ProcedureName, 文本型, , 所要运行的过程或函数名。

    .参数 P1, 文本型, 可空, 可以被扩展。

    .参数 P2, 文本型, 可空

    .参数 P3, 文本型, 可空

    .参数 P4, 文本型, 可空

    .参数 P5, 文本型, 可空

    .参数 P6, 文本型, 可空

    .参数 P7, 文本型, 可空

    .参数 P8, 文本型, 可空

    .判断开始 (是否为空 (P1) = 真)

        返回 (JS脚本.通用方法 (“Run”, ProcedureName, P1).取文本 ())

    .判断 (是否为空 (P2) = 真)

        返回 (JS脚本.通用方法 (“Run”, ProcedureName, P1, P2).取文本 ())

    .判断 (是否为空 (P3) = 真)

        返回 (JS脚本.通用方法 (“Run”, ProcedureName, P1, P2, P3).取文本 ())

    .判断 (是否为空 (P4) = 真)

        返回 (JS脚本.通用方法 (“Run”, ProcedureName, P1, P2, P3, P4).取文本 ())

    .判断 (是否为空 (P5) = 真)

        返回 (JS脚本.通用方法 (“Run”, ProcedureName, P1, P2, P3, P4, P5).取文本 ())

    .判断 (是否为空 (P6) = 真)

        返回 (JS脚本.通用方法 (“Run”, ProcedureName, P1, P2, P3, P4, P5, P6).取文本 ())

    .判断 (是否为空 (P7) = 真)

        返回 (JS脚本.通用方法 (“Run”, ProcedureName, P1, P2, P3, P4, P5, P6, P7).取文本 ())

    .默认

        返回 (JS脚本.通用方法 (“Run”, ProcedureName, P1, P2, P3, P4, P5, P6, P7, P8).取文本 ())

    .判断结束

  5. 5

    4-5 启动窗口程序集部分

    1.进入启动窗口程序集,删除所有内容(包括启动窗口创建完毕事件子程序),只留一个程序集中的头部。

    2.同粘贴模块的私有成员名一样,程序集的变量名是粘贴不上去的。这里特别说明下,从一个易程序复制粘贴到另一个易程序是完美没有任何问题的。

    所以,先自行添加一个程序集变量

    变量名:许可证一字数显示操作

    类型:整数型      留空默认就是整数型

    2.把下面内容粘贴到启动窗口程序集。

    .版本 2

    .支持库 EThread

    .支持库 iext

    .程序集 窗口程序集_启动窗口

    .程序集变量 许可证一字数显示操作, 整数型

    .子程序 __启动窗口_创建完毕

    许可证一字数显示操作 = 创建进入许可证 ()

    .子程序 __启动窗口_将被销毁

    删除进入许可证 (许可证一字数显示操作)

    .子程序 笨办法统计字数, , , 用易自带基本命令实现.局部变量 全字数, 文本型, , , 去除了空格、和回车的所有字符数。.局部变量 时间, 整数型

    .局部变量 全字数, 整数型

    .局部变量 时间, 整数型

    时间 = 取整 (取启动时间 ())

    全字数 = 取整 (取文本长度 (子文本替换 (到全角 (删全部空 (编辑框1.内容)), 到全角 (#换行符), “”, , , 假)) ÷ 2)

    状态条1.置文本 (1, 到文本 (全字数))

    状态条1.置文本 (5, 到文本 (取整 (取启动时间 () - 时间)))

    .子程序 JS计算文本长度

    .局部变量 JS脚本, 对象, , , JScript

    .局部变量 时间, 整数型

    .局部变量 全字数, 文本型

    .局部变量 中文字数, 文本型

    时间 = 取整 (取启动时间 ())

    JS脚本.创建 (“ScriptControl”, )

    JS脚本.写属性 (“Language”, “JScript”)

    JS脚本.数值方法 (“ExecuteStatement”, #JS命令)

    全字数 = JS脚本.通用方法 (“Run”, “求字数全”, 编辑框1.内容).取文本 ()

    中文字数 = JS脚本.通用方法 (“Run”, “求字数中文”, 编辑框1.内容).取文本 ()

    状态条1.置文本 (1, 全字数)

    状态条1.置文本 (3, 中文字数)

    状态条1.置文本 (5, 到文本 (取整 (取启动时间 () - 时间)))

    .子程序 多线程统计字数

    .局部变量 时间, 整数型

    .局部变量 文本, 文本型

    .局部变量 全字数, 文本型, , , 去除了空格、和回车的所有字符数。

    .局部变量 中文字数, 文本型, , , 只统计纯中文(标点符号也不统计在内)

    .局部变量 脚本, JS脚本类模块

    时间 = 取启动时间 ()

    脚本.脚本执行 (#JS命令)

    全字数 = 脚本.脚本运行 (“求字数全”, 编辑框1.内容)

    中文字数 = 脚本.脚本运行 (“求字数中文”, 编辑框1.内容)

    进入许可区 (许可证一字数显示操作)  ' 许可证作用:防止多线程在同一时刻,写组件属性(等同于变量)。

    状态条1.置文本 (1, 全字数)

    状态条1.置文本 (3, 中文字数)

    状态条1.置文本 (5, 到文本 (取整 (取启动时间 () - 时间)))

    退出许可区 (许可证一字数显示操作)

    .子程序 _编辑框1_内容被改变

    .判断开始 (单选框土法统计.选中 = 真)

        清空状态条 ()

        笨办法统计字数 ()

    .判断 (单选框JS统计.选中 = 真)

        清空状态条 ()

        JS计算文本长度 ()

    .默认

        清空状态条 ()

        启动线程 (&多线程统计字数, , )

    .判断结束

    .子程序 清空状态条

    状态条1.置文本 (1, “”)

    状态条1.置文本 (3, “”)

    状态条1.置文本 (5, “”)

  6. 6

    好了,测试是OK的

    END

注意事项

  • 如果更大长度的文本,要实现实时统计字数,只有寻他法了。
  • 经测试,代码可以直接粘贴到易语言
经验内容仅供参考,如果您需解决具体问题(尤其法律、医学等领域),建议您详细咨询相关领域专业人士。