gpt4 book ai didi

java - 创建具有未定义数量的子对象的层次结构对象

转载 作者:搜寻专家 更新时间:2023-10-30 21:31:43 24 4
gpt4 key购买 nike

我目前正在研究将 Valve Map 格式(.vmf 文件)解析为 Java 可读对象的“代码解析器”。

在 vmf 文件中,

  • 有两种类型的对象:类和属性。
  • 类有一个名称并且可以包含其他类和属性。
  • 属性有一个名称和无限数量的值。

因此我创建了一个VMFClass 对象类和一个VMFProperty 对象类。我用自己创建的 HierarchyObject 创建了一个列表,其中包含 VMFClass/VMFProperty 对象、一个 UUID 和 parentUUID。VMFClass 对象包含 2 个列表,一个带有子 VMFClasses,一个带有属性。

我的问题是我不知道如何实现一个类包含它的所有子类,因为我不知道子类有多少子类等等...

这是我的代码(Github):

层次对象:

package net.minecraft.sourcecraftreloaded.utils;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

public class HierarchyObject {
private static Map<Long, Long> usedUUIDs = new HashMap<>();
private long parentUUID;
private long UUID;
private Object object;

/**
*
* @param Object
* @param parent -1 is maximum level
*/
public HierarchyObject(Object object, long parent) {
this.object = object;
this.parentUUID = parent;
while (true) {
long random = (long) (Math.random() * Long.MAX_VALUE);
if (usedUUIDs.containsKey(random)) {
this.UUID = random;
usedUUIDs.put(random, parent);
break;
}
}
}

public long getUUID() {
return UUID;
}
public long getParentUUID() {
return parentUUID;
}

public static long getParentUUIDbyUUID(long UUID) {
if (usedUUIDs.containsKey(UUID)) {
return usedUUIDs.get(UUID);
}
return -1;
}

public Object getObject() {
return object;
}

public static boolean hasChild(long UUID){
if(usedUUIDs.containsValue(UUID)){
return true;
}
if(UUID == -1){
return true;
}
return false;
}

public boolean hasChild(){
return hasChild(this.UUID);
}

public static long[] getChildUUIDs(long UUID){
if(hasChild(UUID)){
List<Long> cUUIDs = new ArrayList<>();
for(int i = 0; i < usedUUIDs.size(); i++){
for (Map.Entry<Long, Long> e : usedUUIDs.entrySet()) {
if(e.getValue().longValue() == UUID){
cUUIDs.add(e.getKey());
}
}
}
return ListUtils.toPrimitivebyList(cUUIDs);
}
return null;
}
}

VMFProperty:

package net.minecraft.sourcecraftreloaded.source;

public class VMFProperty{

private String name;
private String[] values;

public VMFProperty(String name, String... values) {
this.name = name;
this.values = values;
}

public String getName() {
return name;
}
public String[] getValues() {
return values;
}

@Override
public boolean equals(Object paramObject){
if(paramObject instanceof VMFProperty){
return ((VMFProperty)paramObject).name.equals(this.name) && ((VMFProperty)paramObject).values.equals(this.values);
}
return false;
}
}

VMFClass:

package net.minecraft.sourcecraftreloaded.source;

import java.util.List;

public class VMFClass{
private List<VMFClass> classes;
private List<VMFProperty> properties;
private String name;

public VMFClass(String name, List<VMFClass> classes, List<VMFProperty> properties) {
this.name = name;
this.classes = classes;
this.properties = properties;
}

public String getName() {
return name;
}
public List<VMFClass> getClasses() {
return classes;
}
public List<VMFProperty> getProperties() {
return properties;
}
public void add(VMFClass vmfclass) {
classes.add(vmfclass);
}
public void add(VMFProperty vmfproperty) {
properties.add(vmfproperty);
}
public void remove(VMFClass vmfclass) {
classes.remove(vmfclass);
}
public void remove(VMFProperty vmfproperty) {
properties.remove(vmfproperty);
}

@Override
public boolean equals(Object paramObject){
if(paramObject instanceof VMFClass){
return ((VMFClass)paramObject).properties.equals(this.properties) && ((VMFClass)paramObject).classes.equals(this.classes) && ((VMFClass)paramObject).name.equals(this.name);
}
return false;
}
}

