mirror of
https://github.com/ppy/osu.git
synced 2025-01-23 02:22:55 +08:00
Merge pull request #26863 from EVAST9919/scrolling-alloc
Further reduce allocation overhead in `ScrollingHitObjectContainer`
This commit is contained in:
commit
ca36919f10
@ -75,8 +75,10 @@ namespace osu.Game.Rulesets.Osu.Mods
|
|||||||
{
|
{
|
||||||
double time = playfield.Time.Current;
|
double time = playfield.Time.Current;
|
||||||
|
|
||||||
foreach (var drawable in playfield.HitObjectContainer.AliveObjects)
|
foreach (var entry in playfield.HitObjectContainer.AliveEntries)
|
||||||
{
|
{
|
||||||
|
var drawable = entry.Value;
|
||||||
|
|
||||||
switch (drawable)
|
switch (drawable)
|
||||||
{
|
{
|
||||||
case DrawableHitCircle circle:
|
case DrawableHitCircle circle:
|
||||||
|
@ -49,8 +49,10 @@ namespace osu.Game.Rulesets.Osu.Mods
|
|||||||
{
|
{
|
||||||
var cursorPos = playfield.Cursor.AsNonNull().ActiveCursor.DrawPosition;
|
var cursorPos = playfield.Cursor.AsNonNull().ActiveCursor.DrawPosition;
|
||||||
|
|
||||||
foreach (var drawable in playfield.HitObjectContainer.AliveObjects)
|
foreach (var entry in playfield.HitObjectContainer.AliveEntries)
|
||||||
{
|
{
|
||||||
|
var drawable = entry.Value;
|
||||||
|
|
||||||
switch (drawable)
|
switch (drawable)
|
||||||
{
|
{
|
||||||
case DrawableHitCircle circle:
|
case DrawableHitCircle circle:
|
||||||
|
@ -48,8 +48,10 @@ namespace osu.Game.Rulesets.Osu.Mods
|
|||||||
{
|
{
|
||||||
var cursorPos = playfield.Cursor.AsNonNull().ActiveCursor.DrawPosition;
|
var cursorPos = playfield.Cursor.AsNonNull().ActiveCursor.DrawPosition;
|
||||||
|
|
||||||
foreach (var drawable in playfield.HitObjectContainer.AliveObjects)
|
foreach (var entry in playfield.HitObjectContainer.AliveEntries)
|
||||||
{
|
{
|
||||||
|
var drawable = entry.Value;
|
||||||
|
|
||||||
var destination = Vector2.Clamp(2 * drawable.Position - cursorPos, Vector2.Zero, OsuPlayfield.BASE_SIZE);
|
var destination = Vector2.Clamp(2 * drawable.Position - cursorPos, Vector2.Zero, OsuPlayfield.BASE_SIZE);
|
||||||
|
|
||||||
if (drawable.HitObject is Slider thisSlider)
|
if (drawable.HitObject is Slider thisSlider)
|
||||||
|
@ -4,9 +4,11 @@
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using osu.Framework.Extensions.ListExtensions;
|
||||||
using osu.Framework.Graphics;
|
using osu.Framework.Graphics;
|
||||||
using osu.Framework.Graphics.Containers;
|
using osu.Framework.Graphics.Containers;
|
||||||
using osu.Framework.Graphics.Performance;
|
using osu.Framework.Graphics.Performance;
|
||||||
|
using osu.Framework.Lists;
|
||||||
|
|
||||||
namespace osu.Game.Rulesets.Objects.Pooling
|
namespace osu.Game.Rulesets.Objects.Pooling
|
||||||
{
|
{
|
||||||
@ -35,7 +37,7 @@ namespace osu.Game.Rulesets.Objects.Pooling
|
|||||||
/// <remarks>
|
/// <remarks>
|
||||||
/// The enumeration order is undefined.
|
/// The enumeration order is undefined.
|
||||||
/// </remarks>
|
/// </remarks>
|
||||||
public IEnumerable<(TEntry Entry, TDrawable Drawable)> AliveEntries => aliveDrawableMap.Select(x => (x.Key, x.Value));
|
public readonly SlimReadOnlyDictionaryWrapper<TEntry, TDrawable> AliveEntries;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Whether to remove an entry when clock goes backward and crossed its <see cref="LifetimeEntry.LifetimeStart"/>.
|
/// Whether to remove an entry when clock goes backward and crossed its <see cref="LifetimeEntry.LifetimeStart"/>.
|
||||||
@ -63,6 +65,8 @@ namespace osu.Game.Rulesets.Objects.Pooling
|
|||||||
lifetimeManager.EntryBecameAlive += entryBecameAlive;
|
lifetimeManager.EntryBecameAlive += entryBecameAlive;
|
||||||
lifetimeManager.EntryBecameDead += entryBecameDead;
|
lifetimeManager.EntryBecameDead += entryBecameDead;
|
||||||
lifetimeManager.EntryCrossedBoundary += entryCrossedBoundary;
|
lifetimeManager.EntryCrossedBoundary += entryCrossedBoundary;
|
||||||
|
|
||||||
|
AliveEntries = aliveDrawableMap.AsSlimReadOnly();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
@ -105,7 +105,7 @@ namespace osu.Game.Rulesets.UI
|
|||||||
// If required, we can make this lookup more efficient by adding support to get next-future-entry in LifetimeEntryManager.
|
// If required, we can make this lookup more efficient by adding support to get next-future-entry in LifetimeEntryManager.
|
||||||
var candidate =
|
var candidate =
|
||||||
// Use alive entries first as an optimisation.
|
// Use alive entries first as an optimisation.
|
||||||
hitObjectContainer.AliveEntries.Select(tuple => tuple.Entry).Where(e => !isAlreadyHit(e)).MinBy(e => e.HitObject.StartTime)
|
hitObjectContainer.AliveEntries.Keys.Where(e => !isAlreadyHit(e)).MinBy(e => e.HitObject.StartTime)
|
||||||
?? hitObjectContainer.Entries.Where(e => !isAlreadyHit(e)).MinBy(e => e.HitObject.StartTime);
|
?? hitObjectContainer.Entries.Where(e => !isAlreadyHit(e)).MinBy(e => e.HitObject.StartTime);
|
||||||
|
|
||||||
// In the case there are no non-judged objects, the last hit object should be used instead.
|
// In the case there are no non-judged objects, the last hit object should be used instead.
|
||||||
|
@ -21,7 +21,7 @@ namespace osu.Game.Rulesets.UI
|
|||||||
{
|
{
|
||||||
public IEnumerable<DrawableHitObject> Objects => InternalChildren.Cast<DrawableHitObject>().OrderBy(h => h.HitObject.StartTime);
|
public IEnumerable<DrawableHitObject> Objects => InternalChildren.Cast<DrawableHitObject>().OrderBy(h => h.HitObject.StartTime);
|
||||||
|
|
||||||
public IEnumerable<DrawableHitObject> AliveObjects => AliveEntries.Select(pair => pair.Drawable).OrderBy(h => h.HitObject.StartTime);
|
public IEnumerable<DrawableHitObject> AliveObjects => AliveEntries.Values.OrderBy(h => h.HitObject.StartTime);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Invoked when a <see cref="DrawableHitObject"/> is judged.
|
/// Invoked when a <see cref="DrawableHitObject"/> is judged.
|
||||||
|
@ -188,7 +188,7 @@ namespace osu.Game.Rulesets.UI.Scrolling
|
|||||||
// We are not using AliveObjects directly to avoid selection/sorting overhead since we don't care about the order at which positions will be updated.
|
// We are not using AliveObjects directly to avoid selection/sorting overhead since we don't care about the order at which positions will be updated.
|
||||||
foreach (var entry in AliveEntries)
|
foreach (var entry in AliveEntries)
|
||||||
{
|
{
|
||||||
var obj = entry.Drawable;
|
var obj = entry.Value;
|
||||||
|
|
||||||
updatePosition(obj, Time.Current);
|
updatePosition(obj, Time.Current);
|
||||||
|
|
||||||
|
@ -198,8 +198,10 @@ namespace osu.Game.Screens.Play
|
|||||||
foreach (var nested in playfield.NestedPlayfields)
|
foreach (var nested in playfield.NestedPlayfields)
|
||||||
applyToPlayfield(nested);
|
applyToPlayfield(nested);
|
||||||
|
|
||||||
foreach (DrawableHitObject obj in playfield.HitObjectContainer.AliveObjects)
|
foreach (var entry in playfield.HitObjectContainer.AliveEntries)
|
||||||
{
|
{
|
||||||
|
var obj = entry.Value;
|
||||||
|
|
||||||
if (appliedObjects.Contains(obj))
|
if (appliedObjects.Contains(obj))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user