手把手一步一步教你使用Java开发一个大型街机动作闯关类游戏11之游戏资源加载器

简介: 手把手一步一步教你使用Java开发一个大型街机动作闯关类游戏11之游戏资源加载器

游戏资源加载器


上一节我们实现了一组动画,但也是基于某个角色的,但我们整个游戏中可能有多个角色(hero,enemy,boss),每个角色都有一组动画(多个图片资源)。

这一节我们实现一个资源加载器,在游戏启动之前,统一加载全部图片资源到内存中,并有效管理。

首先,我们将角色,行为,图片路径等信息定义到一个xml文件中:

<animations speed="50">
    <jasmin>
        <start>
            <image>jasmin/aa1.png</image>
            <image>jasmin/aa1.png</image>
            <image>jasmin/aa1.png</image>
            <image>jasmin/aa1.png</image>
            <image>jasmin/aa1.png</image>
            <image>jasmin/aa1.png</image>
            <image>jasmin/aa1.png</image>
            <image>jasmin/aa2.png</image>
            <image>jasmin/aa2.png</image>
            <image>jasmin/aa3.png</image>
            <image>jasmin/aa4.png</image>
            <image>jasmin/aa4.png</image>
        </start>
        <walk>
            <image>jasmin/002.png</image>
            <image>jasmin/003.png</image>
            <image>jasmin/004.png</image>
            <image>jasmin/005.png</image>
            <image>jasmin/006.png</image>
            <image>jasmin/007.png</image>
        </walk>
        <idle>
            <image>jasmin/042.png</image>
            <image>jasmin/042.png</image>
            <image>jasmin/042.png</image>
            <image>jasmin/043.png</image>
            <image>jasmin/043.png</image>
            <image>jasmin/043.png</image>
            <image>jasmin/044.png</image>
            <image>jasmin/044.png</image>
            <image>jasmin/044.png</image>
            <image>jasmin/042.png</image>
            <image>jasmin/042.png</image>
            <image>jasmin/042.png</image>
            <image>jasmin/043.png</image>
            <image>jasmin/043.png</image>
            <image>jasmin/043.png</image>
            <image>jasmin/044.png</image>
            <image>jasmin/044.png</image>
            <image>jasmin/044.png</image>
            <image>jasmin/042.png</image>
            <image>jasmin/042.png</image>
            <image>jasmin/042.png</image>
            <image>jasmin/043.png</image>
            <image>jasmin/043.png</image>
            <image>jasmin/043.png</image>
            <image>jasmin/044.png</image>
            <image>jasmin/044.png</image>
            <image>jasmin/044.png</image>
            <image>jasmin/037.png</image>
            <image>jasmin/037.png</image>
            <image>jasmin/038.png</image>
            <image>jasmin/038.png</image>
            <image>jasmin/039.png</image>
            <image>jasmin/039.png</image>
            <image>jasmin/040.png</image>
            <image>jasmin/040.png</image>
            <image>jasmin/041.png</image>
            <image>jasmin/041.png</image>
        </idle>
        <run>
            <image>jasmin/008.png</image>
            <image>jasmin/009.png</image>
            <image>jasmin/010.png</image>
            <image>jasmin/011.png</image>
            <image>jasmin/012.png</image>
        </run>
        <jump>
            <image>jasmin/056.png</image>
            <image>jasmin/057.png</image>
            <image>jasmin/057.png</image>
            <image>jasmin/058.png</image>
            <image>jasmin/058.png</image>
            <image>jasmin/058.png</image>
            <image>jasmin/058.png</image>
        </jump>
        <attk>
            <image>jasmin/106.png</image>
            <image>jasmin/106.png</image>
            <image>jasmin/107.png</image>
            <image>jasmin/107.png</image>
        </attk>
        <attkRun>
            <image>jasmin/185.png</image>
            <image>jasmin/186.png</image>
            <image>jasmin/187.png</image>
            <image>jasmin/187.png</image>
        </attkRun>
        <attkKick>
            <image>jasmin/109.png</image>
            <image>jasmin/110.png</image>
            <image>jasmin/111.png</image>
            <image>jasmin/112.png</image>
        </attkKick>
        <attkKickLast>
            <image>jasmin/113.png</image>
            <image>jasmin/114.png</image>
            <image>jasmin/115.png</image>
        </attkKickLast>
        <block>
            <image>jasmin/027.png</image>
            <image>jasmin/027.png</image>
            <image>jasmin/028.png</image>
            <image>jasmin/028.png</image>
            <image>jasmin/028.png</image>
            <image>jasmin/028.png</image>
            <image>jasmin/028.png</image>
            <image>jasmin/028.png</image>
            <image>jasmin/028.png</image>
            <image>jasmin/028.png</image>
            <image>jasmin/028.png</image>
            <image>jasmin/028.png</image>
            <image>jasmin/028.png</image>
            <image>jasmin/028.png</image>
            <image>jasmin/028.png</image>
            <image>jasmin/028.png</image>
        </block>
        <attkShoot>
            <image>jasmin/jasmsh/01.png</image>
            <image>jasmin/jasmsh/02.png</image>
        </attkShoot>
        <dead>
            <image>jasmin/027.png</image>
            <image>jasmin/028.png</image>
            <image>jasmin/030.png</image>
            <image>jasmin/031.png</image>
            <image>jasmin/032.png</image>
            <image>jasmin/032.png</image>
            <image>jasmin/032.png</image>
            <image>jasmin/032.png</image>
            <image>jasmin/032.png</image>
            <image>jasmin/032.png</image>
            <image>jasmin/032.png</image>
            <image>jasmin/032.png</image>
        </dead>
    </jasmin>
