View Javadoc
1   /*
2    * Copyright (c) 2001-2017, Zoltan Farkas All Rights Reserved.
3    *
4    * This library is free software; you can redistribute it and/or
5    * modify it under the terms of the GNU Lesser General Public
6    * License as published by the Free Software Foundation; either
7    * version 2.1 of the License, or (at your option) any later version.
8    *
9    * This library is distributed in the hope that it will be useful,
10   * but WITHOUT ANY WARRANTY; without even the implied warranty of
11   * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12   * GNU General Public License for more details.
13   *
14   * You should have received a copy of the GNU Lesser General Public
15   * License along with this program; if not, write to the Free Software
16   * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
17   *
18   * Additionally licensed with:
19   *
20   * Licensed under the Apache License, Version 2.0 (the "License");
21   * you may not use this file except in compliance with the License.
22   * You may obtain a copy of the License at
23   *
24   *      http://www.apache.org/licenses/LICENSE-2.0
25   *
26   * Unless required by applicable law or agreed to in writing, software
27   * distributed under the License is distributed on an "AS IS" BASIS,
28   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
29   * See the License for the specific language governing permissions and
30   * limitations under the License.
31   */
32  package org.spf4j.recyclable.impl;
33  
34  import java.lang.ref.Reference;
35  import java.util.ArrayDeque;
36  import java.util.Deque;
37  import org.spf4j.base.IntMath;
38  import org.spf4j.base.ReferenceType;
39  import org.spf4j.recyclable.SizedRecyclingSupplier;
40  
41  /**
42   * recycling supplier that allows you to recycle objects.
43   * Recycling objects is dangerous business, care should be used.
44   * @author zoly
45   */
46  public final class Powerof2ThreadLocalRecyclingSupplier<T> implements SizedRecyclingSupplier<T> {
47  
48      private final SizedRecyclingSupplier.Factory<T> factory;
49  
50      private final ReferenceType refType;
51  
52      private final ThreadLocal<Deque<Reference<T>> []> localObjects;
53  
54      public Powerof2ThreadLocalRecyclingSupplier(final Factory<T> factory, final ReferenceType refType) {
55          this.factory = factory;
56          this.refType = refType;
57          localObjects = new ThreadLocal<Deque<Reference<T>> []>() {
58  
59              @Override
60              protected  Deque<Reference<T>>[] initialValue() {
61                  Deque<Reference<T>>[] result =  new Deque[28];
62                  for (int i = 0; i < result.length; i++) {
63                      result[i] = new ArrayDeque<>();
64                  }
65                  return result;
66              }
67          };
68      }
69  
70      @Override
71      public T get(final int size) {
72          Deque<Reference<T>>[] available = localObjects.get();
73          int idx = IntMath.closestPowerOf2(size);
74          Deque<Reference<T>> refs = available[idx];
75          if (refs.isEmpty()) {
76              int actualSize = 1 << idx;
77              return factory.create(actualSize);
78          } else {
79              T result;
80              do {
81                  Reference<T> removeLast = refs.removeLast();
82                  result = removeLast.get();
83              } while (result == null && !refs.isEmpty());
84              if (result == null) {
85                  int actualSize = 1 << idx;
86                  return factory.create(actualSize);
87              } else {
88                  return result;
89              }
90          }
91      }
92  
93      @Override
94      public void recycle(final T object) {
95          int size = factory.size(object);
96          int idx = IntMath.closestPowerOf2(size);
97          Deque<Reference<T>>[] available = localObjects.get();
98          Deque<Reference<T>> refs = available[idx];
99          refs.addLast(refType.create(object));
100     }
101 
102     @Override
103     public String toString() {
104         return "Powerof2ThreadLocalRecyclingSupplier{" + "factory=" + factory + ", refType=" + refType + '}';
105     }
106 
107 }