自由学习记录(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.idend
})print(t1 == t2) -- 输出: true
__lt__le:小于和小于等于
local t1 = { value = 5 }
local t2 = { value = 10 }setmetatable(t1, {__lt = function(a, b)return a.value < b.valueend
})print(t1 < t2) -- 输出: true

表的行为

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

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

local t = { name = "test" }setmetatable(t, {__tostring = function(table)return "表的名字是:" .. table.nameend
})print(t) -- 输出: 表的名字是:test

__len:自定义表的长度

控制 # 操作符的行为:

local t = { a = 1, b = 2 }setmetatable(t, {__len = function()return 100end
})print(#t) -- 输出: 100

__call:使表可以被调用

让表像函数一样调用:

local t = {}setmetatable(t, {__call = function(_, a, b)return a + bend
})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 脚本中编写逻辑,修改数据即可。
  • 工具和约定的作用:策划只需了解绑定的变量和接口,配合文档和自动生成的工具,可以轻松完成任务配置或逻辑调整。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

1- 9 C 语言面向对象

面向对象的基本特性&#xff1a;封装&#xff0c;继承&#xff0c;多态 1.0 面向过程概念 当我们在编写程序时&#xff0c;通常采用以下步骤&#xff1a; 1. 将问题的解法分解成若干步骤 2. 使用函数分别实现这些步骤 3. 依次调用这些函数 这种编程风格的被称作 面向过程…

路面泥泞,坑洼,裂缝,路面损坏,马路牙检测 YOLO标记资源整理

数据集介绍 可识别常见的路面泥泞&#xff0c;坑洼&#xff0c;裂缝&#xff0c;路面损坏&#xff0c;马路牙等多种路面状况。 数据集分割 训练集89&#xff05; 2052图片 validation集8% 186图片 test集3% 78图片 预处理 自动定向&#xff1a; 已应用 自动调…

Jmeter中的测试片段和非测试原件

1&#xff09;测试片段 1--测试片段 功能特点 重用性&#xff1a;将常用的测试元素组合成一个测试片段&#xff0c;便于在多个线程组中重用。模块化&#xff1a;提高测试计划的模块化程度&#xff0c;使测试计划更易于管理和维护。灵活性&#xff1a;可以通过模块控制器灵活地…

Cocos编辑器

1、下载 下载地址&#xff1a;https://www.cocos.com/creator-download 2、编辑器界面介绍 官方链接&#xff1a;https://docs.cocos.com/creator/3.8/manual/zh/editor/ 3、项目结构 官方链接&#xff1a;https://docs.cocos.com/creator/3.8/manual/zh/getting-started/…

JAVA题目笔记(二十)Stream流综合练习+方法引用

一、数据过滤 import java.util.*; import java.util.stream.Collectors;public class Co {public static void main(String[] args) {List<Integer> listnew ArrayList<>();Collections.addAll(list,1,2,3,4,5,6,7,8,9,10);List<Integer> newlist list.str…

计算机操作系统——进程控制(Linux)

进程控制 进程创建fork&#xff08;&#xff09;函数fork() 的基本功能fork() 的基本语法fork() 的工作原理fork() 的典型使用示例fork() 的常见问题fork() 和 exec() 结合使用总结 进程终止与$进程终止的本质进程终止的情况正常退出&#xff08;Exit&#xff09;由于信号终止非…

摄像头原始数据读取——V4L2(mmap模式,V4L2_MEMORY_MMAP)

摄像头原始数据读取——V4L2(mmap模式,V4L2_MEMORY_MMAP) 内存映射模式&#xff0c;是将设备在内核态申请的用于存储视频数据的物理内存映射到用户空间&#xff0c;使得用户应用程序可以直接访问和操作设备数据物理内存&#xff0c;避免了数据的拷贝。因此采集速度较快&#x…

零地址挂页

零地址 如果我们有比较好的C编程基础&#xff0c;我们就会知道&#xff0c;我们在代码中定义了一个零地址或者空指针&#xff0c;那么它实际上会指向虚拟内存的零地址&#xff0c;多数操作系统&#xff0c;包括Win&#xff0c;在进程创建的时候&#xff0c;都会空出前64k的空间…

QT6学习第四天 感受QT的文件编译

QT6学习第四天 感受QT的文件编译 使用纯代码编写程序新建工程 使用其他编辑器纯代码编写程序并在命令行运行使用 .ui 表单文件生成界面使用自定义 C 窗口类使用现成的QT Designer界面类 使用纯代码编写程序 我们知道QT Creator中可以用拖拽的方式在 .ui 文件上布局&#xff0c…

windows安全中心,永久卸载工具分享

使用方法 2024Goby红队版工具分享&#xff0c;附2024年漏洞POC下载 下载链接&#xff1a; https://pan.quark.cn/s/4fc2712a2afc一路回车&#xff0c;选项Y即可 耐心等待几秒种&#xff0c;自动重启 此时打开windows安全中心&#xff0c;已经完全不能使用了&#xff0c;响应…

jvm核心组件介绍

1. 类加载器&#xff08;ClassLoader&#xff09;&#xff1a; • 想象它是一个快递员&#xff0c;负责把Java类&#xff08;.class文件&#xff09;这个“包裹”从磁盘这个“发货地”送到JVM内部这个“目的地”。类加载器确保每个类只被加载一次&#xff0c;并维护一个类的层级…

目标检测,图像分割,超分辨率重建

目标检测和图像分割 目标检测和图像分割是计算机视觉中的两个不同任务&#xff0c;它们的输出形式也有所不同。下面我将分别介绍这两个任务的输出。图像分割又可以分为&#xff1a;语义分割、实例分割、全景分割。 语义分割&#xff08;Semantic Segmentation&#xff09;&…

Python编程技巧:多变量赋值的优雅艺术

在Python编程的世界里&#xff0c;有许多令人惊叹的语法特性&#xff0c;而多变量赋值就像是一颗闪耀的明珠&#xff0c;它不仅让代码更优雅&#xff0c;还能提升程序的执行效率。今天我们就深入探讨这个看似简单却蕴含深意的编程技巧。 基础认识 传统的变量赋值方式&#xff…

CentOS 7 安装部署 KVM

1.关闭虚拟机 打开相关选项 打开虚拟机centos7 连接xshell 测试网络&#xff0c;现在就是没问题的&#xff0c;因为我们要使用网络源 安装 GNOME 桌面环境 安装KVM 模块 安装KVM 调试工具 构建虚拟机的命令行工具 qemu 组件,创建磁盘、启动虚拟机等 输入这条命令&#xff0c;…

微信小程序学习指南从入门到精通

&#x1f5fd;微信小程序学习指南从入门到精通&#x1f5fd; &#x1f51d;微信小程序学习指南从入门到精通&#x1f51d;✍前言✍&#x1f4bb;微信小程序学习指南前言&#x1f4bb;一、&#x1f680;文章列表&#x1f680;二、&#x1f52f;教程文章的好处&#x1f52f;1. ✅…

【C++】读取数量不定的输入数据

读取数量不定的输入数据 似乎是一个很实用的东西&#xff1f; 问题&#xff1a; 我们如何对用户输入的一组数&#xff08;事先不知道具体有多少个数&#xff09;求和&#xff1f; 这需要不断读取数据直至没有新的输入为止。&#xff08;所以我们的代码就是这样设计的&#x…

基于vite创建的react18项目的单元测试

题外话 最近一个小伙伴进了字节外包&#xff0c;第一个活就是让他写一个单元测试。 嗯&#xff0c;说实话&#xff0c;在今天之前我只知道一些理论&#xff0c;但是并没有实操过&#xff0c;于是我就试验了一下。 通过查询资料&#xff0c;大拿们基本都说基于vite的项目&…

如何用通义灵码助力项目开发 | OceanBase obdiag 项目共建实践

本文来自 obdiag 项目共建的用户分享 一、背景 我的数据库探索之旅始于OceanBase。作为一位满怀好奇心的DBA&#xff0c;我内心始终怀揣着对数据库内部运作机制的无尽向往。开源如同一把钥匙&#xff0c;为我们这些求知欲旺盛的“好奇猫”解锁了通往新知的神秘大门。在众多分布…

idea_卸载与安装

卸载与安装 卸载1、设置 -> 应用2、查找到应用&#xff0c;点击卸载3、把删除记录和设置都勾选上4、删除其它几个位置的残留 安装1、下载安装包2、欢迎安装 -> Next3、选择安装目录 -> Next4、创建快捷图标和添加到环境变量5、确认文件夹的名称 -> Install6、完成安…

day01

Hm-Footer.vue <template><div class"hm-footer">我是hm-footer</div></template><script>export default {}</script><style>.hm-footer{height:100px;line-height:100px;text-align:center;font-size:30px;background-…