VMFObject(执行所有代码的类):

package net.minecraft.sourcecraftreloaded.source;

import java.io.File;
import java.util.ArrayList;
import java.util.List;

import net.minecraft.sourcecraftreloaded.utils.HierarchyObject;

public class VMFObject {
private String rawfile = "";
private List<VMFClass> toplevelclasses;

private static final String INVALID_CHARS = "\\*,;<>|?=`´#'+~^°!§$%&()[].:-_";

public VMFObject(List<VMFClass> toplevelclasses) {
this.toplevelclasses = toplevelclasses;
}

public VMFObject() {
this(new ArrayList<VMFClass>());
}

public void write(File file) {
VMFWriter.write(file, rawfile);
}

public VMFObject read(File file) throws VMFParsingException {
this.rawfile = VMFReader.read(file);
parse();
return this;
}
public List<VMFClass> getClasses() {
return toplevelclasses;
}

private void parse() throws VMFParsingException {
evaluate();
get();
}

private void evaluate() throws VMFParsingException {
char[] textchars = rawfile.toCharArray();
int[] c = new int[]{0, 0, 0};
int line = 0;
int linepos = 0;
for (int i : textchars) {
linepos++;
if (textchars[i] == '\n') {
line++;
linepos = 0;
c[3] = 0;
if (c[3] % 2 != 0) {
throw new VMFParsingException("Invalid quotes on line" + line + ":" + linepos);
}
}
if (textchars[i] == '{') {
c[1]++;
}
if (textchars[i] == '}') {
c[2]++;
}
if (textchars[i] == '"') {
c[3]++;
if (c[1] - c[2] == 0) {

}
}
if (textchars[i] == '/' && textchars[i + 1] == '/') {
while (true) {
i++;
if (textchars[i] == '\n') {
break;
}
}
}
if (textchars[i] == '/' && textchars[i + 1] == ' ') {
throw new VMFParsingException("Invalid Character '/' on line" + line + ":" + linepos);
}
if (INVALID_CHARS.indexOf(textchars[i]) != -1) {
throw new VMFParsingException("Invalid Character '" + textchars[i] + "' on line" + line + ":" + linepos);
}
}
if (c[1] != c[2]) {
throw new VMFParsingException("Unbalanced brackets in vmf File");
}
}

public void add(VMFClass vmfclass) {
toplevelclasses.add(vmfclass);
}

private void get() throws VMFParsingException {
List<HierarchyObject> content = new ArrayList<>();
long curparent = -1;
String[] text = rawfile.split("\n");
for (int i = 0; i < text.length; i++) {
String line = text[i].trim();
if (line.startsWith("//")) {
continue;
} else {
byte quotec = 0;
char[] linechar = line.toCharArray();
boolean readp = false;
List<String> reads = new ArrayList<>();
byte creads = 0;
for (int y = 0; y < linechar.length; y++) {
if (linechar[y] == '/' && linechar[y + 1] == '/') {
break;
}
if (linechar[y] == '"') {
quotec++;
if (quotec % 2 == 0) {
readp = false;
creads++;
} else {
readp = true;
}
}
if (readp) {
reads.set(creads, reads.get(creads) + linechar[y]);
}
if (linechar[y] == '{') {
HierarchyObject object = new HierarchyObject(new VMFClass(line.substring(line.substring(0, y).lastIndexOf(' '), y).trim(), null, null), curparent);
content.add(object);
curparent = object.getUUID();
}
if (linechar[y] == '}') {
curparent = HierarchyObject.getParentUUIDbyUUID(curparent);
}

}
content.add(new HierarchyObject(new VMFProperty(reads.remove(0), reads.toArray(new String[reads.size()])), curparent));
}
}
buildObject(content);
}

private void buildObject(List<HierarchyObject> content) {
long curUUID = -1;
for(int i = 0; i < HierarchyObject.getChildUUIDs(curUUID).length; i++){
HierarchyObject.getChildUUIDs(curUUID);
}
//TODO implement
}
}

