Lua的文件操作

先简单介绍一下被迫使用Lua的IO的情境:

游戏支持玩家自定义上传头像,在排行榜中会显示玩家列表(包括本服、跨服),原有的做法是先检测CCUserDefault中是否存在指定图片的key以及它的状态。然后在下载头像、下载完成后设置对应的状态。这样导致的一个问题就是CCUserDefault的读写完全失效了。整个游戏下载的补丁包判断和其它判断就完全失效了,不得卸载游戏后重装。个人目前的推测是由于多线程引起的,暂时没有有效的依据

下载头像使用的是libcurl,嗯,又是它,在做项目这么久的过程中,发现它其实有很多地方比较坑。其中有一点我一直没搞明白,同样的一样地址,系统自带的浏览器(IOS、Android均支持)就能正常返回,而游戏中使用libcurl去下载就是死活返回errcode 28 (CURLE_OPERATION_TIMEDOUT),libcurl我设置的是60秒超时,绝对足够了

之后我做了优化,在设置自定义头像的时候,先检测本地是否有该文件,如果有直接就设置了,如果没有就放置在加载队列中,等下载完成后再设置头像,只开一条线程去下载图片。(同一张头像的url只请求一次,也避免对CCUserDefault的读写操作)。

通过libcurl下载一个“头像id.jpg.partial”的文件,然后下载完成重新写一个“头像id.jpg”的文件。在下载完成的时候,只做了简单的一个文件大小判断,如果文件小于300B就认为它是有问题的,直接删除相应的文件

-- filePath为当前下载完成的临时头像文件路径
local targetIconUrl = string.gsub(filePath, ".partial", "")
 
local inpFile = io.open(filePath, "rb")
local outFile = io.open(targetIconUrl, "wb")
 
if inpFile ~= nil then
    -- 最大8KB的内存
    local buffSize = 2^13
    while true do 
        local bytes = inpFile:read(buffSize)
        if not bytes then
            break
        end

        outFile:write(bytes)
    end

    inpFile:close()
end

-- 获取下载icon的大小
if outFile ~= nil then
    local current = outFile:seek()
    local fileSize = outFile:seek("end")
    outFile:seek("set", current)

    cclog("==> targetIconUrl : "..tostring(targetIconUrl)..", fileSize : "..tostring(fileSize))

    outFile:close()

    -- 小于300字节均认为不正常的数据
    if fileSize < 300 then
        FileUtil:DeleteFile_(filePath)
        FileUtil:DeleteFile_(targetIconUrl)

        self:DownloadNextIconHandler()
        do return end
    end
end

本来,直接调用对应的FileUtil中的FileRename方法就可以实现文件的重命名,但是线上的版本没有导出相应的方法,导致目前只能通过Lua的IO来实现。

最近再看lua的源码时,才真正意识到luaconf.h中定义的 LUAI_MAXCSTACK 是 cclosure的upvalue上限,而lua内存上限似乎没有找到明确的代码。

而file:read调用的是liolib.c

底层通过调用fread方法来获得文件的内容,默认每次最多读取512(LUAL_BUFFERSIZE的值)

然后调用file:seek(“end”)来获取文件大小

底层调用feek方法来实现

本以为到这里就结束了,实际上我遇到另外一个问题。如果头像因审核问题被删除了,导致404,结果底层libcurl方法没有判断http status code,直接判断CURLcode的值是否为CURLE_OK,导致将得到的文件直接写入了。但我从崩溃的日志上得到的信息是,小米4这台设备上获得的文件大小为18378

之后就直接报

invalid address or address of corrupt block 0x7c0eaa40 passed to dlfree

之后我修改了libcurl下载文件的代码,但要等下次打整包的时候才能用上

把不是jpeg的图片直接对CCSprite进行路径赋值的时候就over了,所以需要一个检测文件是否为jpeg的方法

-- 判断资源是否为jpg
function PCUtils:CheckIsJpeg(filePath)
    local isJpeg = false

    if FileUtil:CheckFileExistWithFullPath(filePath) then
        local inpFile = io.open(filePath, "rb")
        -- 读取前三位
        local bytes = inpFile:read(3)
        if bytes then
            local fileHeadIden = ""
            for _, b in ipairs{string.byte(bytes, 1, -1)} do
                local val = string.format("%02X", b)
                fileHeadIden = fileHeadIden..val
            end

            if string.upper(fileHeadIden) == "FFD8FF" then
                isJpeg = true
            else
                cclog("==> filePath : "..tostring(filePath)..", fileHeadIden : "..tostring(fileHeadIden))
            end
        end

        inpFile:close()
    end

    return isJpeg
end

读取文件的前三位,转换为16进制,然后对比JPEG的头部,判断是否为JPEG格式的文件,这个是我想起自己之前写过的一篇文章《node.js获取图片文件的真实类型

文件一些方法和代码,比如为何是r + b,以及2^13(8KB内存)这种技巧,都是参考《Lua程序设计 第二版》第21章  I/O库,网上应该有中文版的PDF下载,自行搜索吧…

本文参考:

lua cclosure 的 upvalue 数量限制

Lua编码的那些陷阱

Lua2.4 内存分配 mem.c

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏帘卷西风的专栏

关于cocos2dx程序的BUG调试解决方案

      今天说一下手机游戏开发的调试问题吧。不得不说的是和PC平台游戏、软件开发相比,手机上开发游戏和软件要困难的多。原因是多方面的,比如说开发环境比较复杂...

10720
来自专栏美团技术团队

LsLoader——通用移动端Web App离线化方案

背景 由于JavaScript(以下简称JS)语言的特性,前端作用域拆分一直是前端开发中的首要关卡。从简单的全局变量分配,到RequireJS实现的AMD模块方...

481170
来自专栏weixuqin 的专栏

基于 Django 的手机管理系统

21140
来自专栏数据和云

MySQL DBA之路 | 性能配置调优篇

一、简介 数据库服务器需要CPU、内存、 磁盘和网络才能运行,了解这些资源对于DBA来说非常重要,因为任何的超载行为都可能成为限制因素,导致数据库服务器性能不佳...

48960
来自专栏微服务生态

论代码级性能优化变迁之路(一)

大家好,很久没有和大家一起讨论技术了,那么今天我将和大家一起探讨我负责的某项目的性能变迁之路。

9220
来自专栏喵了个咪的博客空间

phalapi-进阶篇5(数据库读写分离以及多库使用)

#phalapi-进阶篇5(数据库读写分离以及多库使用)# ? ##前言## 先在这里感谢phalapi框架创始人@dogstar,为我们提供了这样一个优秀的开...

40190
来自专栏极客编程

用Node操作Firebird

  Firebird是一个跨平台的关系数据库系统,目前能够运行在Windows、linux和各种Unix操作系统上,提供了大部分SQL-99标准的功能。它既能作...

9720
来自专栏小轻论坛

再推荐大家一个很耐撕的系统优化工具

Avast Cleanup Premium 是Avast公司开发的一款强大的电脑优化清理软件,可以进行维护、加速、释放磁盘空间、修复电脑问题等四大模块,还可以删...

33550
来自专栏Linyb极客之路

浅谈高性能数据库集群——读写分离

最近学习了阿里资深技术专家李运华的架构设计关于读写分离的教程,颇有收获,总结一下。

15710
来自专栏杨建荣的学习笔记

关于switchover的流程和补充(r9笔记第4天)

对于Oracle Data Guard中的Switchover一般是计划内的操作,自己其实也处理了不少的故障,也算是轻门熟路。复杂的事情简单做,简单的事情重复做...

30350

扫码关注云+社区

领取腾讯云代金券