mirror of
https://github.com/mcMMO-Dev/mcMMO.git
synced 2024-11-30 00:56:47 +01:00
Get all logs in Tree Feller, and optimize performance
Tree Feller has been shown, both anecdotally and with timings, to put a strain on the server, and therefore is worthy of the effort of optimization. Prior to this change, on jungle trees, Tree Feller would take around 20-40 milliseconds to process a Jungle Tree after the JIT kicked in, and around 15-25 milliseconds for a normal tree. Additionally, logs would be left up in the air for jungle trees. After this change, Tree Feller takes 2-5 milliseconds on normal trees, and 10-15 milliseconds on jungle trees, and no logs are left up in the air.
This commit is contained in:
parent
44e3e4d1e1
commit
2fba223d82
@ -1,9 +1,11 @@
|
|||||||
package com.gmail.nossr50.skills.woodcutting;
|
package com.gmail.nossr50.skills.woodcutting;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
import java.util.LinkedHashSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import org.bukkit.Material;
|
import org.bukkit.Material;
|
||||||
|
import org.bukkit.block.BlockFace;
|
||||||
import org.bukkit.block.BlockState;
|
import org.bukkit.block.BlockState;
|
||||||
import org.bukkit.enchantments.Enchantment;
|
import org.bukkit.enchantments.Enchantment;
|
||||||
import org.bukkit.inventory.ItemStack;
|
import org.bukkit.inventory.ItemStack;
|
||||||
@ -121,56 +123,43 @@ public final class Woodcutting {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Processes Tree Feller for generic Trees
|
* The x/y differences to the blocks in a flat cylinder around the center
|
||||||
*
|
* block, which is excluded.
|
||||||
* @param blockState Block being checked
|
|
||||||
* @param treeFellerBlocks List of blocks to be removed
|
|
||||||
*/
|
*/
|
||||||
protected static void processRegularTrees(BlockState blockState, List<BlockState> treeFellerBlocks) {
|
private static final int[][] directions = {
|
||||||
List<BlockState> futureCenterBlocks = new ArrayList<BlockState>();
|
new int[] {-2, -1}, new int[] {-2, 0}, new int[] {-2, 1},
|
||||||
|
new int[] {-1, -2}, new int[] {-1, -1}, new int[] {-1, 0}, new int[] {-1, 1}, new int[] {-1, 2},
|
||||||
// Handle the blocks around 'block'
|
new int[] { 0, -2}, new int[] { 0, -1}, new int[] { 0, 1}, new int[] { 0, 2},
|
||||||
for (int y = 0; y <= 1; y++) {
|
new int[] { 1, -2}, new int[] { 1, -1}, new int[] { 1, 0}, new int[] { 1, 1}, new int[] { 1, 2},
|
||||||
for (int x = -1; x <= 1; x++) {
|
new int[] { 2, -1}, new int[] { 2, 0}, new int[] { 2, 1},
|
||||||
for (int z = -1; z <= 1; z++) {
|
};
|
||||||
BlockState nextBlock = blockState.getBlock().getRelative(x, y, z).getState();
|
|
||||||
handleBlock(nextBlock, futureCenterBlocks, treeFellerBlocks);
|
|
||||||
|
|
||||||
if (WoodcuttingManager.treeFellerReachedThreshold) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Recursive call for each log found
|
|
||||||
for (BlockState futureCenterBlock : futureCenterBlocks) {
|
|
||||||
if (WoodcuttingManager.treeFellerReachedThreshold) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
processRegularTrees(futureCenterBlock, treeFellerBlocks);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Processes Tree Feller for Red Mushrooms (Dome Shaped)
|
* Processes Tree Feller in a recursive manner
|
||||||
*
|
*
|
||||||
* @param blockState Block being checked
|
* @param blockState Block being checked
|
||||||
* @param treeFellerBlocks List of blocks to be removed
|
* @param treeFellerBlocks List of blocks to be removed
|
||||||
*/
|
*/
|
||||||
protected static void processRedMushroomTrees(BlockState blockState, List<BlockState> treeFellerBlocks) {
|
protected static void processTree(BlockState blockState, LinkedHashSet<BlockState> treeFellerBlocks) {
|
||||||
List<BlockState> futureCenterBlocks = new ArrayList<BlockState>();
|
List<BlockState> futureCenterBlocks = new ArrayList<BlockState>();
|
||||||
|
|
||||||
// Handle the blocks around 'block'
|
// Check the block up and take different behavior (smaller search) if it's a log
|
||||||
for (int y = 0; y <= 1; y++) {
|
if (handleBlock(blockState.getBlock().getRelative(BlockFace.UP).getState(), futureCenterBlocks, treeFellerBlocks)) {
|
||||||
for (int x = -1; x <= 1; x++) {
|
for (int[] dir : directions) {
|
||||||
for (int z = -1; z <= 1; z++) {
|
handleBlock(blockState.getBlock().getRelative(dir[0], 0, dir[1]).getState(), futureCenterBlocks, treeFellerBlocks);
|
||||||
BlockState nextBlock = blockState.getBlock().getRelative(x, y, z).getState();
|
|
||||||
BlockState otherNextBlock = blockState.getBlock().getRelative(x, y - (y * 2), z).getState();
|
|
||||||
|
|
||||||
handleBlock(nextBlock, futureCenterBlocks, treeFellerBlocks);
|
if (WoodcuttingManager.treeFellerReachedThreshold) {
|
||||||
handleBlock(otherNextBlock, futureCenterBlocks, treeFellerBlocks);
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// Cover DOWN
|
||||||
|
handleBlock(blockState.getBlock().getRelative(BlockFace.DOWN).getState(), futureCenterBlocks, treeFellerBlocks);
|
||||||
|
// Search in a cube
|
||||||
|
for (int y = -1; y <= 1; y++) {
|
||||||
|
for (int[] dir : directions) {
|
||||||
|
handleBlock(blockState.getBlock().getRelative(dir[0], y, dir[1]).getState(), futureCenterBlocks, treeFellerBlocks);
|
||||||
|
|
||||||
if (WoodcuttingManager.treeFellerReachedThreshold) {
|
if (WoodcuttingManager.treeFellerReachedThreshold) {
|
||||||
return;
|
return;
|
||||||
@ -185,7 +174,7 @@ public final class Woodcutting {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
processRedMushroomTrees(futureCenterBlock, treeFellerBlocks);
|
processTree(futureCenterBlock, treeFellerBlocks);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -196,7 +185,7 @@ public final class Woodcutting {
|
|||||||
* @param inHand tool being used
|
* @param inHand tool being used
|
||||||
* @return True if the tool can sustain the durability loss
|
* @return True if the tool can sustain the durability loss
|
||||||
*/
|
*/
|
||||||
protected static boolean handleDurabilityLoss(List<BlockState> treeFellerBlocks, ItemStack inHand) {
|
protected static boolean handleDurabilityLoss(LinkedHashSet<BlockState> treeFellerBlocks, ItemStack inHand) {
|
||||||
Material inHandMaterial = inHand.getType();
|
Material inHandMaterial = inHand.getType();
|
||||||
|
|
||||||
if (inHandMaterial == Material.AIR) {
|
if (inHandMaterial == Material.AIR) {
|
||||||
@ -221,27 +210,35 @@ public final class Woodcutting {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handle a block addition to the list of blocks to be removed and to the list of blocks used for future recursive calls of 'processRecursively()'
|
* Handle a block addition to the list of blocks to be removed and to the
|
||||||
|
* list of blocks used for future recursive calls of
|
||||||
|
* 'processRecursively()'
|
||||||
*
|
*
|
||||||
* @param blockState Block to be added
|
* @param blockState Block to be added
|
||||||
* @param futureCenterBlocks List of blocks that will be used to call 'processRecursively()'
|
* @param futureCenterBlocks List of blocks that will be used to call
|
||||||
|
* 'processRecursively()'
|
||||||
* @param treeFellerBlocks List of blocks to be removed
|
* @param treeFellerBlocks List of blocks to be removed
|
||||||
|
* @return true if and only if the given blockState was a Log not already
|
||||||
|
* in treeFellerBlocks.
|
||||||
*/
|
*/
|
||||||
private static void handleBlock(BlockState blockState, List<BlockState> futureCenterBlocks, List<BlockState> treeFellerBlocks) {
|
private static boolean handleBlock(BlockState blockState, List<BlockState> futureCenterBlocks, LinkedHashSet<BlockState> treeFellerBlocks) {
|
||||||
if (!BlockUtils.affectedByTreeFeller(blockState) || mcMMO.getPlaceStore().isTrue(blockState) || treeFellerBlocks.contains(blockState)) {
|
if (mcMMO.getPlaceStore().isTrue(blockState) || treeFellerBlocks.contains(blockState)) {
|
||||||
return;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
treeFellerBlocks.add(blockState);
|
|
||||||
|
|
||||||
if (treeFellerBlocks.size() > treeFellerThreshold) {
|
if (treeFellerBlocks.size() > treeFellerThreshold) {
|
||||||
WoodcuttingManager.treeFellerReachedThreshold = true;
|
WoodcuttingManager.treeFellerReachedThreshold = true;
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Without this check Tree Feller propagates through leaves until the threshold is hit
|
// Without this check Tree Feller propagates through leaves until the threshold is hit
|
||||||
if (BlockUtils.isLog(blockState)) {
|
if (BlockUtils.isLog(blockState)) {
|
||||||
|
treeFellerBlocks.add(blockState);
|
||||||
futureCenterBlocks.add(blockState);
|
futureCenterBlocks.add(blockState);
|
||||||
}
|
return true;
|
||||||
|
} else if (BlockUtils.isLeaves(blockState)) {
|
||||||
|
treeFellerBlocks.add(blockState);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
package com.gmail.nossr50.skills.woodcutting;
|
package com.gmail.nossr50.skills.woodcutting;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.LinkedHashSet;
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
import org.bukkit.Material;
|
import org.bukkit.Material;
|
||||||
import org.bukkit.block.Block;
|
import org.bukkit.block.Block;
|
||||||
@ -73,24 +72,9 @@ public class WoodcuttingManager extends SkillManager {
|
|||||||
*/
|
*/
|
||||||
public void processTreeFeller(BlockState blockState) {
|
public void processTreeFeller(BlockState blockState) {
|
||||||
Player player = getPlayer();
|
Player player = getPlayer();
|
||||||
List<BlockState> treeFellerBlocks = new ArrayList<BlockState>();
|
LinkedHashSet<BlockState> treeFellerBlocks = new LinkedHashSet<BlockState>();
|
||||||
|
|
||||||
switch (blockState.getType()) {
|
Woodcutting.processTree(blockState, treeFellerBlocks);
|
||||||
case LOG:
|
|
||||||
case HUGE_MUSHROOM_1:
|
|
||||||
Woodcutting.processRegularTrees(blockState, treeFellerBlocks);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case HUGE_MUSHROOM_2:
|
|
||||||
Woodcutting.processRedMushroomTrees(blockState, treeFellerBlocks);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
if (ModUtils.isCustomLogBlock(blockState)) {
|
|
||||||
Woodcutting.processRegularTrees(blockState, treeFellerBlocks);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the player is trying to break too many blocks
|
// If the player is trying to break too many blocks
|
||||||
if (treeFellerReachedThreshold) {
|
if (treeFellerReachedThreshold) {
|
||||||
@ -122,7 +106,7 @@ public class WoodcuttingManager extends SkillManager {
|
|||||||
*
|
*
|
||||||
* @param treeFellerBlocks List of blocks to be dropped
|
* @param treeFellerBlocks List of blocks to be dropped
|
||||||
*/
|
*/
|
||||||
private void dropBlocks(List<BlockState> treeFellerBlocks) {
|
private void dropBlocks(LinkedHashSet<BlockState> treeFellerBlocks) {
|
||||||
Player player = getPlayer();
|
Player player = getPlayer();
|
||||||
int xp = 0;
|
int xp = 0;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user