一直想学lua,学它如何与C结合来作逻辑,所以找了云风的一份代码来研究。这份代码是个框架库,叫ejoy2d,据云风的博客说,他们最新的手机游戏用的就是这套框架,所以实用性应该很强,虽然我不是学游戏的,但应该也能学习到很多。
话不多说,开始看源码。

框架支持windows, linux, ios三种环境,这里只看windows的,用的是mingw。

程序入口main在 window.c:

  1. 创建窗口
  2. ejoy2d_win_init 用来初始化ejoy2d
  3. 进行windows消息循环

ejoy2d_win_init定义在winfw.c中,一共就下面这几步:

  1. create_game 初始ejoy2d引擎
  2. 加载初始化脚本 startscript
  3. lua_pcall这里调用了一下,调的哪个函数下面说
  4. screen_init
  5. ejoy2d_game_start

先看看create_game到底做了啥:

  1. 创建lua_State
  2. 打开一堆库,包括了所有的ejoy2d的库
  3. shader_init 初始化shader环境
  4. label_load 加载字体(不知道为啥字体叫label。。。回头细看)

再看看加载的 startscript 是个啥:

  1. local path, script = ...
  2. require("ejoy2d.framework").WorkDir = ''
  3. assert(script, 'I need a script name')
  4. path = string.match(path,[[(.*)\[^\]*$]])
  5. package.path = path .. [[\?.lua;]] .. path .. [[\?\init.lua;.\?.lua;.\?\init.lua]]
  6. local f = assert(loadfile(script))
  7. f(script)

因为最后生成的ej2d.exe调用时是 ej2d.exe xxx.lua,所以根据上面的代码,那么path中就会保存ej2d.exe的所在路径,而script则保存xxx.lua,所以上面这段脚本最后会设置package的搜索路径(这里为后面自定义脚本加载ejoy2d的lua库做好铺垫),同时执行给定的脚本。

看到这里,应该能明白上面第3步里其实就是执行的这段脚本,至于为啥不用dofile来加载startscript,是想在这里打印错误信息的原因。

行,到这儿create_game看完了,再回到ejoy2d_win_init接着往下看。

screen_init,代码里很简单,就是初始化好opengl的viewport。

ejoy2d_game_start这个函数看代码,应该是依次调了 EJOY_INITEJOY2D_UPDATEEJOY2D_DRAWFRAME这三个函数,不过这里用到了LUA_REGISTRYINDEX这个东西,lua我不熟,所以看看PIL后又结合网上其他人的说法,这是lua的“注册表”,这个东西其实也是一张表,但是一张全局的表,可以通过伪索引来搜索,貌似是比其他的方式快?

好的,问题来了:挖掘机技术哪家。。。打住,不是这个,而是EJOY_INITEJOY2D_UPDATEEJOY2D_DRAWFRAME这三个函数是从哪儿来的,源码里没有找到对应的实现,没办法,只能看看example了。

略过 examples/ex01.lua 前面的一堆代码,我们看到这个:

  1. local game = {}
  2. function game.update()
  3. end
  4. function game.drawframe()
  5. -- use shader.draw to draw a polygon to screen (for debug use)
  6. shader.draw(TEXID, {
  7. 88, 0, 88, 45, 147, 45, 147, 0, -- texture coord
  8. -958, -580, -958, 860, 918, 860, 918, -580, -- screen coord, 16x pixel, (0,0) is the center of screen
  9. })
  10. end
  11. function game.touch(what, x, y)
  12. end
  13. ej.start(game)

这里定义了一个game对象,实现了三个方法:update, drawframe, touch,最后作为参数传入了ej.start里。等等,前两个函数好像跟上面的有关系。看看ej.start(game)都干了些什么。

打开 ejoy2d/init.lua:

  1. function fw.EJOY2D_INIT()
  2. shader.init()
  3. end
  4. function ejoy2d.start(callback)
  5. fw.EJOY2D_UPDATE = assert(callback.update)
  6. fw.EJOY2D_DRAWFRAME = assert(callback.drawframe)
  7. fw.EJOY2D_TOUCH = function(x,y,what,id)
  8. return callback.touch(touch[what],x,y,id)
  9. end
  10. fw.EJOY2D_GESTURE = function(what, x1, y1, x2, y2, state)
  11. return callback.gesture(gesture[what], x1, y1, x2, y2, state)
  12. end
  13. fw.inject()
  14. end

哈,这里把刚才game中定义的三个方法都放到对应的EJOY2D_XX里了,同时调用inject函数,完成了lua函数的注入。此时再回到ejoy2d_game_start中,一切合理了。

