2015-06-07 22:51:24 +02:00
From 44f7767ad24c505b3433c914d3d9bb7a2af305b1 Mon Sep 17 00:00:00 2001
2014-10-20 00:58:49 +02:00
From: Aikar <aikar@aikar.co>
2015-03-08 02:16:09 +01:00
Date: Sun, 8 Mar 2015 04:03:56 -0500
2014-10-20 00:58:49 +02:00
Subject: [PATCH] Player lookup improvements
Minecraft and CraftBukkit both use Arrays to store online players,
and any time a player needs to be looked up by name or UUID,
the system iterates all online players and does a name or UUID comparison.
This is very ineffecient and can reduce performance on servers with high player count.
By using a map based approach for player lookups, player lookup should
be consistent in performance regardless of how many players are online.
diff --git a/src/main/java/net/minecraft/server/PlayerList.java b/src/main/java/net/minecraft/server/PlayerList.java
2015-06-07 22:51:24 +02:00
index ab1d9a5..617f26c 100644
2014-10-20 00:58:49 +02:00
--- a/src/main/java/net/minecraft/server/PlayerList.java
+++ b/src/main/java/net/minecraft/server/PlayerList.java
2014-12-13 05:52:49 +01:00
@@ -49,6 +49,31 @@ public abstract class PlayerList {
2015-03-08 02:16:09 +01:00
private static final SimpleDateFormat g = new SimpleDateFormat("yyyy-MM-dd \'at\' HH:mm:ss z");
2014-10-20 00:58:49 +02:00
private final MinecraftServer server;
2015-06-06 19:15:22 +02:00
public final List<EntityPlayer> players = new java.util.concurrent.CopyOnWriteArrayList(); // CraftBukkit - ArrayList -> CopyOnWriteArrayList: Iterator safety
2014-10-20 00:58:49 +02:00
+ // PaperSpigot start - Player lookup improvements
+ public final Map<String, EntityPlayer> playerMap = new java.util.HashMap<String, EntityPlayer>() {
+ @Override
+ public EntityPlayer put(String key, EntityPlayer value) {
+ return super.put(key.toLowerCase(), value);
+ }
+
+ @Override
+ public EntityPlayer get(Object key) {
+ // put the .playerConnection check done in other places here
2014-11-28 02:17:45 +01:00
+ EntityPlayer player = super.get(key instanceof String ? ((String) key).toLowerCase() : key);
2014-10-20 00:58:49 +02:00
+ return (player != null && player.playerConnection != null) ? player : null;
+ }
+
+ @Override
+ public boolean containsKey(Object key) {
+ return get(key) != null;
+ }
+
+ @Override
+ public EntityPlayer remove(Object key) {
2014-11-28 02:17:45 +01:00
+ return super.remove(key instanceof String ? ((String) key).toLowerCase() : key);
2014-10-20 00:58:49 +02:00
+ }
+ };
+ // PaperSpigot end
2015-03-08 02:16:09 +01:00
private final Map<UUID, EntityPlayer> j = Maps.newHashMap();
2014-11-28 02:17:45 +01:00
private final GameProfileBanList k;
private final IpBanList l;
2015-03-08 02:16:09 +01:00
@@ -287,6 +312,7 @@ public abstract class PlayerList {
2014-11-28 02:17:45 +01:00
2015-02-04 00:53:51 +01:00
public void onPlayerJoin(EntityPlayer entityplayer, String joinMessage) { // CraftBukkit added param
2014-10-20 00:58:49 +02:00
this.players.add(entityplayer);
+ this.playerMap.put(entityplayer.getName(), entityplayer); // PaperSpigot
2015-03-08 02:16:09 +01:00
this.j.put(entityplayer.getUniqueID(), entityplayer);
// this.sendAll(new PacketPlayOutPlayerInfo(PacketPlayOutPlayerInfo.EnumPlayerInfoAction.ADD_PLAYER, new EntityPlayer[] { entityplayer})); // CraftBukkit - replaced with loop below
2014-10-20 00:58:49 +02:00
WorldServer worldserver = this.server.getWorldServer(entityplayer.dimension);
2015-03-08 02:16:09 +01:00
@@ -358,6 +384,7 @@ public abstract class PlayerList {
2014-10-20 00:58:49 +02:00
worldserver.kill(entityplayer);
worldserver.getPlayerChunkMap().removePlayer(entityplayer);
this.players.remove(entityplayer);
+ this.playerMap.remove(entityplayer.getName()); // PaperSpigot
2015-03-08 02:16:09 +01:00
UUID uuid = entityplayer.getUniqueID();
EntityPlayer entityplayer1 = (EntityPlayer) this.j.get(uuid);
@@ -395,6 +422,8 @@ public abstract class PlayerList {
2014-10-20 00:58:49 +02:00
EntityPlayer entityplayer;
2015-03-08 02:16:09 +01:00
+ // PaperSpigot start - Use exact lookup below
2014-11-28 02:17:45 +01:00
+ /*
2014-10-20 00:58:49 +02:00
for (int i = 0; i < this.players.size(); ++i) {
entityplayer = (EntityPlayer) this.players.get(i);
if (entityplayer.getUniqueID().equals(uuid)) {
2015-03-08 02:16:09 +01:00
@@ -406,6 +435,9 @@ public abstract class PlayerList {
2014-10-20 00:58:49 +02:00
while (iterator.hasNext()) {
entityplayer = (EntityPlayer) iterator.next();
2014-12-07 20:28:45 +01:00
+ */
+ if ((entityplayer = this.a(uuid)) != null) {
2015-03-08 02:16:09 +01:00
+ // PaperSpigot end
2014-12-07 20:28:45 +01:00
savePlayerFile(entityplayer); // CraftBukkit - Force the player's inventory to be saved
2014-10-20 00:58:49 +02:00
entityplayer.playerConnection.disconnect("You logged in from another location");
}
2015-06-07 22:51:24 +02:00
@@ -1020,6 +1052,7 @@ public abstract class PlayerList {
2014-10-20 00:58:49 +02:00
}
public EntityPlayer getPlayer(String s) {
+ if (true) { return playerMap.get(s); } // PaperSpigot
Iterator iterator = this.players.iterator();
EntityPlayer entityplayer;
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftOfflinePlayer.java b/src/main/java/org/bukkit/craftbukkit/CraftOfflinePlayer.java
2014-12-07 20:29:55 +01:00
index 8be0698..4a908c6 100644
2014-10-20 00:58:49 +02:00
--- a/src/main/java/org/bukkit/craftbukkit/CraftOfflinePlayer.java
+++ b/src/main/java/org/bukkit/craftbukkit/CraftOfflinePlayer.java
@@ -144,14 +144,10 @@ public class CraftOfflinePlayer implements OfflinePlayer, ConfigurationSerializa
}
public Player getPlayer() {
- for (Object obj : server.getHandle().players) {
- EntityPlayer player = (EntityPlayer) obj;
- if (player.getUniqueID().equals(getUniqueId())) {
- return (player.playerConnection != null) ? player.playerConnection.getPlayer() : null;
- }
- }
-
- return null;
+ // PaperSpigot - Improved player lookup, replace entire method
2014-12-07 20:28:45 +01:00
+ final EntityPlayer playerEntity = server.getHandle().a(getUniqueId());
2014-10-20 00:58:49 +02:00
+ return playerEntity != null ? playerEntity.getBukkitEntity() : null;
+ // PaperSpigot end
}
@Override
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftServer.java b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
2015-06-06 19:15:22 +02:00
index 46bc341..b62a6a2 100644
2014-10-20 00:58:49 +02:00
--- a/src/main/java/org/bukkit/craftbukkit/CraftServer.java
+++ b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
2015-01-29 22:26:20 +01:00
@@ -401,7 +401,12 @@ public final class CraftServer implements Server {
2014-10-20 00:58:49 +02:00
public Player getPlayer(final String name) {
Validate.notNull(name, "Name cannot be null");
- Player found = null;
+ // PaperSpigot start - Improved player lookup changes
+ Player found = getPlayerExact(name);
+ if (found != null) {
+ return found;
+ }
+ // PaperSpigot end
String lowerName = name.toLowerCase();
int delta = Integer.MAX_VALUE;
for (Player player : getOnlinePlayers()) {
2015-01-29 22:26:20 +01:00
@@ -422,15 +427,10 @@ public final class CraftServer implements Server {
2014-10-20 00:58:49 +02:00
public Player getPlayerExact(String name) {
2014-11-28 02:17:45 +01:00
Validate.notNull(name, "Name cannot be null");
2014-10-20 00:58:49 +02:00
- String lname = name.toLowerCase();
-
- for (Player player : getOnlinePlayers()) {
- if (player.getName().equalsIgnoreCase(lname)) {
- return player;
- }
- }
-
- return null;
+ // PaperSpigot start - Improved player lookup, replace whole method
+ EntityPlayer player = playerList.playerMap.get(name);
+ return player != null ? player.getBukkitEntity() : null;
2015-03-08 02:16:09 +01:00
+ // PaperSpigot end
2014-11-28 02:17:45 +01:00
}
@Override
2014-10-20 00:58:49 +02:00
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
2015-05-19 01:41:57 +02:00
index 3c5d90d..ae75b36 100644
2014-10-20 00:58:49 +02:00
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
2015-05-19 01:41:57 +02:00
@@ -105,13 +105,7 @@ public class CraftPlayer extends CraftHumanEntity implements Player {
2014-10-20 00:58:49 +02:00
}
public boolean isOnline() {
- for (Object obj : server.getHandle().players) {
- EntityPlayer player = (EntityPlayer) obj;
- if (player.getName().equalsIgnoreCase(getName())) {
- return true;
- }
- }
- return false;
2014-12-07 20:28:45 +01:00
+ return server.getHandle().a(getUniqueId()) != null; // PaperSpigot - replace whole method
2014-10-20 00:58:49 +02:00
}
public InetSocketAddress getAddress() {
--
2015-06-06 19:15:22 +02:00
2.4.2.windows.1
2014-12-07 20:29:55 +01:00