-
Notifications
You must be signed in to change notification settings - Fork 29
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Jan Monterrubio
authored and
Jan Monterrubio
committed
Feb 18, 2021
1 parent
72b878c
commit d063092
Showing
19 changed files
with
229 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,217 @@ | ||
--- | ||
title: "Memory Challenge" | ||
weight: 15 | ||
description: Put all the pieces together with this challenge to test your skills | ||
--- | ||
|
||
The provided WorkshopService has an intentional memory problem, use the previous sections as reference to try to pin point it. | ||
|
||
## Tips | ||
|
||
* With the Dominator Tree (Eclipse MAT) determine what object would free up the most memory (if we could get rid of it) | ||
* With the Histogram (Eclipse MAT) determine what type of object has the most instances | ||
* Using the Incoming Objects view, find out what's pointing to them. | ||
|
||
Once you've found the large object, look through it's properties and determine why this object holds on to so many other objects. | ||
|
||
{{% alert title="Solution" color="warning" %}} | ||
This section contains a solution, make sure you've given things a shot before spoiling the fun. | ||
{{% /alert %}} | ||
|
||
{{% spoiler title="Show Solution" %}} | ||
## Dominator Tree | ||
|
||
A first glance at the dominator tree will point us to a `io.dropwizard.jetty.MutableServletContextHandler`, which is something controlled by the DropWizard framework. Let's dig through its references until we find some classes in the `cchesser` package. | ||
|
||
data:image/s3,"s3://crabby-images/2d553/2d55323de4f0fd892e5641ac5c8213bd23f66e65" alt="" | ||
|
||
The first instance of a class we "own" is `cchesser.javaperf.workshop.resources.WorkshopResource` | ||
|
||
data:image/s3,"s3://crabby-images/1097b/1097b739383722b11b15c6b4bec271d59de67618" alt="" | ||
|
||
Let's look at it's properties a bit, and we'll find a `cchesser.javaperf.workshop.cache.CleverCache` reference with about 70% of the memory consumption. | ||
|
||
data:image/s3,"s3://crabby-images/2e863/2e8638504277e837ff178a5657bdd719c5e6b309" alt="" | ||
|
||
We'll look at its properties in a bit more detail in the following sections. | ||
|
||
## Histogram View | ||
|
||
A first glance at the Histogram View (sorted by retained heap), shows us about 2,000,000 instances of `cchesser.javaperf.workshop.data.Searcher$SearchResultElement` | ||
|
||
data:image/s3,"s3://crabby-images/12d61/12d6120a5dcf91dc62709c22e0235807c8f4f853" alt="" | ||
|
||
Looking through the references to an instance of this type, we end up being referenced by the `data` field in `cchesser.javaperf.workshop.cache.CleverCache$CacheEntry` | ||
|
||
data:image/s3,"s3://crabby-images/c400d/c400dab050b5844d43f5f77d6ee1fa56f23f64bf" alt="" | ||
|
||
We can further inspect the Incoming References for a particular entry, and see what is pointing to it, tracing it back to something we own: | ||
|
||
data:image/s3,"s3://crabby-images/8dca2/8dca2ab72e114dfb95c145eb4b44561c6670fb11" alt="" | ||
|
||
Once again, we've ended up at `cchesser.javaperf.workshop.resources.WorkshopResource`, specifically a field called `resultsByContext`. | ||
|
||
We'll take a closer look at this `cchesser.javaperf.workshop.cache.CleverCache` object. | ||
|
||
## Inspecting Clever Cache | ||
|
||
First, lets determine how many instances of `cchesser.javaperf.workshop.cache.CleverCache` we have. We've found a single instance that is about 70Mb (in our sample dump), so let's double check if there's more of these. | ||
|
||
We can do that in a couple of ways: | ||
|
||
1. Use the histogram view to filter on `CleverCache` | ||
|
||
data:image/s3,"s3://crabby-images/22acb/22acb0f6a058bb752c60513b8358e89a4be46da2" alt="" | ||
|
||
2. Use OQL to find instances of that object: | ||
|
||
```sql | ||
select * from cchesser.javaperf.workshop.cache.CleverCache | ||
``` | ||
|
||
data:image/s3,"s3://crabby-images/afa94/afa94903bc1b8d295b50762f3095f8c4e28590ed" alt="" | ||
|
||
### Fields/References | ||
|
||
We can find the fields and references for an instance using either Eclipse MAT or VisualVM. | ||
|
||
In Eclipse Mat, the fields for an instance display in an __Attributes__ tab | ||
|
||
data:image/s3,"s3://crabby-images/61f83/61f83fe3c8f0100093224add6d872386d62e55d0" alt="" | ||
|
||
|
||
In VisualVM, we can load the __Objects__ view and apply a filter on class name to then inspect the fields and references | ||
|
||
data:image/s3,"s3://crabby-images/09268/09268ce7402454b534753373287dffed2d076d0c" alt="" | ||
|
||
|
||
Looking at the properties, we find that there's a `cacheLimit` of `250`. Let's find out exactly how many entries are in the cache (the `innerCache` field). | ||
|
||
## Finding Sizes | ||
|
||
Let's write a query that will show us what the count of entries in the cache is. | ||
|
||
* For our query, we need to pull the `cacheLimit` and `size` of the `innerCache` map, | ||
|
||
We can do this a few ways: | ||
|
||
### Eclipse MAT / OQL | ||
|
||
we can do that with the following query: | ||
```sql | ||
SELECT c.cacheLimit AS "max entries", c.innerCache.size AS entries FROM cchesser.javaperf.workshop.cache.CleverCache c | ||
``` | ||
|
||
data:image/s3,"s3://crabby-images/ddc5f/ddc5fe8a2753ad36f56debf87af7ec73cfbeb348" alt="" | ||
|
||
|
||
### Eclipse MAT / Calcite | ||
```sql | ||
select c.cacheLimit as "max entries", getSize(c.innerCache) as "entries" | ||
from cchesser.javaperf.workshop.cache.CleverCache c | ||
``` | ||
|
||
data:image/s3,"s3://crabby-images/1cdae/1cdae611174a386f9ef7860013377c0d0a9b3f40" alt="" | ||
|
||
|
||
### VisualVM / OQL | ||
|
||
```js | ||
var caches = heap.objects("cchesser.javaperf.workshop.cache.CleverCache") | ||
|
||
function report(cache) { | ||
return { maxEntries: cache.cacheLimit, entries: cache.innerCache.size } | ||
} | ||
|
||
map(caches, report) | ||
``` | ||
|
||
data:image/s3,"s3://crabby-images/78fa5/78fa55b29f6ac97aa5307233a6d1710b7f4bc5a4" alt="" | ||
|
||
{{% pageinfo color="warning" %}} | ||
That doesn't seem right at all, we want at most 250 items but we have about 10,000. Let's find out why that is the case. | ||
{{% /pageinfo %}} | ||
|
||
|
||
## Memory Problem Explained | ||
|
||
The interaction between the `WorkshopResource` and `CleverCache` happens through the `/search` resource. | ||
|
||
_cchesser.javaperf.workshop.WorkshopResource_ | ||
```java | ||
@GET | ||
@Path("/search") | ||
@Produces(MediaType.APPLICATION_JSON) | ||
@Timed | ||
public Searcher.SearchResult searchConference(@QueryParam("q") String term, @QueryParam("c") String context) { | ||
return fetchResults(term, context); | ||
} | ||
|
||
private Searcher.SearchResult fetchResults(String term, String context) { | ||
|
||
if (context != null && !context.isEmpty()) { | ||
if (resultsByContext.exists(context)) { | ||
return resultsByContext.fetch(context); | ||
} | ||
} | ||
|
||
// does not exist in cache, compute and store | ||
|
||
SearchResult results = searcher.search(term); | ||
resultsByContext.store(results.getResultsContext(), results); | ||
return results; | ||
} | ||
``` | ||
|
||
If a context is sent, the previously cached values for the search are returned, otherwise we cache those result values in case they need to be referenced again. | ||
|
||
_javaperf.workshop.cache.CleverCache_ | ||
```java | ||
/** | ||
* Stores the given key value pair in the cache | ||
* | ||
* @param key the unique identifier associated with the given value | ||
* @param value the value mapped to the given key | ||
*/ | ||
public void store(K key, V value) { | ||
if (isFull()) { | ||
K keyToRemove = getLFUKey(); | ||
CleverCache<K, V>.CacheEntry<V> removedEntry = innerCache.remove(keyToRemove); | ||
} | ||
|
||
CacheEntry<V> newEntry = new CacheEntry<V>(); | ||
newEntry.data = value; | ||
newEntry.frequency = 0; | ||
|
||
innerCache.put(key, newEntry); | ||
} | ||
// other methods | ||
private boolean isFull() { | ||
return innerCache.size() == cacheLimit; | ||
} | ||
``` | ||
|
||
This interaction is **NOT** thread safe. Consider this scenario: | ||
|
||
* Thread 1 invokes `store` when the `innerCache.size()` is `249` | ||
* `isFull()` evaluates to `false` | ||
* Execution halts prior to `innerCache.put` | ||
* Thread 2 invokes `store` when the `innerCache.size()` is `249` | ||
* `isFull()` evaluates to false | ||
* Execution halts prior to `innerCache.put` | ||
* Thread 1 resumes and executes `innerCache.put`, `innerCache.size()` is now `250` | ||
* Thread 2 resumes and executes `innerCache.put`, `innerCache.size()` is now `251` | ||
* For the lifecycle of the jvm `isFull()` will no longer return `true` | ||
|
||
Fortunately, the DropWizard documentation calls out the need to consider thread safety: | ||
|
||
data:image/s3,"s3://crabby-images/cd528/cd52888d80319e52f9da78b4728a5eacbc4df240" alt="" | ||
|
||
Since the `WorkshopResource` has a `resultsByContext` field that will get shared across threads, that type should be thread safe. | ||
|
||
### Potential Solutions | ||
|
||
* The naive solution would be to just change `isFull()` to consider `>=`, though that might lead to ConcurrentModificationExceptions | ||
* The proper solution would be to make the class Thread Safe | ||
|
||
{{% /spoiler %}} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.