自由学习记录(25)

只要有修改,子表就不用元表的参数了,用自己的参数(只不过和元表里的那个同名)

子表用__index“继承”了父表的值,此时子表仍然是空表

一定是创建这样一个同名的变量在原本空空的子表里,

传参要传具体的变量

__index

这样会报错

local obj = {}
x=1
setmetatable(obj, { __index = x}) 
print(obj.health)

Lua 对于 __index 元方法有严格要求,它必须是一个表一个函数

local obj = {}
setmetatable(obj, { __index = 42 }) -- 错误:尝试访问时 Lua 会报错

print(obj.health) -- 报错:bad argument #2 to 'setmetatable' (index must be table or function)

 在 Lua 中,当 __index 被设置为 nil 时,Lua 的行为是将其视为没有定义 __index 元方法。这不会引发错误,而是简单地返回 nil

Lua 不强制要求 __index 必须被定义或赋值,只在查找键失败时才会检查是否存在有效的 __index

local obj = {}
setmetatable(obj, { __index = nil })

print(obj.health) -- 输出: nil

当 Lua 查找一个表中不存在的键时:

  • 如果该表的元表中定义了 __index
    • 如果 __index 是表,Lua 会在这个表中继续查找键。
    • 如果 __index 是函数,Lua 会调用该函数,并将原表和键作为参数传递(重点在先找到,再传入参数,而且这时候的参数的self什么的也是自己此时传入的参数)
  • 如果该表的元表存在,但 __indexnil 或未定义或者有但是里面没找到,Lua 会直接返回 nil,不会报错。

__index 的函数可以没有返回值

Lua 将该返回空的值也视为 nil

__index 的函数在被调用时,会自动接受两个参数(调用的表,缺失的键)

这两个参数是自动传入的,并且在实现 __index 时是强制要求的,不提供这两个参数会导致错误

local obj = {}
setmetatable(obj, {
    __index = function(table, key)
        print("Table is:", table)
        print("Key is:", key)
        return "Default value" -- 显式返回一个值
    end
})

print(obj.health) -- 输出:
-- Table is: <table: 0x...>
-- Key is: health
-- Default value

setmetatable()

通过特殊的元方法来改变表的行为(延伸)实现默认的操作逻辑,比如算术运算比较索引函数调用等。Lua 会在对应的情况下自动调用它们。

以下是元表里常用的内容:


索引相关

__index:自定义键的访问行为

__index接受的是一个访问对象,可以是表可以是函数,但不可以是单个的变量

当访问表中不存在的键时,Lua 会去元表里的 __index 方法找值,表里有值就拿表,有方法可以返回值就接收这个返回值

示例 1:使用表提供默认值

local defaults = { health = 100, mana = 50 }
local t = {}
setmetatable(t, { __index = defaults })

print(t.health) -- 输出: 100(默认值)
print(t.attack) -- 输出: nil(没有定义)

示例 2:使用函数动态生成值

local t = {}
setmetatable(t, {
    __index = function(_, key)
        return "键 " .. key .. " 不存在"
    end
})

print(t.unknown) -- 输出: 键 unknown 不存在

__newindex:自定义键的赋值行为

当试图给表中不存在的键赋值时,Lua 会调用 __newindex 方法。

  • 可以拦截并自定义赋值逻辑。

示例:限制某些键的赋值

local t = {}
setmetatable(t, {
    __newindex = function(_, key, value)
        print("你不能直接添加新键 " .. key .. ",但我记录下来了!")
    end
})

t.newKey = 123 -- 输出: 你不能直接添加新键 newKey,但我记录下来了!
print(t.newKey) -- 输出: nil

算术操作

元表可以通过定义算术相关的元方法,改变表在算术操作中的行为。以下是常用的元方法:

__add:加法

定义两个表相加时的行为:

local t1 = { value = 5 }
local t2 = { value = 10 }

setmetatable(t1, {
    __add = function(a, b)
        return { value = a.value + b.value }
    end
})