</animations>

接下来,我们定义一个读取xml文档的类:

package utils;
import org.w3c.dom.Document;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import java.io.File;
public class XMLHandler {
    private Document doc;
    public XMLHandler(String filename){
        try{
            File toRead = new File("src/res/" + filename);
            DocumentBuilderFactory dbFactory = DocumentBuilderFactory.newInstance();
            DocumentBuilder dBuilder = dbFactory.newDocumentBuilder();
            doc = dBuilder.parse(toRead);
            doc.getDocumentElement().normalize();
        }
        catch(Exception e){
            e.printStackTrace();
        }
    }
    public Document getDoc(){
        return this.doc;
    }
}

接下来我们定义一个加载资源的类

package world;
import config.Config;
import org.w3c.dom.Element;
import org.w3c.dom.NodeList;
import sprite.Animation;
import sprite.Animator;
import sprite.GameAnimations;
import sprite.Image;
import utils.XMLHandler;
import java.util.ArrayList;
import java.util.HashMap;
public class RessourceLoader {
    public static void main(String[] args) {
        RessourceLoader ressourceLoader = new RessourceLoader();
        ressourceLoader.loadAnimations();
    }
    public void loadAnimations(){
        XMLHandler rxml2 = new XMLHandler(Config.ANIMATION_FILE);
        String animSpeed = rxml2.getDoc().getDocumentElement().getAttribute("speed");
        int speed  = Integer.parseInt(animSpeed);
        NodeList nList2 = rxml2.getDoc().getDocumentElement().getChildNodes();
        Element child2;
        NodeList iList;
        Animation anim;
        ArrayList<Image> images = new ArrayList<Image>();
        ArrayList<GameAnimations> anims = new ArrayList<GameAnimations>();
        String aniName = null;
        Image img;
        for (int i = 0; i < nList2.getLength(); i++) {
            if (nList2.item(i).getNodeType() == 1) {
                String keyName = nList2.item(i).getNodeName();
                NodeList nList3 = nList2.item(i).getChildNodes();
                for (int j = 0; j < nList3.getLength(); j++) {
                    if (nList3.item(j).getNodeType() == 1) {
                        aniName = nList3.item(j).getNodeName();
                        Element child3 = (Element) nList3.item(j);
                        iList = child3.getChildNodes();
                        for(int k = 0; k < iList.getLength(); k++){
                            if (iList.item(k).getNodeType() == 1){
//                              System.out.println("LOAD Image: "+iList.item(k).getTextContent());
                                img = new Image("res/" + iList.item(k).getTextContent());
                                images.add(img);
                            }
                        }
                        anims.add(new GameAnimations(aniName,new Animation(images)));
                        images = new ArrayList<Image>();
                    }
                }
                Animator.gameAnimations.put(keyName, anims);
                anims = new ArrayList<GameAnimations>();
            }
        }
    }
}

Animator中新增HashMap<String, ArrayList<GameAnimations>> gameAnimations对象。

该对象的key是角色的名字,值是一个GameAnimations列表

public static HashMap<String, ArrayList<GameAnimations>> gameAnimations = new HashMap<String,ArrayList<GameAnimations>>();

GameAnimations类定义如下:

就是一个动画名和动画对象,动画名是walk这样的行为

package sprite;
public class GameAnimations {
    public Animation animations;
    public String animname;
    public GameAnimations(String animname, Animation img){
        this.animname = animname;
        this.animations = img;
    }
}

加载资源时我们填充HashMap<String, ArrayList<GameAnimations>> gameAnimations,参见上面RessourceLoader类的最后。

Animator.gameAnimations.put(keyName, anims);

构造某个角色的Animator时,我们添加这个角色的所有行为动画到_frames中

public Animator(String name){
        super();
        _frames = new HashMap<String, Animation>();
        _tm = new Timer(120,this);
        _tm.start();
        for(int i = 0;i<=gameAnimations.get(name).size()-1;i++){
            this.addAnimation(gameAnimations.get(name).get(i).animations, gameAnimations.get(name).get(i).animname);
        }
    }
    public void addAnimation(Animation anim, String name){
        _frames.put(name,anim);
    }