//TODO 部分是层次结构对象应该“转换”为实际对象的地方。

最佳答案

概览

在我看来,您的类布局过于复杂。

让我们尝试简化它...

您所描述的 VMF 模型本质上是一个链表树。

这是模型的样子:

                    [.vmf file] (root)
/ \
_____/ \ _____
/ \
/ \
(VMFClass) (VMFClass)
/ \ / \
/ \ / \
/ \ / \
(VMFClass) (VMFProperties) (VMFClass) (VMFProperties)
/ \
/ \
/ \
(VMFClass) (VMFProperties)

您需要什么:

  • 一个 Parser 类(在您的例子中,您有 VMFObject,但我们称这个类为 VMFParser)。
  • 您拥有的 VMFClassVMFProperty 类没问题。

你不需要的:

  • HierarchyObject 类。 VMFParser 可以是层次结构(例如 linked-list 树模型)的主要 Controller 和容器。
  • 所有 UUID(父 UUID、子 UUID 等)这些都是复杂的东西,但我明白你为什么要拥有它们。您不需要它们来跟踪层次结构——Java 会为我们做这件事!!

VMF类

public class VMFClass
{
// name of the class
private String name;

// reference back up to the parent
private VMFClass parentClass = null;

// all direct children go here
private List<VMFClass> children = new ArrayList<VMFClass>();

// I don't think you need a list of properties here since your VMFProperty class holds onto an array of properties
private VMFProperty properties;

// set the parent of this class
public void setParent (VMFClass parent)
{
this.parentClass = parent;
}

// get the direct children
public List<VMFClass> getChildren()
{
return this.children;
}

// rest of methods...
}

VMFParser

class VMFParser
{
private String rawfile = "";

// this is really the container for everything - think of it as the file shell
private VMFClass root = new VMFClass("root", null, null);

// construct yourself with the file
public VMFParser (String fileName)
{
this.rawfile = fileName;
}

public void parse ()
{
// all the parsing code goes here
read();
evaluate();
get();

// now at this point your hierarchy is built and stored in the
// root object in this class.

// Use the traverse method to go through it
}

private void get() throws VMFParsingException
{
// keep a reference to the current VMFClass parent
// starts out as root
VMFClass currParentClass = root;

// main parse loop
for (...)
{
// if you find a class
VMFClass currClass = new VMFClass(/* params here */);

// add this class to the parent
currParentClass.add(currClass);

// set the parent of this class
currClass.setParent(currParentClass);

// if you find a property
// parse and add all the properties to the property
VMFProperty property = new VMFProperty (/* value params here */);

// attach this property to the last VMF class that got parsed
currClass.setPoperties(property);

// If you nest deeper into classes, then the parent becomes the current class
currParentClass = currClass;

// If you go back out of a class
currParentClass = currClass.getParent();
}
}

// Traverse the hierarchy
public void traverse ()
{
traverseTree(root);
}

private void traverseTree (VMFClass root)
{
System.out.println("Class Name: " + root.getName());

// print out any properties
VMFProperty prop = root.getProperty();

if (prop != null)
{
System.out.println("Property Name: " + prop.getName());

String [] props = prop.getValues();
for (String s: props)
{
System.out.println("Value: " + s);
}
}

// get all child classes
List<VMFClass> children = root.getChildren();
for (VMFClass c: children)
{
traverseTree(c);
}
}
}

客户端代码

示例

public static void main(String[] args)
{
VMFParser vmfParser = null;
try
{
vmfParser = new VMFParser("myFile.vmf");
vmfParser.parse();

// access the vmfParser for the hierarchy
vmfParser.traverse();
}
catch (VMFParsingException vpe)
{
// do something here
vpe.printStackTrace();
}
finally
{
// clean up...
}
}

关于java - 创建具有未定义数量的子对象的层次结构对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36810400/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com