我的世界Java版1.21.4的Fabric模组开发教程(十四)方块实体

这是适用于Minecraft Java版1.21.4的Fabric模组开发系列教程专栏第十四章——方块实体。想要阅读其他内容,请查看或订阅上面的专栏。

方块实体(Block Entity) 指的是一种用于存储方块额外数据的方法。但这种数据和为了控制方块状态而在自定义方块类中创建的属性不太相同,通常方块实体中的数据可能需要进行逻辑处理和计算,且一般不会直接体现在方块上。

不过,本章中创建的带有方块实体的自定义方块始终使用了方块实体中的数据来控制方块外观——光源方块(light_block),玩家通过使用(右键点击)“光源方块”来动态切换其光照强度,每点击一下,方块的光照强度将+1;如果光照强度已经为15,则重置光照强度为0。

要创建“光源方块”,需要按照以下步骤推进:

  • 创建方块实体类;
  • 创建方块实体注册类;
  • 注册方块实体;
  • 创建带有方块实体功能的方块类;
  • 注册方块并将其添加到物品组中;
  • 完成创建方块的其他工作;
  • 在方块实体类中添加数据;
  • 在方块类中使用方块实体;

创建方块实体类

要创建方块对应的实体,需要创建一个方块实体类,并使其继承BlockEntity类。


方块实体类BlockEntity

BlockEntity类一般作为方块实体类的父类,用于保存一个世界中方块的额外数据。通常,方块的数据由预定义的、有限的方块状态数据集合保存。但是,方块需要存储一些不能被预定义的数据,例如箱子里的物品、告示牌上的文字等,方块实体可以存储这些数据。

通常,继承了BlockEntity类的自定义方块实体类需要调用其构造方法;

public BlockEntity(BlockEntityType<?> type, BlockPos pos, BlockState state) {this.type = type;this.pos = pos.toImmutable();this.validateSupports(state);this.cachedState = state;
}

其中需要传递方块实体类型BlockEntityType对象、方块位置BlockPos对象和方块状态BlockState对象。

方块实体类型类BlockEntityType

BlockEntityType类用于定义和管理方块实体类型。在创建完实体类后,还需要在游戏注册表中注册实体类,此时将返回一个BlockEntityType对象,需要在实体类的构造方法中使用。


1.创建com/example/test/block/entity/custom目录,然后在其中创建LightBlockEntity并继承BlockEntity类;

public class LightBlockEntity extends BlockEntity {}

2.声明构造方法,在方法体中调用其父类的构造方法;

public LightBlockEntity(BlockEntityType<?> type, BlockPos pos, BlockState state) {super(type, pos, state);
}

稍后我们将修改此处的BlockEntityType对象。

创建方块实体注册类

方块实体注册类用于将所有方块实体注册到游戏注册表中。一般,其中有三部分内容:

  • 封装的register()方法:用于注册方块实体,并返回一个BlockEntityType对象;
  • 类型为BlockEntityType<T>的静态常量:代表已经注册的方块实体。稍后将被对应的方块实体类的构造方法所使用;
  • initialize():用于初始化方块实体注册类。需要在入口点类的onInitialize()方法中调用;

1.在com/example/test/block/entity目录中创建ModBlockEntities.java

public class ModBlockEntities {}

2.声明静态私有方法register()。根据官方文档,register()的写法应该如下:

private static <T extends BlockEntity> BlockEntityType<T> register(String name,FabricBlockEntityTypeBuilder.Factory<? extends T> entityFactory,Block... blocks
) {Identifier id = Identifier.of(FabricDocsReference.MOD_ID, name);return Registry.register(Registries.BLOCK_ENTITY_TYPE, id, FabricBlockEntityTypeBuilder.<T>create(entityFactory, blocks).build());
}