整合到一起,测试一下


修改GameApp

1.定义RessourceLoader对象

private static RessourceLoader _resLoader = new RessourceLoader();

2.加载资源,设定jasmin角色动画,设置jasmin角色当前动画行为为idle

_resLoader.loadAnimations();
    animator = new Animator("jasmin");
    animator.setAnimation("idle");

3.按下左右箭头,切换角色行为

if(_input.isKeyUp(KeyEvent.VK_RIGHT))
            {
                System.out.println("VK_RIGHT");
                animator.setAnimation("walk");
            }
            else if(_input.isKeyUp(KeyEvent.VK_LEFT))
            {
                animator.setAnimation("idle");
                System.out.println("VK_LEFT");
            }

如果您迷路了请看这里:


项目源码


项目源码

目录
相关文章
|
2月前
|
Java 开发者
在 Java 中,一个类可以实现多个接口吗?
这是 Java 面向对象编程的一个重要特性,它提供了极大的灵活性和扩展性。
165 57
|
18天前
|
JSON Java Apache
Java基础-常用API-Object类
继承是面向对象编程的重要特性,允许从已有类派生新类。Java采用单继承机制,默认所有类继承自Object类。Object类提供了多个常用方法,如`clone()`用于复制对象,`equals()`判断对象是否相等,`hashCode()`计算哈希码,`toString()`返回对象的字符串表示,`wait()`、`notify()`和`notifyAll()`用于线程同步,`finalize()`在对象被垃圾回收时调用。掌握这些方法有助于更好地理解和使用Java中的对象行为。
|
2月前
|
存储 缓存 安全
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
69 8
|
2月前
|
Java Android开发
Eclipse 创建 Java 类
Eclipse 创建 Java 类
32 0
|
6天前
|
监控 Java
java异步判断线程池所有任务是否执行完
通过上述步骤,您可以在Java中实现异步判断线程池所有任务是否执行完毕。这种方法使用了 `CompletionService`来监控任务的完成情况,并通过一个独立线程异步检查所有任务的执行状态。这种设计不仅简洁高效,还能确保在大量任务处理时程序的稳定性和可维护性。希望本文能为您的开发工作提供实用的指导和帮助。
44 17
|
16天前
|
Java
Java—多线程实现生产消费者
本文介绍了多线程实现生产消费者模式的三个版本。Version1包含四个类:`Producer`(生产者)、`Consumer`(消费者)、`Resource`(公共资源)和`TestMain`(测试类)。通过`synchronized`和`wait/notify`机制控制线程同步,但存在多个生产者或消费者时可能出现多次生产和消费的问题。 Version2将`if`改为`while`,解决了多次生产和消费的问题,但仍可能因`notify()`随机唤醒线程而导致死锁。因此,引入了`notifyAll()`来唤醒所有等待线程,但这会带来性能问题。
Java—多线程实现生产消费者
|
2天前
|
缓存 安全 算法
Java 多线程 面试题
Java 多线程 相关基础面试题
|
18天前
|
安全 Java Kotlin
Java多线程——synchronized、volatile 保障可见性
Java多线程中,`synchronized` 和 `volatile` 关键字用于保障可见性。`synchronized` 保证原子性、可见性和有序性,通过锁机制确保线程安全;`volatile` 仅保证可见性和有序性,不保证原子性。代码示例展示了如何使用 `synchronized` 和 `volatile` 解决主线程无法感知子线程修改共享变量的问题。总结:`volatile` 确保不同线程对共享变量操作的可见性,使一个线程修改后,其他线程能立即看到最新值。
|
18天前
|
消息中间件 缓存 安全
Java多线程是什么
Java多线程简介:本文介绍了Java中常见的线程池类型,包括`newCachedThreadPool`(适用于短期异步任务)、`newFixedThreadPool`(适用于固定数量的长期任务)、`newScheduledThreadPool`(支持定时和周期性任务)以及`newSingleThreadExecutor`(保证任务顺序执行)。同时,文章还讲解了Java中的锁机制,如`synchronized`关键字、CAS操作及其实现方式,并详细描述了可重入锁`ReentrantLock`和读写锁`ReadWriteLock`的工作原理与应用场景。
|
19天前
|
安全 Java 编译器
深入理解Java中synchronized三种使用方式:助您写出线程安全的代码
`synchronized` 是 Java 中的关键字,用于实现线程同步,确保多个线程互斥访问共享资源。它通过内置的监视器锁机制,防止多个线程同时执行被 `synchronized` 修饰的方法或代码块。`synchronized` 可以修饰非静态方法、静态方法和代码块,分别锁定实例对象、类对象或指定的对象。其底层原理基于 JVM 的指令和对象的监视器,JDK 1.6 后引入了偏向锁、轻量级锁等优化措施,提高了性能。
42 3