Lua-简洁、轻量、可扩展的脚本语言
转自:http://rmingwang.com/The-Programming-Language-Lua.html
1. 一、Lua安装及常用库的安装
2. 1.1 Ubuntu
$ sudo apt-get install lua5.2
3. 1.2 OS X
$ brew install lua luarocks # luarocks是lua的模块管理工具
$ sudo ln -s /usr/bin/make /usr/bin/gmake # 解决 sh: gmake: command not found
4. 1.3 luasocket库的安装
$ luarocks install luasocket
$ lua
> socket = require("socket")
> print(socket._VERSION)
LuaSocket 2.0.2
5. 1.4 lua-cjson库的安装
$ luarocks install lua-cjson
$ lua
> cjson = require("cjson")
> print(cjson.encode({ name = "linguofeng" }))
{"name":"linguofeng"}
6. 二、HelloWorld
$ lua
> print("Hello World!!")
Hello World!!
7. 三、数据类型 有8种基本数据类型
类型 说明
nil 全局变量默认值,如果要删除一个全局变量可以赋值为nil
boolean 布尔值
number 数字型
string 字符串型
userdata 用户自定义类型,一般是C/C++中的类型
function 函数
thread 线程
table 表
print(type(nil)) -- 输出 nil
print(type(99.7+12*9)) -- 输出 number
print(type(true)) -- 输出 boolean
print(type("Hello Wikipedia")) -- 输出 string
print(type(print)) -- 输出 function
print(type{1, 2, test = "test"}) -- 输出 table
8. 四、函数 第一类值
第一类值指:在Lua中函数和其他值(数值、字符串)一样,函数可以被存放在变量中,也可以存放在表中,可以作为函数的参数,还可以作为函数的返回值。
function add(x, y) -- 定义一个函数add,并接收两个参数
local a = x + y -- 定义一个局部变量a,接收x+y的和,局部变量仅在函数add中有效
return a -- 返回
end -- 结束add函数
print("15 + 64 = " .. add(15, 64)); -- 打印add(15, 64)的结果
local x = 1 -- local 关键字表示该变量为局部变量,作用域为当前上下文
-- 无该关键字修饰时为全局变量,作用于整个Lua状态机
local add = function(x, y) -- 局部函数,作用于当前脚本(chumk)
Lib = {}
Lib.add = function(x, y) -- 表函数,作用于Lib表
9. 函数闭包
闭包是一个内部函数以及它的upvalues,内部函数使用了外部(父函数)局部变量。
function newCounter()
local i = 0 -- i为匿名函数的外部局部变量(upvalue)
return function() -- 匿名内部函数
i = i + 1 -- 使用了i,所以该匿名函数是一个闭包
return i
end
end
c1 = newCounter() -- 得到一个匿名函数的变量(闭包)
print(c1()) -- 调用匿名函数,打印出1
print(c1()) -- 调用匿名函数,打印出2
c2 = newCounter()
print(c2()) --> 1
print(c1()) --> 3
print(c2()) --> 2
10. 五、控制语句
for int i = 0, 10, 2 do -- for循环,2表示步长,省略时为1
print("i = " .. i) -- .. 表示字符串连接符
end -- 结束for
if a > b then -- if条件判断语句
print("a > b")
else
print("b > a")
end
while a > b do -- while循环
print("")
end
repeat -- repeat-until循环
print("")
until a > b
11. 六、逻辑运算符 and、or、not
逻辑运算符认为false和nil是假(false),其他为真,0也是true.
a and b -- 如果a为false,则返回a,否则返回b
a or b -- 如果a为true,则返回a,否则返回b
x = x or v -- 如果x为false或者nil时则给x赋初始值v
-- 等价于
if not x then
x = v
end
-- 三元运算符
a ? b : c => a and b or c -- and 的优先级别比 or 高
not -- not 的结果只返回false或true,作用类似于"非" "!"取反的意思
print(not nil) -- true
print(not false) -- true
print(not 0) -- false
12. 七、协同程序 coroutine
7.1 创建协同
co = coroutine.create(function () -- 创建一个协同函数,接收一个匿名函数,返回thread类型
print("hi")
end)
print(co) -- thread: 0x7fe1834127d0
7.2 协同的三个状态:挂起态(suspended)、运行态(running)、停止态(dead)。
print(coroutine.status(co)) -- 查看协同的状态,默认状态是挂起态 suspended
coroutine.resume(co) -- 改变协同的状态为运行太 hi
print(coroutine.status(co)) -- 协同运行完以后将变量停止态 dead
7.3 如此挂起正在运行的协同
co = coroutine.create(function ()
print("hi")
coroutine.yield() -- 协同运行到此状态将变成挂起
print("你好")
end)
coroutine.resume(co) -- hi
coroutine.resume(co) -- 你好
coroutine.resume(co) -- false,协同结束后将不能再使用
7.4 协同数据交换
co = coroutine.create(function (x, y) -- 接收两个参数
print("hi", coroutine.yield(x + y)) -- 返回一个值,同时参数也传递给了coroutine.yield
return 100 -- 第三种返回值的方式
end)
print(coroutine.resume(co, 12, 87)) -- 传递两个参数并接收返回值(true, 99)
-- 执行coroutine.yield(x + y)之前协同被挂起,但值被返回,因此print函数未被执行,下面执行
print(coroutine.resume(co, 12, 87)) -- 传递两个参数并接收返回值(true, 100)
13. 八、数据结构 table
8.1 表的创建
arrays = {} -- 创建一个空表
arrays[1] = "abc" -- 第一个索引值为1
arrays[2] = 123
arrays["key"] = "value" -- map
for key, value in pairs(arrays) do -- 迭代table
print(key .. " = " .. value)
end
8.2 表的增删改查
list = {123} -- 初始化表
list[2] = "abc" -- 增
list.x = 123
list.y = 987
list[1] = nil -- 删
list.y = nil
list[2] = 456 -- 改
list.x = 987
print(list[2]) -- 查
print(list.x)
print(list['x'])
8.3 数组
list = {} -- 初始空化数组,数组的下标是整数,遵循Lua的标准,下标从1开始
list[1] = "abc"
list[2] = "edg"
list[3] = "hij"
8.4 矩阵(二维数组)
mt = {} -- 创建矩阵matrix
for i = 1, N do -- 创建N行
mt[i] = {} -- 每行都是一个数组(table元素)
for j = 1, M do -- 创建M列
mt[i][j] = "a" -- 第N行第M行的值
end
end
8.5 链表
Singly-linked-list.svg
list = nil
list = {next = list, value = "hello3"}
list = {next = list, value = "hello2"}
list = {next = list, value = "hello1"}
-- 遍历
local l = list
while l do
print(l.value)
l = l.next
end
14. 九、metatable 元表
9.1 元表与元方法
元表也是普通表
t = {}
print(getmetatable(t)) -- 获取表的metatable nil,默认不带
mt = {}
setmetatable(t, mt) -- 设置一个元素
-- metamethod 元表的方法(元方法)
mt.__add -- 加 +
mt.__sub -- 减 -
mt.__mul -- 乘 *
mt.__div -- 除 /
mt.__unm -- 负 -
mt.__pow -- 幂 ^
mt.__concat -- 连接
mt.__eq -- 等于 =
mt.__lt -- 小于 <
mt.__le -- 大于 >
mt.__tostring -- print调用
mt.__metatable -- 设置该元表不被修改与访问
mt.__index -- 当访问不存在的元素时会去查询,相当于子类继承父类一样
mt.__newindex -- 更新表,如果增加一个不存在的元素,会去查询,有直接用,否则增加
9.2 表的代理
记录下表的增查记录
local index = {} -- 私有的key,用来记录原始表在代理表中的下标
local mt = { -- 创建元表
__index = function(t, k)
print("访问了" .. tostring(k) .. "元素")
return t[index][k] -- 从代理表中获取原始表中k下标的数据
end,
__newindex = function(t, k, v)
print("更新了 " .. tostring(k) .. " 元素的值为 " .. tostring(v))
t[index][k] = v -- 更新代理表中下标为index的原始表中的元素
end
}
function setProxy(t)
local proxy = {} -- 创建代理表
proxy[index] = t -- 把原始表加到代理表的index下标中
setmetatable(proxy, mt) -- 设置代理表的元表
return proxy -- 返回代理表,即所有操作都是直接操作代理表
end
p = setProxy({})
p[2] = 'abcdefg' -- 更新了 2 元素的值为 abcdefg
print(p[2]) -- 访问了2元素
15. 十、环境
10.1 全局变量 _G
> _G["ewgegw"] = "ddddddgege"
> table.foreach(_G, print)
string table: 0x7ffce3407a60
xpcall function: 0x7ffce3404780
package table: 0x7ffce3405780
tostring function: 0x7ffce3405020
print function: 0x7ffce3405160
os table: 0x7ffce34073e0
unpack function: 0x7ffce34050d0
ewgegw ddddddgege -- 上面添加的全局变量
require function: 0x7ffce3405e70
getfenv function: 0x7ffce3404db0
setmetatable function: 0x7ffce3404f60
next function: 0x7ffce3404d20
assert function: 0x7ffce3404a80
tonumber function: 0x7ffce3404fc0
io table: 0x7ffce3406bd0
rawequal function: 0x7ffce34051b0
collectgarbage function: 0x7ffce3404ad0
getmetatable function: 0x7ffce3404e00
module function: 0x7ffce3405e20
rawset function: 0x7ffce3405260
math table: 0x7ffce3408290
debug table: 0x7ffce3408c50
pcall function: 0x7ffce3404d70
table table: 0x7ffce3405f10
newproxy function: 0x7ffce34052e0
type function: 0x7ffce3405080
coroutine table: 0x7ffce3405380 -- 对应的是协同的表
_G table: 0x7ffce3404110
select function: 0x7ffce3404ec0
gcinfo function: 0x7ffce3404150
pairs function: 0x7ffce34048c0
rawget function: 0x7ffce3405210
loadstring function: 0x7ffce3404cc0
ipairs function: 0x7ffce3404830
_VERSION Lua 5.1
dofile function: 0x7ffce3404bd0
setfenv function: 0x7ffce3404f10
load function: 0x7ffce3404c70
error function: 0x7ffce3404c20
loadfile function: 0x7ffce3404e60
> table.foreach(_G.os, print)
10.2 非全局变量 setfenv
--pack.lua---------------------------------------------------------------------
local P = {}
-- 改变P表的__index,这里的_G代表全局环境
setmetatable(P, {__index = _G})
-- 改变当前的环境为P,setfenv前的所有定义都是在全局环境中进行的,后面的则都是在新环境中进行的,互不影响
setfenv(1, P)
-- 声明的add函数在环境P中,如果要在外部访问必须P.add
function add(x, y)
print(x .. ' + ' .. y .. ' = ' .. (x + y))
-- 由于当前新的环境中没有print函数,但是__index指向的是全局环境,所以print是全局的函数
end
return P
--pack1.lua--------------------------------------------------------------------
local P = {}
-- 如果需要改变环境后使用全局环境的方法需要记住,这种方法比上面的要快
local print = print
-- 改变当前的环境为P
setfenv(1, P)
-- 声明的add函数在环境P中,如果要在外部访问必须P.add
function add(x, y)
print(x .. ' + ' .. y .. ' = ' .. (x + y))
end
-- 私有方法
local function div(x, y)
end
return P
--main.lua---------------------------------------------------------------------
local p = require 'pack'
p.add(12, 34)
local p1 = require 'pack1'
p1.add(43, 19)
16. 十一、包 package
11.1 包的定义
-- student.lua
student = {}
function student.sayHi()
print('Hello')
end
return student
-- 使用
student = require("student")
student.sayHi() -- Hello
11.2 私有函数
local function _add(x, y) -- 私有局部函数
return x + y
end
utils = { -- utils包
add = _add -- 对外公开的函数
}
return utils
17. 十二、面向对象
12.1 类与继承
--Person.lua
local Person = {name = ''}
function Person:getName()
return self.name
end
function Person:setName(name)
self.name = name
end
function Person:new(object)
object = object or {}
setmetatable(object, {__index = self}) -- 元表 类似继承的意思
return object
end
return Person
--main.lua
-- 类
local Person = require 'Person'
-- 对象
local student = Person:new({age = 23})
student:setName('Tom')
print('name: ' .. student:getName() .. ' age: ' .. student.age)
12.2 多重继承
实际就是记住所有父类,然后访问不存在的元素的时候去查询哪个父类中有,有就执行
local function search (k, plist)
for i=1, table.getn(plist) do
local v = plist[i][k] -- 去所有父类中获取
if v then return v end
end
end
function Person:new(...)
local o = {}
setmetatable(o, {__index = function (t, k)
local v = search(k, arg)
t[k] = v -- save for next access
return v
end})
return o
end
12.3 私有性
local function createAccount(_name) -- 工厂方法
local self = {name = _name}
local function _setName(name)
self.name = name
end
local function _getName()
return self.name
end
-- 公有方法表
local public = {
setName = _setName,
getName = _getName,
--name = self.name -- 不公开私有成员变量
}
return public
end
local account = createAccount('Tom')
print(account.name) -- 无法访问,因为没有公开
18. 十三、weak table
t = {name = 'table'} -- 创建一个普通表
setmatetable(t, {__made = 'k'}) -- k表示t表中的keys是weak的,v表示t表中的values是weak的
k = {} -- 创建一个空表,此时{}被k引用,引用值为1
t[k] = 1 -- 把空表作为key,由于t表的key是weak的,所以k值的引用如果为0则会被gc回收,如
k = {} -- 把k指向另一个新的{},则旧的{}引用值减1变成0了,目前t[k] = 1还有效
t[k] = 2 -- 把新的{}作为key,值是2
collectgarbage() -- 调用gc,清除引用为0的内存空间,此时,第一个{}的引用是0,会被gc掉,所以第一个t[k]以及值会被删除
for k, v in pairs(a) do print(v) end
-- 此时应该只剩下2了,因为第2个k = {}时改变了k指向新的{},而旧的{}引用会变成0,被gc掉
-------------------------------------------------------------------------
a = {};
setmetatable(a, {__mode = 'v'}); -- values为weak,当值的引用为0时,删除
v1 = {name = 'v1'}
v2 = {name = 'v2'}
a[1] = v1
a[2] = v2
v1 = v2
collectgarbage(); -- 调用GC,清掉weak表中没有引用的内存
for k, v in pairs(a) do print(v.name) end
-- 输出v2,因为v1重新指向{name = 'v2'},则{name = 'v1'}引用减1为0
-------------------------------------------------------------------------
a = {};
setmetatable(a, {__mode = 'kv'}); -- 同时检查kv,是上面两种情况的组合
v1 = {name = 'v1'}
v2 = {name = 'v2'}
v = {}
k = {}
a[1] = v
v = {} -- value重新引用新值,旧值被gc,如果旧值在table则对应的记录被清空
a[2] = v
a[k] = 1
k = {} -- key重新引用新值,旧值被gc,如果旧值在table则对应的记录被清空
a[k] = 2
collectgarbage(); -- 调用GC,清掉weak表中没有引用的内存
for k, v in pairs(a) do
print(v)
end
19. 十四、标准库
19.1. 14.1 数学库 Mathematical Functions
函数 说明
math.abs (x) 求绝对值
math.acos (x) 求反余弦
math.asin (x)
math.atan (x)
math.atan2 (y, x)
math.ceil (x)
math.cos (x)
math.cosh (x)
math.deg (x)
math.exp (x)
math.floor (x)
20. 14.2 table库
函数 说明
table.concat (table [, sep [, i [, j]]]) 拼接成字符串,sep代表连接符,i开始位置,j结束位置
table.insert (table, [pos,] value) 插入一个元素,默认是最后一个,pos指定位置
table.maxn (table) 获取最大长度
table.remove (table [, pos]) 删除一个元素,默认删除最后一个,pos指定位置
table.sort (table [, comp]) 排序
tables = {1, 2, 3, 4, 5, 6, 7}
print(#tables) -- 5.1开始使用#获取长度 -- 7
table.insert(tables, 8)
print(table.concat(tables)) -- 12345678
table.insert(tables, 1, 0)
print(table.concat(tables)) -- 012345678
print(table.maxn(tables)) -- 9
table.remove(tables)
print(table.concat(tables)) -- 01234567
table.remove(tables, 5)
print(table.concat(tables)) -- 0123567
print(table.concat(tables, ',')) -- 0,1,2,3,5,6,7
print(table.concat(tables, '-', 2)) -- 1-2-3-5-6-7
print(table.concat(tables, '=', 1, 4)) -- 0=1=2=3
table.sort(tables)
print(table.concat(tables)) -- 0123567
table.sort(tables, function(t1, t2)
if t1 > t2 then
return true
else
return false
end
end)
print(table.concat(tables)) -- 7653210
21. 14.3 string库
函数 说明
string.byte (s [, i [, j]]) 把字符转换成ASCII码
string.char (…) 把ASCII码转换成字符
string.dump (function)
string.find (s, pattern [, init [, plain]]) 查找,pattern查找的字符串,init从那里开始默认为1,plain
string.format (formatstring, …) 格式化字符串
22. 14.4 io库
函数 说明
io.close ([file]) 等效file:close(),如果没有file则关闭默认输出
io.flush () 等效file:flush()
io.input ([file])
io.lines ([filename]) 等效io.input():lines()
io.open (filename [, mode]) 打开一个文件,模式:r,w,a,r+,w+,a+
io.output ([file])
io.popen (prog [, mode]) 依赖系统的,不是所有平台都能用
io.read (…) 等效io.input():read
io.tmpfile () 创建一个临时文件,当程序退出时自动删除
io.type (obj) 判断obj的类型,如果返回file是一个打开的文件句柄,返回close file是一个
关闭的文件句柄,nil不是文件
io.write (…) 等效io.output():write
file:close () 关闭文件,会自动gc掉,但时间不确定
file:flush () 保存任何数据到文件中
file:lines () 迭代文件的每一行
file:read (…) 读取文件,*n,*a,*l,number
file:seek ([whence] [, offset]) 指定位置,默认是cur,1,set,end
file:setvbuf (mode [, size]) 设置buff缓存,no,full,line
file:write (…) 写文件,参数必须是string或者number
local file = io.open('tabletest.lua', 'r')
print(io.type(file))
for line in file:lines() do
--print(line)
end
--file:close()
io.close(file)
print(io.type(file))
------------------------------------------------------
for line in io.input('tabletest.lua'):lines() do
print(line)
end
for line in io.lines('tabletest.lua') do
--print(line)
end
23. 14.5 os库
函数 说明
os.clock () 返回程序所使用的cpu时间
os.date ([format [, time]]) 当前系统日期,或格式化某个日期
os.difftime (t2, t1) 时间差
os.execute ([command]) 执行shell命令
os.exit ([code]) 调用C的exit函数
os.getenv (varname) 获取系统环境变量,变量名,不包含$
os.remove (filename) 删除文件,文件名
os.rename (oldname, newname) 修改文件名
os.setlocale (locale [, category]) 设置地区,"all", “collate”, “ctype”, “monetary”, “numeric”, or “time”
os.time ([table]) 返回当前时间或把时间保存在table中,
os.tmpname () 临时文件名
table.foreach(os, print)
print(os.clock())
print(os.date())
print(os.date('%Y-%m-%d %H:%M'))
print(os.time())
print(os.difftime(1364957757, os.time()))
print(os.getenv ('PATH'))
print(os.tmpname ())
14.6 debug库
函数 说明
24. 十五、标准库
15.1 堆栈,后进先出原则
25. 压入
函数 说明(栈底最后一个元素的索引是1,栈顶第一个元素是-1)
lua_pushnil(lua_State*) 压入一个空值
lua_pushboolean(lua_State*, int) 压入一个布尔值
lua_pushcclosure(lua_State*, lua_CFunction, int) 压入一个C闭包?
lua_pushcfunction(lua_State*, lua_CFunction) 压入一个C函数,由lua_pushcclosure(L, f, 0)宏定义出来
lua_pushlightuserdata(lua_State*, void*) 压入一个指针,不被gc管理
lua_pushinteger(lua_State*, lua_Integer) 压入一个数字
lua_pushnumber(lua_State*, lua_Number) 压入数字
lua_pushstring(lua_State*, const char*) 压入字符串
lua_pushfstring(lua_State*, const char*, …) 压入一个格式化的string
lua_pushvfstring(lua_State*, const char*, va_list) 同上,只是接收一个va_list
lua_pushlstring(lua_State*, const char*, size_t); 压入长字符串
lua_pushliteral(lua_State*, const char*) 压入文字
lua_pushthread(lua_State*) 压入一个线程?
26. 判断类型
函数 说明
lua_isboolean(lua_State*, int) 是否是布尔类型
lua_iscfunction(lua_State*, int) 是否是C函数类型
lua_isfunction(lua_State*, int) 是否是C函数或者Lua函数
lua_islightuserdata(lua_State*, int) 是否是用户自定义类型指针
lua_isnil(lua_State*, int) 是否是空
lua_isnone(lua_State*, int) 是否是有效的
lua_isnoneornil(lua_State*, int) 是否是上面两者
lua_isnumber(lua_State*, int) 是否是数字
lua_isstring(lua_State*, int) 是否是字符串
lua_istable(lua_State*, int) 是否是table
lua_isthread(lua_State*, int) 是否是线程
lua_isuserdata(lua_State*, int) 是否是用户类型,包括full和light
lua_type(lua_State*, int) 返回元素的类型,对应LUA_TNIL等枚举
lua_typename(lua_State*, int) 返回元素的类型名称
27. 获取栈中的元素并转换成C类型
函数 说明 返回值类型
lua_toboolean(lua_State*, int) 把元素转换成C的布尔类型的值 int
lua_tocfunction(lua_State*, int) 把元素转换成C的函数 lua_CFunction
lua_tointeger (lua_State*, int) lua_Integer
lua_tolstring (lua_State*, int, size_t *len) const char*
lua_tonumber (lua_State*, int) lua_Number
lua_topointer (lua_State*, int) const void*
lua_tostring (lua_State*, int) const char*
lua_tothread (lua_State*, int) lua_State
lua_touserdata (lua_State*, int) void*
栈内元素的操作
函数 说明
lua_gettop(lua_State*) 返回栈的元素个数,同时也是栈顶元素的索引
lua_settop(lua_State*, int) 设置某个元素为栈顶元素,该元素之上的元素会被清除
lua_pushvalue(lua_State*, int) 压入(拷贝)一个已经存在栈的元素至栈顶
lua_insert(lua_State*, int) 移动栈顶元素至某个位置
lua_remove(lua_State*, int) 删除栈中某个元素
lua_replace(lua_State*, int) 替换栈顶元素至某个位置,相应那个位置的元素至栈顶
参考:http://www.lua.org/manual/5.1/manual.html
原文地址:http://blog.linguofeng.com/pages/language/lua.html
Lua-简洁、轻量、可扩展的脚本语言的更多相关文章
- 深入LUA脚本语言,让你彻底明白调试原理
这是道哥的第008篇原创 一.前言 上篇文章我们聊了gdb的底层调试机制,明白了gdb是利用操作系统提供的系统信号来调试目标程序的.很多朋友私下留言了,看到能帮助到大家,我心里还是很开心的,其实这也是 ...
- 推荐一款轻量小众却高效免费开源windows热键脚本语言Autohotkey
写在前面的话 Autohotkey是一款轻量小众但高效免费开源的windows热键脚本语言,游戏操纵.鼠标操作.键盘快捷重定义,快捷短语等等,只有你想不到,没有它做不到,神器中的神器呀,相见恨晚. 安 ...
- OWIN轻量型框架介绍
OWIN轻量型框架介绍 阅读目录 引言 框架的特色 如何启动 各项功能 静态路由的3种写法 伪静态路由的支持 处理Form表单提交的文件 流式处理Post请求的数据 多种请求类型自动识别 响应处理 请 ...
- Just 5分钟!使用k3s部署轻量Kubernetes集群快速教程
大小仅有40MB的k3s为想要节省开销进行开发和测试的企业提供了一个很好的选择.本文将用一种极为简洁的方式,教你在5分钟之内使用k3s部署轻量Kubernetes集群. Kubernetes已经改变了 ...
- 对标 Spring Boot & Cloud ,轻量框架 Solon 1.4.8 发布
Solon 是一个轻量的Java基础开发框架.强调,克制 + 简洁 + 开放的原则:力求,更小.更快.更自由的体验.支持:RPC.REST API.MVC.Job.Micro service.WebS ...
- 对标 Spring Boot & Cloud ,轻量框架 Solon 1.4.12 发布
Solon 是一个轻量的Java基础开发框架.强调,克制 + 简洁 + 开放的原则:力求,更小.更快.更自由的体验.支持:RPC.REST API.MVC.Job.Micro service.WebS ...
- Raspkate - 基于.NET的可运行于树莓派的轻量型Web服务器
最近在业余时间玩玩树莓派,刚开始的时候在树莓派里写一些基于wiringPi库的C语言程序来控制树莓派的GPIO引脚,从而控制LED发光二极管的闪烁,后来觉得,是不是可以使用HTML5+jQuery等流 ...
- 一种简单,轻量,灵活的C#对象转Json对象的方案
简单,是因为只有一个类 轻量,是因为整个类代码只有300行 灵活,是因为扩展方式只需要继承重写某个方法即可 补充:修正无法处理可空值类型的bug 首先我将这个类称之为JsonBuilder,我希望它以 ...
- 一种简单,轻量,灵活的C#对象转Json对象的方案(续)
本文参考资料 一种简单,轻量,灵活的C#对象转Json对象的方案 [源码]Literacy 快速反射读写对象属性,字段 一段废话 之前我已经介绍了这个方案的名称为JsonBuilder,这套方案最大的 ...
随机推荐
- HDU 6115 Factory LCA,暴力
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6115 题意:中文题面 分析:直接维护LCA,然后暴力枚举集合维护答案即可. #include < ...
- 动态计算文本的CGSize
// 计算文本的size -(CGSize)sizeWithText:(NSString *)text maxSize:(CGSize)maxSize fontSize:(CGFloat)fontSi ...
- python笔记_查看函数调用栈的一个小技巧
一.背景 最近在看一个开源框架的源码,涉及到的内容非常杂乱,有的函数不知道是在什么时候被谁给调用了?调用的时候传入了什么参数?为了解决这个问题,写了一个小的装饰器. 二.实现 这个装饰器函数主要参考了 ...
- linux命令(34):less命令
1.命令格式: less [参数] 文件 2.命令功能: less 与 more 类似,但使用 less 可以随意浏览文件,而 more 仅能向前移动,却不能向后移动,而且 less 在查看之前不会 ...
- 共享变量 static
一个类,有static变量counter,所有类实例共享 如果多个类实例,通过多线程访问static变量,就会产生覆盖的情况. 会发现counter偏小. 解决方法: AtomicLong count ...
- CentOS7.5***
一.借助谷歌上网助手 二.用ss来实现*** 下载工具 sudo yum install shadowsocks-libev 修改配置文件 sudo chmod 777 /etc/shadowsock ...
- 轻松学习LINUX系列教程推出
本系列多媒体教程已完成的博文: 1.轻松学习Linux之入门篇 http://chenguang.blog.51cto.com/350944/449214 2.轻松学习Linux之本地安装系统 (分区 ...
- 在 Ubuntu 系统安装 Redi
在 Ubuntu 系统安装 Redi 可以使用以下命令: $sudo apt-get update $sudo apt-get install redis-server 启动 Redis $ redi ...
- Bootstrap Table 使用示例及代码
http://issues.wenzhixin.net.cn/bootstrap-table/ <!DOCTYPE html> <html> <head> < ...
- apache 把404页面的url转发给php脚本处理
# .htaccess1 RewriteCond %{REQUEST_FILENAME} !-f 2 RewriteRule ^(.*)$ map.php?host=%{HTTP_HOST}& ...