方法提供三个参数,分别为注册的方块实体名name、Fabric方块实体类型构造器工厂内部类FabricBlockEntityTypeBuilder.Factory对象的表达式和方块对象block。方法首先调用了Identifier.of`()方法创建方块实体的标识符对象,其中使用了模组Id和参数中的方块实体名name,最后返回BlockEntityType对象,同时也是Registry.register()方法的返回值,其中传递了注册键类型为方块实体类型Registries.BLOCK_ENTITY_TYPE、标识符对象以及build()方法返回的BlockEntityType对象;

3.声明initialize()方法,用于初始化方块实体注册类;

public static void initialize() {Test.LOGGER.info("Registering ModBlockEntities !!!");
}

其中可以输出日志信息;

4.在入口点类的onInitialize()方法中调用ModBlockEntities.initialize()方法;

public void onInitialize() {//...ModBlockEntities.initialize();//...
}

完成方块实体注册类的初始化。

注册方块实体

现在,我们可以直接在方块实体注册类中将方块实体注册到游戏注册表中。

1.在ModBlockEntities类中声明静态常量LIGHT_BLOCK_ENTITY,调用register()方法对其初始化;

public static final BlockEntityType<LightBlockEntity> LIGHT_BLOCK_ENTITY =register("light", LightBlockEntity::new, ModBlocks.LIGHT_BLOCK);

方法中分别传递了方块实体路径名“light”、Fabric方块实体类型构造器工厂内部类FabricBlockEntityTypeBuilder.Factory对象的表达式LightBlockEntity::new,这里使用了方块实体类构造方法的表达式,以及方块对象ModBlocks.LIGHT_BLOCK

这里的ModBlocks.LIGHT_BLOCK指的是方块对象,稍后我们将在方块注册类中声明并对其初始化;

2.完成后,回到方块实体类LightBlockEntity,修改构造方法;

public LightBlockEntity(BlockPos pos, BlockState state) {super(ModBlockEntities.LIGHT_BLOCK_ENTITY,pos, state);
}

首先删除BlockEntityType<?> type参数,然后将super()方法中第一个参数改为刚刚注册完毕的方块实体对象ModBlockEntities.LIGHT_BLOCK_ENTITY

创建带有方块实体功能的方块类

与普通方块创建的过程有所不同,创建带有方块实体功能的方块类时,方块类需要继承BlockWithEntity类并重写相关方法。

1.创建com/example/test/block/custom目录,在其中创建LightBlock.java并继承BlockWithEntity类;

public class LightBlock extends BlockWithEntity {}

2.声明构造方法;

public LightBlock(Settings settings) {super(settings);
}

3.重写getCodec()方法,处理方块实体中数据的序列化与反序列化;

@Override
protected MapCodec<? extends BlockWithEntity> getCodec() {return createCodec(LightBlock::new);
}

其中调用createCodec()方法,其中传递抽象方块设置内部类和自定义方块类的函数表达式,此处使用了方块类的构造方法;

4.重写createBlockEntity()方法,在方法体中调用方块实体类的构造方法,用于在创建方块时完成方块实体的初始化。

@Nullable
@Override
public BlockEntity createBlockEntity(BlockPos pos, BlockState state) {return new LightBlockEntity(pos, state);
}

注册方块并将其添加到物品组中

现在,我们在方块注册类中使用自定义方块类注册方块,这与其他方块的注册方式相同。

1.在方块注册类ModBlocks中声明静态常量LIGHT_BLOCK,调用register()方法对其进行初始化;

public static final Block LIGHT_BLOCK = register("light_block",LightBlock::new,AbstractBlock.Settings.create(),true
);

按顺序依次设置了方块名为“light_block”、抽象方块设置内部类对象和方块对象的函数式LightBlock::new、抽象方块设置内部类对象并要求注册方块对应的方块物品;

2.在入口点类的onInitialize()方法中,将方块添加到指定物品组中。

ItemGroupEvents.modifyEntriesEvent(CUSTOM_ITEM_GROUP_KEY).register((itemGroup) -> {...itemGroup.add(ModBlocks.LIGHT_BLOCK.asItem());...});

完成创建方块的其他工作

与创建普通方块相同,我们还要为方块:

  • 创建模型文件;
  • 创建模型描述文件;
  • 创建方块纹理;
  • 创建方块状态文件;
  • 在语言文件中为其添加翻译键值对;

1.在assets/test/models/block目录中创建light_block.json作为方块的模型文件;

{"parent": "minecraft:block/cube_all","textures": {"all": "test:block/light_block"}
}

2.在assets/test/items目录中创建light_block.json作为方块的模型描述文件;

{"model": {"type": "minecraft:model","model": "test:block/ice_wood"}
}

3.在assets/test/textures/block目录中创建方块的纹理图light_block.png

在这里插入图片描述
4.在assets/test/blockstates目录中创建light_block.json作为方块状态文件;

{"variants": {"": {"model": "test:block/light_block"}}
}

5.在zh-cn.json中添加一条翻译键值对;

{..."item.test.light_block": "光源方块",...
}

启动游戏测试

现在,方块的基础内容以及创建完毕,游戏中应当可以正常显示方块。

1.打开游戏,在创造模式物品栏中找到“光源方块”;
在这里插入图片描述
名称、纹理等可以正常显示;

2.将其放置在地面上,可以看到方块模型显示正常;
在这里插入图片描述
但是方块现在没有任何功能,我们要通过方块实体为其添加数据。

在方块实体类中添加数据

下面,我们将方块发光时的光照等级数值存储在方块实体中并编写数据变化的逻辑。

1.声明私有变量level,使其初始值为0,用于存储光照等级;

private int level = 0;

2.为level变量声明一个getLevels()方法,用于获取光照等级;

public int getLevels() {return level;
}

3.声明calculateLevels()方法,用于为光照等级数值添加计算逻辑;

public void calculateLevels() {if(level >= 15){level = 0;}else {level++;}
}

一般方块的最高光照等级为15。因此,右键使用方块时,当光照等级小于15时,光照等级将+1;否则,将光照等级重置为0。

在方块类中使用方块实体

现在,我们可以使用方块实体中的数据来控制“光源方块”的光照强度。

不过,想要控制方块的光照等级,依然需要在方块类中创建用于修改方块状态的属性,然后通过方块实体将已经计算完毕,或者说已经执行完逻辑处理的数据传递给方块属性,从而修改方块的光照强度;

实际上,方块实体中的数据是可以直接使用的。只是在这个例子中,修改方块光源的方法luminance()由内部类AbstractBlock.Settings提供,而其中传递的是泛型为BlockState的函数式接口ToIntFunction的函数表达式,这使得方块的光照强度只能通过方块状态修改;

所以,我们要先为方块类添加用于修改方块状态的属性,然后再使用方块实体。


整型属性类IntProperty

IntProperty用于为方块提供控制方块状态的整型属性。其构造方法已经私有化,所以想要创建一个IntProperty对象,通常调用其静态方法of()

public static IntProperty of(String name, int min, int max) {return new IntProperty(name, min, max);
}

可以看到方法体中直接调用了构造方法,其中需要传递三个参数:

  • String name:指定属性名;
  • int min:指定属性的最小值;
  • int max:指定属性的最大值;

通过minmax变量限制整型数据值的范围。


1.声明静态常量LIGHT_LEVEL,类型为IntProperty,调用静态方法IntProperty.of()对其初始化;

public static final IntProperty LIGHT_LEVEL = IntProperty.of("light_level", 0, 15);

方法中分别设置了属性名为"light_level"以及数值的范围在0~15;

2.重写appendProperties()方法,将属性LIGHT_LEVEL添加到方块中;

@Override
protected void appendProperties(StateManager.Builder<Block, BlockState> builder) {builder.add(LIGHT_LEVEL);
}

调用StateManager.Builder对象的add()方法,将属性附加到方块中;

3.修改构造方法,在super()方法中调用AbstractBlock.Settings对象的luminance()方法,在构造方块对象时直接修改方块的光照强度;

public LightBlock(Settings settings) {super(settings.luminance(state -> state.get(LIGHT_LEVEL)));
}

luminance()方法中传递通过当前方块状态对象的get()方法返回的光照强度的LIGHT_LEVEL的值;

4.重写onUse()方法,在方法体中使用方块实体中的数据来修改当前方块的光照强度LIGHT_LEVEL

@Override
protected ActionResult onUse(BlockState state, World world, BlockPos pos, PlayerEntity player, BlockHitResult hit) {LightBlockEntity lightBlockEntity = (LightBlockEntity) world.getBlockEntity(pos);if(!world.isClient && lightBlockEntity != null) {lightBlockEntity.calculateLevels();world.setBlockState(pos, state.with(LIGHT_LEVEL, lightBlockEntity.getLevels()));}return ActionResult.SUCCESS;
}

首先调用world.getBlockEntity()方法获取指定方块坐标的方块实体对象,方法中传递一个BlockPos对象,即当前方块的坐标对象,将返回的方块实体对象存储在lightBlockEntity中;

接着,在确保当前世界对象为服务器世界对象且方块实体对象不为null的基础上,调用方块实体对象的calculateLevels()方法,处理光照强度的变化逻辑;

然后调用world.setBlockState()方法修改当前方块的方块状态,方法中传递要修改的方块坐标对象pos,以及修改后的方块状态对象,这里调用了state.with()方法,将当前光照等级修改为方块实体中光照等级lightBlockEntity.getLevels()的值;

最后返回ActionResult.SUCCESS,代表操作成功;

5.现在,方块所有功能已经通过方块实体实现,可以再次启动游戏进行测试;

打开游戏,修改世界时间为13000,将“光源方块”放置在地面上,持续右键点击方块,可以发现方块光照强度越来越高。
请添加图片描述

本章小结

本章详细阐述了方块实体的作用与创建的过程。如果没有方块创建和方块状态等前置知识的支撑,阅读本文的难度较高,还请参考专栏的其他文章。感谢各位的阅读,有兴趣可以订阅此专栏!

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

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

相关文章

【UE教程/进阶】UE中的指针与引用

目录直接属性引用共享指针 TSharedPtr实现原理共享引用 TSharedRef弱引用指针 TWeakPtrObject弱指针 FWeakPtr实现原理Object软指针 FSoftObjectPtr原理直接属性引用 在c通过UPROPERTY()宏将属性公开&#xff0c;蓝图中属性类型中的Object Reference。 对一个类型及其子类型的…

早期 CNN 的经典模型—卷积神经网络(LeNet)

目录 LeNet 的设计背景与目标 LeNet 的网络结构&#xff08;经典 LeNet-5&#xff09; 局部感受野详解 一、局部感受野和全连接网络的区别 1. 传统全连接网络的问题 2. 局部感受野的解决方案 二、局部感受野的优势 1. 参数大幅减少 2. 提取局部特征 3. 平移不变性 参数…

RabbitMQ 高级特性之延迟队列

1. 简介 在某些场景下&#xff0c;当生产者发送消息后&#xff0c;可能不需要让消费者立即接收到&#xff0c;而是让消息延迟一段时间后再发送给消费者。 2. 实现方式 2.1 TTL 死信队列 给消息设置过期时间后&#xff0c;若消息在这段时间内没有被消费&#xff0c;就会将消…

uniapp app安卓下载文件 图片 doc xls 数据流文件 app安卓本地路径下载保存

//下载图片 downloadToLocal() {plus.android.requestPermissions([android.permission.WRITE_EXTERNAL_STORAGE],(success) > {uni.saveImageToPhotosAlbum({filePath: /static/x.png,//本地地址success: () > {this.$refs.uToast.show({message: "模版下载成功&am…

Context Engineering:从Prompt Engineering到上下文工程的演进

最近在做Deepresearch以及刷到一个不错的文章&#xff1a;context-engineering-guide &#xff0c;这篇文章揭示了提示工程以及上下文过程在智能体应用开源流程中&#xff0c;包括Deepresearch&#xff0c;MCP在内的一些概念&#xff0c;起到了非常重要的作用&#xff01; Cont…

jenkins部署vue前端项目

文章目录前言一、安装nginx二、jenkins构建项目总结前言 前面已经使用jenkins部署了后端springboot项目&#xff0c;现在开始学习jenkins部署前端Vue项目。 一、安装nginx 访问nginx官网&#xff0c;https://nginx.org/en/download.html下载tar包 上传到服务器目录中 然后到…

设计总监年中复盘:用Adobe XD内容识别布局,告别“手动调距”

时至年中&#xff0c;这不仅是检视上半年项目成果的节点&#xff0c;更是优化团队工作流、为下半年挑战储备动能的关键时期。在海外设计界工作的十余年间&#xff0c;我发现&#xff0c;一个高效的设计团队与一个疲于奔命的团队之间&#xff0c;最大的差别往往就在于是否建立了…

Unity 在Rider中通过Lingma插件使用MCP

环境&#xff1a; Unity 2022.3.12f1 JetBrains Rider 2025.1.4 Lingma 2.5.14 Python 3.13.4 下载包 首先在unity package manager 加入unity-mcp包 https://github.com/justinpbarnett/unity-mcp.git 然后下载uv包&#xff08;要先先下载python&#xff09;,网上很多…

pycharm+SSH 深度学习项目 远程后台运行命令

pycharmSSH 深度学习项目 远程后台运行命令碎碎念&#xff0c;都是实验室里那说关机就关机&#xff0c;说重启就重启的台式机逼得。。学吧记录 运行&#xff1a;nohup /root/miniconda3/bin/python -u "run.py" > /root/log/nohup.log 2>&1 &实时查看日…

【Linux | 网络】应用层(HTTP)

目录一、认识URL二、urlencode和urldecode三、HTTP协议格式&#xff08;使用Fiddler抓包&#xff09;3.1 安装并使用Fiddler抓包3.2 HTTP协议格式3.2.1 HTTP请求3.2.1.1 资源URL路径3.2.1.2 请求方法&#xff08;Method&#xff09;3.2.1.3 Location头字段&#xff08;重定向相…

编程实践:单例模式(懒汉模式+饿汉模式)

说明:本专栏文章有两种解锁方案 1:付费订阅,畅享所有文章 2:免费获取,点击下方链接,关注,自动获取免费链接 https://free-img.400040.xyz/4/2025/04/29/6810a50b7ac8b.jpg 主题:C++ 单例模式 什么是单例模式

破局电机制造四大痛点:MES与AI视觉的协同智造实践

万界星空科技电机行业MES系统解决方案是针对电机制造过程中多工序协同难、质量追溯复杂、设备管理要求高等痛点设计的数字化管理系统。一、电机行业的核心痛点1. 多工序协同困难 电机制造涉及绕线、装配、测试等多道工序&#xff0c;工艺衔接复杂&#xff0c;传统人工调度效率…

HTML 初体验

HTML&#xff08;超文本标记语言&#xff09;全称&#xff1a;HyperText Markup Language。超文本是什么&#xff1f;答&#xff1a;超文本就是网页中的链接。标记是什么&#xff1f;答&#xff1a;标记也叫标签&#xff0c;是带尖括号的文本。需求1&#xff1a;将“我爱中国”…

网络层TCP机制

1.确认应答机制由于发送信息的距离可能较远,可能出现后发的信息先到的情况,怎么办?TCP将每个字节的数据都进行了编号,即为序列号如何分辨一个数据包是普通数据还是应答数据呢2.超时重传由于丢包是一个随机的事件,因此在上述tcp传输的过程中,丢包就存在两种情况但是在发送方的角…

【一起来学AI大模型】微调技术:LoRA(Low-Rank Adaptation) 的实战应用

LoRA&#xff08;Low-Rank Adaptation&#xff09; 的实战应用&#xff0c;使用 Hugging Face 的 peft (Parameter-Efficient Fine-Tuning) 库对大型语言模型进行高效微调。LoRA 因其显著降低资源消耗&#xff08;显存和计算&#xff09;同时保持接近全量微调性能的特点&#x…

RedisJSON 内存占用剖析与调优

一、基础内存模型指针包装 所有 JSON 值&#xff08;标量、对象、数组、字符串等&#xff09;至少占用 8 字节&#xff0c;用于存储一个带类型标记的指针。标量与空容器 null、true、false、小整数&#xff08;静态缓存&#xff09;、空字符串、空数组、空对象 均不分配额外内存…

【LeetCode 热题 100】23. 合并 K 个升序链表——(解法一)逐一合并

Problem: 23. 合并 K 个升序链表 题目&#xff1a;给你一个链表数组&#xff0c;每个链表都已经按升序排列。 请你将所有链表合并到一个升序链表中&#xff0c;返回合并后的链表。 文章目录整体思路完整代码时空复杂度时间复杂度&#xff1a;O(K * N)空间复杂度&#xff1a;O(1…

垃圾收集器-Serial Old

第一章 引言1.1 JVM 中垃圾收集的简要概述JVM&#xff08;Java Virtual Machine&#xff09;作为 Java 程序的运行时环境&#xff0c;负责将字节码加载至内存并执行&#xff0c;同时也承担着内存管理的重任。垃圾收集&#xff08;Garbage Collection&#xff0c;简称 GC&#x…

Docker(02) Docker-Compose、Dockerfile镜像构建、Portainer

Docker-Compose 1、Docker Desktop 在Windows上安装Docker服务&#xff0c;可以使用Docker Desktop这个应用程序。 下载并安装这样的一个安装包 安装好后&#xff1a;执行命令 docker --version 从Docker Hub提取hello-world映像并运行一个容器&#xff1a; docker run h…

大数据时代UI前端的用户体验设计新思维:以数据为驱动的情感化设计

hello宝子们...我们是艾斯视觉擅长ui设计和前端数字孪生、大数据、三维建模、三维动画10年经验!希望我的分享能帮助到您!如需帮助可以评论关注私信我们一起探讨!致敬感谢感恩!一、引言&#xff1a;从 “经验设计” 到 “数据共情” 的体验革命传统 UI 设计常陷入 “设计师主观经…