Redis进阶实践之八Lua的Cjson在Linux下安装、使用和用C#调用Lua脚本

一、引言

         学习Redis也有一段时间了,感触还是颇多的,但是自己很清楚,路还很长,还要继续。上一篇文章简要的介绍了如何在Linux环境下安装Lua,并介绍了在Linux环境下如何编写Lua脚本和运行。由于我们这个系列是以Redis为主,所以也介绍了Redis和Lua如何进行整合、运行。

        在Lua脚本中有一个很重要的数据类型,那就是table类型,大家可以把Lua的table类型暂时理解为数组,只是Lua的table类型的下标可以是数字,可以是字符,除了(nil)类型,其他类型都可以做为table类型的下标。我们在使用Redis和Lua的过程中,比较多的会用到这个table类型,今天的主要任务就是介绍table类型的使用、CJSON的解析和如何通过C#语言来对Redis和Lua的进行整合操作。后续我还会推出针对Lua脚本语法的文章来扩充这个系列。

二、Lua简介
    
         Lua 是一个小巧的脚本语言。其设计目的是为了嵌入应用程序中,从而为应用程序提供灵活的扩展和定制功能。Lua由标准C编写而成,几乎在所有操作系统和平台上都可以编译,运行。Lua并没有提供强大的库,这是由它的定位决定的。所以Lua不适合作为开发独立应用程序的语言。

        在Redis中使用Lua有很多好处,它可以减少网络开销,把多个操作一次性打包执行。Lua脚本天生支持原子性的操作,避免开启第三方事务,提高了性能。代码重用也是一个重要的好处,写好的代码会被加载到Redis内存中,可以供其他客户端使用,减少重复劳动。Lua脚本使用C语言写成的,执行速度很快,并天然具有可移植性,也是代码重用的很好体现。一个完整的Lua解释器,不过200k,在目前所有脚本引擎中,Lua的速度是最快的。这一切都决定了Lua是作为嵌入式脚本的最佳选择。

三、基本操作

    1、在Linux环境下安装CJSON和简单使用

        cjson是一个类型转换工具类,可以把一个字符串转换为Lua的类型,如果类型嵌套多层,还可以转换为Lua的table类型,所以这个工具很重要,在编写Lua脚本的过程中经常使用,所以我们先来介绍cjson这个工具类的安装和使用。

        1.1、下载CJSON软件包,官网地址:https://www.kyne.com.au/~mark/software/lua-cjson.php。

             [root@linux cjson]# pwd
             /root/software/download/cjson

             [root@linux cjson]# wget https://www.kyne.com.au/~mark/software/download/lua-cjson-2.1.0.tar.gz

              
                      



        1.2、解压到当前目录,当然也可以解压到其他目录,解压到其他目录,比如:/usr/local,命令如下:tar zxvf lua-cjson-2.1.0.tar.gz -C(大写的) /usr/local

             当前目录