local result = t1 + t2
print(result.value) -- 输出: 15
__sub__mul__div__mod__pow

这些方法分别用于减法、乘法、除法、取模、幂运算。例如:

local t1 = { value = 2 }
local t2 = { value = 3 }

setmetatable(t1, {
    __mul = function(a, b)
        return { value = a.value * b.value }
    end
})

local result = t1 * t2
print(result.value) -- 输出: 6

比较操作

元表还可以控制比较操作的行为:

__eq:等于
local t1 = { id = 1 }
local t2 = { id = 1 }

setmetatable(t1, {
    __eq = function(a, b)
        return a.id == b.id
    end
})

print(t1 == t2) -- 输出: true
__lt__le:小于和小于等于
local t1 = { value = 5 }
local t2 = { value = 10 }

setmetatable(t1, {
    __lt = function(a, b)
        return a.value < b.value
    end
})

print(t1 < t2) -- 输出: true

表的行为

__tostring:自定义表的字符串表示

用于定义表被转换为字符串时的行为,例如 print

local t = { name = "test" }

setmetatable(t, {
    __tostring = function(table)
        return "表的名字是:" .. table.name
    end
})

print(t) -- 输出: 表的名字是:test

__len:自定义表的长度

控制 # 操作符的行为:

local t = { a = 1, b = 2 }

setmetatable(t, {
    __len = function()
        return 100
    end
})