ejoy2d源码阅读笔记1的更多相关文章

  1. CI框架源码阅读笔记5 基准测试 BenchMark.php

    上一篇博客(CI框架源码阅读笔记4 引导文件CodeIgniter.php)中,我们已经看到:CI中核心流程的核心功能都是由不同的组件来完成的.这些组件类似于一个一个单独的模块,不同的模块完成不同的功 ...

  2. CI框架源码阅读笔记4 引导文件CodeIgniter.php

    到了这里,终于进入CI框架的核心了.既然是“引导”文件,那么就是对用户的请求.参数等做相应的导向,让用户请求和数据流按照正确的线路各就各位.例如,用户的请求url: http://you.host.c ...

  3. CI框架源码阅读笔记3 全局函数Common.php

    从本篇开始,将深入CI框架的内部,一步步去探索这个框架的实现.结构和设计. Common.php文件定义了一系列的全局函数(一般来说,全局函数具有最高的加载优先权,因此大多数的框架中BootStrap ...

  4. CI框架源码阅读笔记2 一切的入口 index.php

    上一节(CI框架源码阅读笔记1 - 环境准备.基本术语和框架流程)中,我们提到了CI框架的基本流程,这里再次贴出流程图,以备参考: 作为CI框架的入口文件,源码阅读,自然由此开始.在源码阅读的过程中, ...

  5. 源码阅读笔记 - 1 MSVC2015中的std::sort

    大约寒假开始的时候我就已经把std::sort的源码阅读完毕并理解其中的做法了,到了寒假结尾,姑且把它写出来 这是我的第一篇源码阅读笔记,以后会发更多的,包括算法和库实现,源码会按照我自己的代码风格格 ...

  6. Three.js源码阅读笔记-5

    Core::Ray 该类用来表示空间中的“射线”,主要用来进行碰撞检测. THREE.Ray = function ( origin, direction ) { this.origin = ( or ...

  7. PHP源码阅读笔记一(explode和implode函数分析)

    PHP源码阅读笔记一一.explode和implode函数array explode ( string separator, string string [, int limit] )此函数返回由字符 ...

  8. AQS源码阅读笔记(一)

    AQS源码阅读笔记 先看下这个类张非常重要的一个静态内部类Node.如下: static final class Node { //表示当前节点以共享模式等待锁 static final Node S ...

  9. libevent源码阅读笔记(一):libevent对epoll的封装

    title: libevent源码阅读笔记(一):libevent对epoll的封装 最近开始阅读网络库libevent的源码,阅读源码之前,大致看了张亮写的几篇博文(libevent源码深度剖析 h ...

随机推荐

  1. Android中锁定文件的方法

    androidSDK中并没有锁定文件相关的api. 但是android是基于linux操作系统的,linux比较底层,灵活性也更大,为了实现锁定文件的效果,大概有以下几种办法: 用chmod命令修改文 ...

  2. proj.4 线程安全

    proj.4 线程安全 本文翻译自proj.4开发者文档 原文链接 http://proj4.org/development/threads.html 线程安全关键问题 全局变量pj_error是在线 ...

  3. 一堆LCT板子

    搞了一上午LCT,真是累死了-- 以前总觉得LCT高大上不好学不好打,今天打了几遍感觉还可以嘛= =反正现在的水平应付不太难的LCT题也够用了,就这样好了,接下来专心搞网络流. 话说以前一直YY不出来 ...

  4. signalR selfhost 版本兼容问题

    一.异常简要说明 最近在学习signalR,i按照http://www.asp.net/signalr/overview/deployment/tutorial-signalr-self-host 这 ...

  5. 设计模式--建造者模式Builder(创建型)

    一.首先看建造者模式的UML图: 二.再来看一个建造者模式的例子 class Builder { public: virtual void BuildHead() {} virtual void Bu ...

  6. CodeIgniter 3 源码学习笔记《一》

    CodeIgniter 是一套给 PHP 网站开发者使用的应用程序开发框架和工具包. 它的目标是让你能够更快速的开发,它提供了日常任务中所需的大量类库, 以及简单的接口和逻辑结构.通过减少代码量,Co ...

  7. 基于netty轻量的高性能分布式RPC服务框架forest<下篇>

    基于netty轻量的高性能分布式RPC服务框架forest<上篇> 文章已经简单介绍了forest的快速入门,本文旨在介绍forest用户指南. 基本介绍 Forest是一套基于java开 ...

  8. React学习笔记-5-初始化阶段介绍

    初始化阶段可以使用的函数:getDefaultProps:只调用一次,实例之间共享引用.只有在组件的第一个实例被初始化的时候,才会调用他,然后react会把这个函数的返回结果保存起来,从第二个实例开始 ...

  9. SQL注入以及如何防止和索引

    SQL注入产生的原因:程序开发过程中不注意规范书写sql语句和对特殊字符进行过滤,导致客户端可以通过全局变量POST和GET提交一些sql语句正常执行. 防止SQL注入: 1.开启配置文件中的magi ...

  10. 爬虫requests模块 1

    让我们从一些简单的示例开始吧. 发送请求¶ 使用 Requests 发送网络请求非常简单. 一开始要导入 Requests 模块: >>> import requests 然后,尝试 ...