cjson

             显示当前目录下的文件和目录
             [root@linux cjson]# ls
             lua-cjson-2.1.0.tar.gz

             解压到当前目录
             [root@linux cjson]# tar zxvf lua-cjson-0.tar.gz [-C /usr/local]
             ...
             complete!


                    

                     解压成功后的目录结构:


                     



         1.3、进入到解压目录的根目录,准备修改Makefile文件,用vim打开的修改文件的时候,注意名称大小写拼写正确。

                            [root@linux cjson]# pwd
              /root/software/download/cjson

              进入到CJSON的解压根目录
              [root@linux cjson]# cd lua-cjson-0
             
              进入到根目录,修改Makefile文件
              [root@linux lua-cjson-0]# vim Makefile

              
                       



         1,4、vim 进入Makefile文件,修改PREFIX选项,修改完毕,按ESC按钮,再按:,最后按wq,保存退出。

             当前目录
             [root@linux lua-cjson-]# pwd
             /root/software/download/cjson/lua-cjson-0

                    
                       



         1.5、使用make命令在解压文件的根目录下进行编译。

             

             [root@linux lua-cjson-]# make
             开始编译

 

                       



         1.6、最后使用 make install 安装cjson软件包。

             ]# make install
             开始安装

                     



         1.7、cjson.so文件很重要,是核心运行文件,如果出现错误,在某某目录没有找到cjson这个文件,就在其他目录找到,并拷贝到目标目录,可以解决所出现的错误。

              [root@linux program]# pwd
              /root/application/program

              拷贝sjson.so文件到指定目录
              [root@linux program]# sudo cp -r cjson/lib/lua/5.1/cjson.so  /usr/local/lib/lua/5.3


                     错误如图:

                      

                         只需要把cjson.so文件拷贝到【/usr/local/lib/lua/5.3(这个版本具体情况要看具体情况)/】目录下就可以了

            1.8、安装完成,开始测试cjson安装是否成功,此步骤有两步,必须全部都完成。

             1、从Linux命令行模式进入lua命令行测试安装是否正确
             [root@linux lua-cjson-]# lua
             Lua 5.3.4 Copyright (C) 1994-2017 Lua.org PUC-Rio
             >local cjson=require("cjson")
             >local cjson2=require cjson.safe"
             //以上所有操作没有提示任何错误

             2、我们在Linux命令行下在执行lua命令,执行lua脚本,测试是否安装成功

             05.lua脚本文件的源码
             local cjson=require 
             local smapleJson=[[{age":23",testArray":{array":[8,1)">9,1)">1,1)">14,1)">15]},1)">baiduwww.baidu.com}]];
             local data=cjson.decode(sampleJson)
             print(data[]);
             print(data["][1])
             //没有提示任何错误,安装成功

在Linux环境下执行刚才的Lua脚本
[root@linux lua-cjson-0]# lua /root/application/program/luascript/05.lua 23 8.0 安装成功

 

                       


 

    2、在Linux环境下,在Lua脚本里面使用cjson来解析类型。

 1           cjsonDemo.lua的源代码如下:
 2 
 3           local cjson = require "
 4           local retTable = {};    --最终产生json的表
 5 
 6           --顺序数值
 7 
 8           local intDatas = {};
 9           intDatas[1] = 100;
10           intDatas[2] = 10011 
12           --数组
13 
14           local aryDatas =15           aryDatas[1] =16           aryDatas[1][键11"] = 值1117           aryDatas[键12值1218           aryDatas[2] =19           aryDatas[2][键21值2120           aryDatas[键22值2221 
22           --对Table赋值
23 
24           retTable[键1值125           retTable[12326           retTable[int_datas"] = intDatas;
27           retTable[aryDatas aryDatas;
28  
29           --将表数据编码成json字符串
30 
31           local jsonStr = cjson.encode(retTable);
32           print(jsonStr);
33 
34           return retTable;
35 
36           以上为Lua源码

 

          开始执行命令
          [root@linux luascript]# pwd
          /root/application/program/luascript  lua脚本存放的位置

          [root@linux luascript]# lua cjsonDemo.lua
          {100,1)">"],1)">2123,1)">":[{"},{"}]}


          执行效果如图:
         
                 



    3、在Windows环境下,用C#执行lua脚本

            3.1、在VS2015中建一个C#控制台应用程序,并添加LuaInterface.dll的引用

                       LuaInterface下载地址:http://luaforge.net/projects/luainterface/ (下载luainterface-1.5.3,这里面的资源比较多)

                 LuaInterface.Lua类是CLR访问Lua解释器的主要接口,一个LuaInterface.Lua类对象就代表了一个Lua解释器(或Lua执行环境),Lua解释器可以同时存在多个,并且它们之间是完全相互独立的。
                   

           3.2、在Windows环境下,Lua脚本文件使用上面提到的源文件 cjsonDemo.lua。

                        文件所在地址如下:C:\Users\Administrator\Desktop\Redis\LuaScript\cjsonDemo.lua

                源码如下:

                local retTable = {};    --最终产生json的表

                --顺序数值
                local intDatas = {};
                intDatas[;
                intDatas[;

                --数组
                local aryDatas = {};
                aryDatas[;
                aryDatas[对Table赋值
                retTable[;
                retTable[ intDatas;
                retTable[ aryDatas;

                return retTable;


         3.3、测试代码如下:

           using LuaInterface;   引入的dll
           using System;
            System.Collections.Generic;
            System.Linq;
            System.Text;
            System.Threading.Tasks;

           namespace CSharpPinvokeLuaDemo
           {
               class Program
               {
                   static void Main(string[] args)
                   {
                         新建一个Lua解释器,每一个Lua实例都相互独立,一个global全局域
                        Lua lua = new Lua();
                        ---------------------------------------------------
                         Lua的索引操作[]可以创建、访问、修改global域
                        lua[20;
                        lua[nameMr.huang;

                        string luaCode = print(\"This is Lua code\");
                        lua.DoString(luaCode);执行lua脚本代码
                        object[] result=lua.DoFile(@"C:\Users\Administrator\Desktop\Redis\LuaScript\cjsonDemo.lua");执行lua脚本文件,这里我直接用了绝对定位

                        double age = (double)lua[];

                        Console.WriteLine(age = {0},age);
                        Console.WriteLine(width = {0}width]);
                        Console.ReadKey();
                   }
              }
         }


         3.4、有可能抛出的异常:FileLoadException,异常内容:其他信息: 混合模式程序集是针对“v2.0.50727”版的运行时生成的,在没有配置其他信息的情况下,无法在 4.0 运行时中加载该程序集。

                 该问题解决不是很难,只要在配置文件里增加一点配置就能解决。配置代码如下:

                <startup useLegacyV2RuntimeActivationPolicy=true">
                    <supportedRuntime version=v4.0" sku=.NETFramework,Version=v4.5.2" />
                    <supportedRuntime version=v2.0.50727"/>
                </startup>


              在原来.NET2.0,.NET3.5的时候,由于程序运行环境本质还是.NET2.0,而到了.NET4.0由于整个程序集的版本更新,以前使用.NET2.0所编写的程序集与.NET4.0的程序集继续互操作的时候就会出现上面所说的兼容性问题。通过MSDN,我们可以知道,startup配置节中的useLegacyV2RuntimeActivationPolicy属性是在.NET4.0中新增的,默认是false,表示:使用默认的 .NET Framework 4 激活策略,该激活策略将加载 .NET Framework 4 通过使用公共语言运行时 (CLR) 版本 4 所创建的程序集,以及 CLR 早期版本通过使用受支持的低于版本 4 的最高 CLR 版本所创建的程序集。

             现在如果当程序在.NET4.0环境下要使用.NET2.0及.NET3.5的程序时就必须将useLegacyV2RuntimeActivationPolicy设置为true,同时还要注意,需要在startup配置节的字节中添加supportedRuntime配置节,并指定为“v4.0”,表示使用.NET4.0运行时来运行程序。


四、结束

       今天就写到这里了,就给今天做一个总结吧。今天主要完成了lua-cjson在Linux下的安装和使用,当然也有出现问题的解决。同时也测试了在Linux环境下,在Lua脚本中使用cjson工具类完成table类型的解析。最后也测试了一下如何使用C#来调用lua脚本,并执行。但是有一个事情,还没做,就是在Windows环境下,如何在Lua脚本里使用lua-cjson来解析类型,里面涉及到的内容很多,一篇文章无法写完,只好把相关的东西放到下一片文章吧。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


文章浏览阅读752次。关系型数据库关系型数据库是一个结构化的数据库,创建在关系模型(二维表模型)基础上,一般面向于记录SQL语句(标准数据查询语言)就是一种基于关系型数据库的语言,用于执行对关系型数据库中数据的检索和操作主流的关系数据库包括Oracle、Mysql、SQL Server、Microsoft Access、DB2等非关系型数据库NoSQL(nOSQL=Not Only SQL),意思是“不仅仅是SQL”,是非关系型数据库的总称。除了主流的关系型数据库外的数据库,都认为是非关系型主流的NoSQ.._redis是非关系型数据库吗
文章浏览阅读687次,点赞2次,收藏5次。商城系统中,抢购和秒杀是很常见的营销场景,在一定时间内有大量的用户访问商场下单,主要需要解决的问题有两个:1. 高并发对数据库产生的压力;2. 竞争状态下如何解决商品库存超卖;高并发对数据库产生的压力对于第一个问题,使用缓存来处理,避免直接操作数据库,例如使用 Redis。竞争状态下如何解决商品库存超卖对于第二个问题,需要重点说明。常规写法:查询出对应商品的库存,判断库存数量否大于 0,然后执行生成订单等操作,但是在判断库存是否大于 0 处,如果在高并发下就会有问题,导致库存_php库存结余并发
文章浏览阅读1.4k次。MongoTemplate开发spring-data-mongodb提供了MongoTemplate和MongoRepository两种方式访问MongoDB,MongoRepository的方式访问较为简单,MongoTemplate方式较为灵活,这两种方式在Java对于MongoDB的运用中相辅相成。_springboot插入指定的mongodb数据库
文章浏览阅读887次,点赞10次,收藏19次。1.背景介绍1. 背景介绍NoSQL数据库是一种非关系型数据库,它的特点是可以存储非结构化的数据,并且可以处理大量的数据。HBase是一个分布式、可扩展的列式存储系统,它是基于Google的Bigtable设计的。HBase是一个开源的NoSQL数据库,它的核心功能是提供高性能的随机读写访问。在本文中,我们将对比HBase与其他NoSQL数据库,例如Redis、MongoDB、Cass...
文章浏览阅读819次。MongoDB连接失败记录_edentialmechanisn-scram-sha-1
文章浏览阅读470次。mongodb抽取数据到ES,使用ELK内部插件无法获取数据,只能试试monstache抽取mongodb数据,但是monstache需要mongodb replica set 模式才能采集数据。############monstache-compose文件。#replicas set 启动服务。# 默认备份节点不能读写,可以设置。# mydb指的是需要同步的数据库。#登录主mongodb初始化rs。#primary 创建用户。# ip地址注意要修改。# ip地址注意要修改。_monstache csdn
文章浏览阅读913次,点赞4次,收藏5次。storage:fork: trueadmin登录切换数据库注意: use 代表创建并使用,当库中没有数据时默认不显示这个库删除数据库查看表清单> show tables # 或者 > show collections表创建db.createCollection('集合名称', [options])table1字段类型描述capped布尔(可选)如果为 true,则创建固定集合。固定集合是指有着固定大小的集合,当达到最大值时,它会自动覆盖最早的文档。_mongodb5
文章浏览阅读862次。Centos7.9设置MongoDB开机自启(超全教程,一条龙)_mongodb centos开机启动脚本
文章浏览阅读1.3k次,点赞6次,收藏21次。NoSQL数据库使用场景以及架构介绍
文章浏览阅读856次,点赞21次,收藏20次。1.背景介绍1. 背景介绍NoSQL数据库是一种非关系型数据库,它的设计目标是为了解决传统关系型数据库(如MySQL、Oracle等)在处理大量不结构化数据方面的不足。NoSQL数据库可以处理大量数据,具有高性能、高可扩展性和高可用性。但是,与关系型数据库不同,NoSQL数据库没有固定的模式,数据结构也不一定是表格。在NoSQL数据库中,数据存储和查询都是基于键值对、列族、图形等不同的...
文章浏览阅读416次。NoSQL定义:非关系型、分布式、开放源码和具有横向扩展能力的下一代数据库。由c++编写的开源、高性能、无模式的基于分布式文件存储的文档型数据库特点:高性能、高可用性、高扩展性、丰富的查询支持、可替换已完场文档某个指定的数据字段应用场景:社交场景:使用mongodb存储用户信息游戏场景:用户信息,装备积分物流场景:订单信息,订单状态场景操作特点:数据量大;读写操作频繁;价值较低的数据,对事物性要求不高开源、c语言编写、默认端口号6379、key-value形式存在,存储非结构化数据。_nosql
文章浏览阅读1.5k次,点赞3次,收藏2次。Exception in thread "main" redis.clients.jedis.exceptions.JedisConnectionException: Failed to create socket. at redis.clients.jedis.DefaultJedisSocketFactory.createSocket(DefaultJedisSocketFactory.java:110) at redis.clients.jedis.Connection.connect(Conne_redis.clients.jedis.exceptions.jedisconnectionexception: failed to create so
文章浏览阅读6.5k次,点赞3次,收藏12次。readAnyDatabase(在所有数据库上都有读取数据的权限)、readWriteAnyDatabase(在所有数据库上都有读写数据的权限)、userAdminAnyDatabase(在所有数据库上都有管理user的权限)、dbAdminAnyDatabase(管理所有数据库的权限);:clusterAdmin(管理机器的最高权限)、clusterManager(管理和监控集群的权限)、clusterMonitor(监控集群的权限)、hostManager( 管理Server);_mongodb创建用户密码并授权
文章浏览阅读593次。Redis是一个基于内存的键值型NoSQL数据库,在实际生产中有着非常广泛的用处_搭建本地redis
文章浏览阅读919次。Key 的最佳实践[业务名]:[数据名]:[id]足够简短:不超过 44 字节不包含特殊字符Value 的最佳实践:合理的拆分数据,拒绝 BigKey选择合适数据结构Hash 结构的 entry 数量不要超过 1000(默认是 500,如果达到上限则底层会使用哈希表而不是 ZipList,内存占用较多)设置合理的超时时间批量处理的方案:原生的 M 操作Pipeline 批处理注意事项:批处理时不建议一次携带太多命令。Pipeline 的多个命令之间不具备原子性。_redis高级实战
文章浏览阅读1.2k次。MongoDB 递归查询_mongodb数据库 递归
文章浏览阅读1.2k次。通过实际代码例子介绍:如何通过MongoTemplate和MongoRepository操作数据库数据_springboot操作mongodb
文章浏览阅读687次,点赞7次,收藏2次。首先欢迎大家阅读此文档,本文档主要分为三个模块分别是:Redis的介绍及安装、RedisDesktopManager可视化工具的安装、主从(哨兵)模式的配置。_redis 主从配置工具
文章浏览阅读764次。天下武功,无坚不摧,唯快不破!我的名字叫 Redis,全称是 Remote Dictionary Server。有人说,组 CP,除了要了解她外,还要给机会让她了解你。那么,作为开发工程师的你,是否愿意认真阅读此心法抓住机会来了解我,运用到你的系统中提升性能。我遵守 BSD 协议,由意大利人 Salvatore Sanfilippo 使用 C 语言编写的一个基于内存实现的键值型非关系(NoSQL)..._redis 7.2 源码
文章浏览阅读2k次。MongoDB 的增删改查【1】_mongodb $inc