print(#t) -- 输出: 100

__call:使表可以被调用

让表像函数一样调用:

local t = {}

setmetatable(t, {
    __call = function(_, a, b)
        return a + b
    end
})

print(t(3, 5)) -- 输出: 8

实际使用场景

元表的功能非常强大,常见的使用场景包括:

  1. 默认值表:使用 __index 为表提供默认值。
  2. 运算符重载:让自定义类型支持算术或比较运算。
  3. 面向对象编程:通过 __index 和元表实现类与对象。
  4. 只读表:使用 __newindex 拦截赋值行为,防止表被修改。
  5. 代理表:通过 __index__call 动态生成数据。

修正语法思路

关于元表绑定父表的一个小细节

local base = {
    name = "base",
    speak = function(self)
        print("Name is " .. self.name)
    end
}

local derived = {}
setmetatable(derived, { __index = base })

-- 调用方法
derived:speak()  -- 输出: Name is nil

解释

  • derived 没有 name 属性,speak 是从元表 base 中继承的。
  • self 绑定到调用者 derived,所以 self.namenil

如果在 derived 中添加属性,行为会改变:

derived.name = "derived"

derived:speak() -- 输出: Name is derived

当使用 : 调用方法时,Lua 会自动将表作为第一个参数传入函数,并绑定到变量 self

self 完全取决于调用者传入的第一个参数

同一个函数,被别的表继承之后,:如果要执行这个函数,self指的就是那个新的继承的表对象

这个self是动态的

这里看上去是在给Object这个表对象写新方法,但这个表对象会被作为元表去被别的表继承,

function这个东西写哪都一样,只是一种简写,对于{}调用自己的方法,然后会有一个self指代自己(这个自己不是真的要是自己,而是使用这个方法时的自己,self的功能重点不在是定义在谁的{},而在对表之间的“继承”关系的强调) 

现在再看这个继承的代码看着简单,但是通用性很强,里面的确有很多知识点

把Object声明在全局,突出class的感觉

new和subClass方法也写成全局的

不用:语法糖实现Object万物之父

所有的方法都要传入参数,这个参数不会

参数能不能传入,只靠local对参数的规定

-- Object 万物之父类
local Object = {}

-- 创建新对象方法
function Object.new(self)
    local instance = {}
    setmetatable(instance, { __index = self }) -- 设置元表,继承父类的方法
    return instance
end

-- 添加一个通用的方法
function Object.say(self, message)
    print("[" .. tostring(self) .. "] says: " .. message)
end

-- 返回 Object 类
return Object

要实现一个万物之父的Object,

首先创建这个大表{},因为是万物之父,所以这个表里包含了new一个新的自己的能力

在这体现为function Object:new()

       .....

        end

封装

然后是对这个自我实例化的函数的实现,也就是给这个万物之父Object表{}对象,写一个可以创建出新的独立的{}表的方法(这个创建的新{}对象,里面还要实现和Object{}对象里一模一样的功能)

那问题就来到了这个新{}对象的创建,以及这个新{}要怎么复制Object对象里的各种方法和数据

这里巧妙的运用了闭包,即Object{}对象里的new方法,里面存在的各种变量,也是有生命周期的

这一点同c#

所以在这个Object{}对象里的new函数里

local obj={}

...

return obj

这个return的值在外部用变量接了就可以用了

二次强调,这个new的方法是属于Object这个{}的

lua创建一个对象的流程:先写好一个实实在在的{},然后想有新的“实例”,就通过元表setmetatable()把新的“实例”去引用上已经存在的{}里的各种...,这个部分是__index在发挥作用

换句话说,这个功能本是setmetatable()设置元表的一个小分支,

但意外的是这个小分支是实现面向对象的继承和创建新对象的关键

setmetatable()先写元表再写上面的self.__index=self

Lua热更新

Lua 是 Unity 热更新的一种常见选择,但并不是唯一的方案。

以下是详细的解释和 Lua 在 Unity 热更新中的应用场景:

Lua 热更新的核心原理

热更新的目标是在不重新发布客户端版本的情况下更新游戏逻辑。Lua 作为脚本语言,可以加载和运行外部脚本文件,更新逻辑时只需替换脚本资源,而无需重新打包整个应用。

在 Unity 中,Lua 热更新通常通过以下方式实现:

  • 核心逻辑(如主角行为、UI 逻辑)写在 Lua 脚本中。
  • 在 C# 程序中嵌入 Lua 虚拟机(通常使用 LuaBridgeSLuaXLua 等工具)
  • 在运行时动态加载和执行 Lua 脚本文件,达到热更新的效果。

为什么 Lua 是热门选择

  • 游戏行业传统:Lua 广泛应用于 Cocos2d、Corona、Roblox 等游戏引擎,因此在游戏行业中积累了丰富的生态和成熟的实践经验
  • Unity 插件支持:Unity 中有多个成熟的 Lua 框架(如 XLuaSLuaToLua),可以快速上手,降低实现热更新的技术门槛。
  • 适配热更新需求:Lua 支持动态加载脚本,结合 Unity 的 AssetBundle 等机制,可以在运行时替换特定逻辑,满足热更新需求。

热更新的其他方案(非 Lua)

  1. ILRuntime(C# 热更新)

    • ILRuntime 是一个开源的 .NET 运行时框架,可以让游戏逻辑以 C# 写成的 DLL 文件形式热更新。
    • 优点:无需学习新的语言,直接使用 C# 编写代码;性能接近原生逻辑。
    • 缺点:复杂度较高,调试和配置成本稍高于 Lua。
  2. HybridCLR

    • Unity 原生不支持代码热更新,但使用 HybridCLR(开源的 AOT+Interpreter 混合运行时),可以实现类似 Lua 的热更新效果。
    • 优点:完全基于 C#,不需要引入额外语言;性能优越。
    • 缺点:配置复杂,需要对 Unity 和 CLR 有深入理解。
  3. AssetBundle 热更新

    • 通过资源的动态加载更新 UI、关卡、角色等内容,而不是直接更新逻辑。
    • 优点:简单易用,适合仅更新资源的场景。
    • 缺点:无法更新游戏逻辑。
  4. Python 或其他脚本语言

    • 某些项目可能会选择 Python 或其他轻量脚本语言(如 JavaScript)作为热更新脚本语言,但使用较少,生态不如 Lua 完善。

如果更倾向于使用 C# 完成所有开发任务,也可以探索 ILRuntimeHybridCLR 等更贴近 Unity 原生的热更新方案。

Lua的实现流程

eg:活动与任务系统

  • 实际场景
    游戏中的限时活动、节日任务等需要随时调整或增加,比如:

    • 在春节期间增加一个“新年集福活动”。

    • 在万圣节期间添加“收集南瓜”的任务,完成后奖励独特皮肤。

  • 为何需要热更新
    这些任务的规则(比如“收集多少南瓜”、“奖励什么物品”)可能需要动态调整。如果每次都通过重新发布客户端更新,这会导致玩家需要频繁下载新版本,影响体验。

  • 热更新解决方案
    开发者只需通过服务器发送新的 Lua 脚本,定义任务逻辑,客户端运行时加载这些脚本即可完成更新。

如果开发者对游戏代码进行了加密或封装,玩家无法随意加载和替换脚本内容。否则,恶意脚本可能造成数据泄露或作弊行为

游戏需要内置一个脚本虚拟机(比如 Lua VM),支持动态加载和执行外部脚本。如果没有这种机制,热更新就无法通过脚本实现。


热更新的限制

游戏逻辑中哪些部分允许用 Lua 控制,哪些是固定在 C# 或引擎层(Unity 原生)中,通常是预先定义好的

一些核心机制(如图形渲染、网络通信)往往不会交给 Lua 脚本,而是保留在底层代码中。

游戏服务器通常会校验客户端发来的逻辑更新(如活动配置、技能参数)。如果玩家任意替换脚本但未通过服务器校验,可能无法生效。

策划使用Lua

程序员需要预先用框架(如 Unity 的 XLuaToLua 或自研绑定系统)把 Lua 与游戏的核心逻辑或数据结构连接起来,让 Lua 脚本中的变量和程序代码的数据能够互相访问。以下是常见的绑定机制:

通过 Lua 表映射游戏数据

程序员定义一个数据表(如角色属性、任务数据),并通过接口将其暴露给 Lua 脚本。例如:

public class Player
{
    public int health = 100;
    public int mana = 50;
}

策划如何知道哪些变量可以用?

程序员一般要提供一份脚本文档或模板,列出策划可以操作的变量、数据结构和接口。

  • 文档需要详细说明:
    • player.health 表示角色当前血量,类型是整数;
    • enemy.attackPower 表示敌人的攻击力。
    • ...

程序员和策划需要约定数据和逻辑的名称

-- Lua 脚本
local tasks = {
    { id = 1, name = "收集苹果", required = 10, reward = 100 },
    { id = 2, name = "击败敌人", required = 5, reward = 200 }
}
public void LoadTaskData(LuaTable tasks)
{
    foreach (var task in tasks)
    {
        Debug.Log($"任务 {task.name}: 收集 {task.required} 件物品,奖励 {task.reward} 金币");
    }
}

也可以通过工具自动生成绑定 

存在一些框架(如 XLua、ToLua)支持自动生成 Lua 脚本的 API 文档和绑定代码

程序员在 Unity 中使用 XLua,会自动生成可以在 Lua 中调用的 C# 函数和变量清单。策划直接查阅清单即可知道哪些数据可以修改。

  • 程序员的职责:通过绑定框架(如 XLua)将游戏数据和 Lua 脚本连接起来,并定义清晰的接口。
  • 策划的职责:按照约定好的变量名和函数名,在 Lua 脚本中编写逻辑,修改数据即可。
  • 工具和约定的作用:策划只需了解绑定的变量和接口,配合文档和自动生成的工具,可以轻松完成任务配置或逻辑调整。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/923902.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

leetcode 3206. 交替组 I 简单

给你一个整数数组 colors &#xff0c;它表示一个由红色和蓝色瓷砖组成的环&#xff0c;第 i 块瓷砖的颜色为 colors[i] &#xff1a; colors[i] 0 表示第 i 块瓷砖的颜色是 红色 。colors[i] 1 表示第 i 块瓷砖的颜色是 蓝色 。 环中连续 3 块瓷砖的颜色如果是 交替 颜色&…

彻底解决 macOS 下Matplotlib 中文显示乱码问题

彻底解决 macOS 下Matplotlib 中文显示乱码问题 在使用 Python 的 Matplotlib 库进行数据可视化时&#xff0c;中文字符的显示常常会出现乱码问题&#xff0c;尤其在 macOS 系统上。在网上找了一大堆方法&#xff0c;花了很久&#xff0c;发现不是要安装各种字体就是要改配置&…

深度学习笔记24_天气预测

&#x1f368; 本文为&#x1f517;365天深度学习训练营 中的学习记录博客&#x1f356; 原作者&#xff1a;K同学啊 | 接辅导、项目定制 一、我的环境 1.语言环境&#xff1a;Python 3.9 2.编译器&#xff1a;Pycharm 3.深度学习环境&#xff1a;TensorFlow 2.10.0 二、GPU设置…

podman 源码 5.3.1编译

1. 构建环境 在麒麟V10服务器操作系统上构建&#xff1a;Kylin-Server-V10-GFB-Release-2204-Build03-ARM64.iso。由于只是编译 podman 源码&#xff0c;没必要特地在物理机或服务上安装一个这样的操作系统&#xff0c;故采用在虚拟机里验证。 2. 安装依赖 参考资料&#xf…

【K8S系列】深入解析 Kubernetes 中的 Deployment

Kubernetes&#xff08;K8s&#xff09;是一个开源的容器编排平台&#xff0c;旨在自动化应用程序的部署、扩展和管理。在 Kubernetes 中&#xff0c;Deployment 是一种用于管理无状态应用的工作负载资源&#xff0c;提供了丰富的功能&#xff0c;包括版本控制、滚动更新和回滚…

玩转 Burp Suite (1)

内容预览 ≧∀≦ゞ 玩转 Burp Suite (1)声明Burp Suite 简介Dashboard&#xff08;仪表盘&#xff09;1. 默认任务管理2. 暂停任务3. 新建扫描任务4. 使用总结 Target&#xff08;目标&#xff09;1. SIte Map &#xff08;站点地图&#xff09;2. Scope&#xff08;范围&#…

【ArcGISPro】Sentinel-2数据处理

错误 默认拉进去只组织了4个波段,但是实际有12个波段 解决方案 数据下载 Sentinel-2 数据下载-CSDN博客 数据处理 数据查看 创建镶嵌数据集 在数据管理工具箱中找到创建镶嵌数据集

智慧环保大数据解决方案

1. 智慧环保概述 智慧环保是“数字环保”的延伸&#xff0c;借助物联网技术整合环境监控对象&#xff0c;通过云计算实现环境管理与决策的智能化。其核心在于快速感知城市环境指标&#xff0c;保障人体健康与生命安全。 2. 智慧环保总体目标 智慧环保的总体目标是建立全面感…

【H2O2|全栈】JS进阶知识(八)ES6(4)

目录 前言 开篇语 准备工作 浅拷贝和深拷贝 浅拷贝 概念 常见方法 弊端 案例 深拷贝 概念 常见方法 弊端 逐层拷贝 原型 构造函数 概念 形式 成员 弊端 显式原型和隐式原型 概念 形式 constructor 概念 形式 原型链 概念 形式 结束语 前言 开篇语…

03-微服务搭建

1、搭建分布式基本环境 分布式组件 功能 SpringCloud Alibaba - Nacos 注册中心&#xff08;服务发现/注册&#xff09;、配置中心&#xff08;动态配置管理&#xff09; SpringCloud Alibaba - Sentinel 服务容错&#xff08;限流、降级、熔断&#xff09; SpringCloud …

Vue前端开发2.3.2-4 绑定指令

本文介绍了Vue中的绑定指令&#xff0c;包括属性绑定指令v-bind、事件绑定指令v-on以及双向数据绑定指令v-model。通过创建单文件组件&#xff0c;演示了如何使用这些指令来控制DOM属性、监听事件和实现表单输入与数据的双向同步。同时&#xff0c;探讨了v-model的修饰符如.num…

uniapp开发支付宝小程序自定义tabbar样式异常

解决方案&#xff1a; 这个问题应该是支付宝基础库的问题&#xff0c;除了依赖于官方更新之外&#xff0c;开发者可以利用《自定义 tabBar》曲线救国 也就是创建一个空内容的自定义tabBar&#xff0c;这样即使 tabBar 被渲染出来&#xff0c;但从视觉上也不会有问题 1.官方文…

双向链表、循环链表、栈

双向循环链表 class Node:#显性定义出构造函数def __init__(self,data):self.data data #普通节点的数据域self.next None #保存下一个节点的链接域self.prior None #保存前一个节点饿链接域 class DoubleLinkLoop:def __init__(self, node Node):self.head nodeself.siz…

【大数据学习 | Spark-Core】RDD的缓存(cache and checkpoint)

1. 单应用缓存&#xff1a;cache 1.1 cache算子 cache算子能够缓存中间结果数据到各个executor中&#xff0c;后续的任务如果需要这部分数据就可以直接使用避免大量的重复执行和运算。 rdd 存储级别中默认使用的算子cache算子&#xff0c;cache算子的底层调用的是persist算子…

上海乐鑫科技一级代理商飞睿科技,ESP32-C61高性价比WiFi6芯片高性能、大容量

在当今快速发展的物联网市场中&#xff0c;无线连接技术的不断进步对智能设备的性能和能效提出了更高要求。为了满足这一需求&#xff0c;乐鑫科技推出了ESP32-C61——一款高性价比的Wi-Fi 6芯片&#xff0c;旨在为用户设备提供更出色的物联网性能&#xff0c;并满足智能设备连…

如何选择黑白相机和彩色相机

我们在选择成像解决方案时黑白相机很容易被忽略&#xff0c;因为许多新相机提供鲜艳的颜色&#xff0c;鲜明的对比度和改进的弱光性能。然而&#xff0c;有许多应用&#xff0c;选择黑白相机将是更好的选择&#xff0c;因为他们产生更清晰的图像&#xff0c;更好的分辨率&#…

ubuntu22开机自动登陆和开机自动运行google浏览器自动打开网页

一、开机自动登陆 1、打开settings->点击Users 重启系统即可自动登陆桌面 二、开机自动运行google浏览器自动打开网页 1、安装google浏览器 sudo wget https://dl.google.com/linux/direct/google-chrome-stable_current_amd64.deb sudo dpkg -i ./google-chrome-stable…

MVC、EL、JSTL

1.MVC设计模式 三层&#xff1a; MVC&#xff1a; M&#xff08;Model&#xff09;模型&#xff1a;负责业务逻辑处理&#xff0c;数据库访问。 V&#xff08;View&#xff09;视图&#xff1a;负责与用户交互。 C&#xff08;Controller&#xff09;控制器&#xff1a;负责流程…

Python的3D可视化库 - vedo (3)visual子模块 点对象的可视化控制

文章目录 3 PointsVisual的方法3.1 对象属性3.1.1 顶点大小3.1.2 复制属性3.1.3 颜色设置3.1.4透明度设置 3.2 对象光效3.2.1 点的形状3.2.2 点的表面光效 3.3 尾随线和投影3.3.1 尾随线3.3.2 投影 3.4 给对象附加文字说明3.4.1 标注3.4.2 2D标注3.4.3 气泡说明3.4.4 旗标说明3…

MySQL系列之远程管理(安全)

导览 前言Q&#xff1a;如何保障远程登录安全一、远程登录的主要方式1. 用户名/口令2. SSH3. SSL/TLS 二、使用TLS协议加密连接1. 服务端2. 客户端 结语精彩回放 前言 在我们的学习或工作过程中&#xff0c;作为开发、测试或运维人员&#xff0c;经常会通过各类客户端软件&…