1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17 package org.apache.commons.collections4.trie.analyzer;
18
19 import org.apache.commons.collections4.trie.KeyAnalyzer;
20
21
22
23
24
25
26
27
28 public class StringKeyAnalyzer extends KeyAnalyzer<String> {
29
30 private static final long serialVersionUID = -7032449491269434877L;
31
32
33 public static final StringKeyAnalyzer INSTANCE = new StringKeyAnalyzer();
34
35
36 public static final int LENGTH = Character.SIZE;
37
38
39 private static final int MSB = 0x8000;
40
41
42 private static int mask(final int bit) {
43 return MSB >>> bit;
44 }
45
46
47
48
49
50
51 @Deprecated
52 public StringKeyAnalyzer() {
53
54 }
55
56 @Override
57 public int bitIndex(final String key, final int offsetInBits, final int lengthInBits,
58 final String other, final int otherOffsetInBits, final int otherLengthInBits) {
59
60 boolean allNull = true;
61
62 if (offsetInBits % LENGTH != 0 || otherOffsetInBits % LENGTH != 0
63 || lengthInBits % LENGTH != 0 || otherLengthInBits % LENGTH != 0) {
64 throw new IllegalArgumentException("The offsets and lengths must be at Character boundaries");
65 }
66
67 final int beginIndex1 = offsetInBits / LENGTH;
68 final int beginIndex2 = otherOffsetInBits / LENGTH;
69
70 final int endIndex1 = beginIndex1 + lengthInBits / LENGTH;
71 final int endIndex2 = beginIndex2 + otherLengthInBits / LENGTH;
72
73 final int length = Math.max(endIndex1, endIndex2);
74
75
76
77
78 char k = 0, f = 0;
79 for (int i = 0; i < length; i++) {
80 final int index1 = beginIndex1 + i;
81 final int index2 = beginIndex2 + i;
82
83 if (index1 >= endIndex1) {
84 k = 0;
85 } else {
86 k = key.charAt(index1);
87 }
88
89 if (other == null || index2 >= endIndex2) {
90 f = 0;
91 } else {
92 f = other.charAt(index2);
93 }
94
95 if (k != f) {
96 final int x = k ^ f;
97 return i * LENGTH + Integer.numberOfLeadingZeros(x) - LENGTH;
98 }
99
100 if (k != 0) {
101 allNull = false;
102 }
103 }
104
105
106 if (allNull) {
107 return NULL_BIT_KEY;
108 }
109
110
111 return EQUAL_BIT_KEY;
112 }
113
114 @Override
115 public int bitsPerElement() {
116 return LENGTH;
117 }
118
119 @Override
120 public boolean isBitSet(final String key, final int bitIndex, final int lengthInBits) {
121 if (key == null || bitIndex >= lengthInBits) {
122 return false;
123 }
124
125 final int index = bitIndex / LENGTH;
126 final int bit = bitIndex % LENGTH;
127
128 return (key.charAt(index) & mask(bit)) != 0;
129 }
130
131 @Override
132 public boolean isPrefix(final String prefix, final int offsetInBits,
133 final int lengthInBits, final String key) {
134 if (offsetInBits % LENGTH != 0 || lengthInBits % LENGTH != 0) {
135 throw new IllegalArgumentException(
136 "Cannot determine prefix outside of Character boundaries");
137 }
138
139 final String s1 = prefix.substring(offsetInBits / LENGTH, lengthInBits / LENGTH);
140 return key.startsWith(s1);
141 }
142
143 @Override
144 public int lengthInBits(final String key) {
145 return key != null ? key.length() * LENGTH : 0;
146 }
147 }