Update davecom entry use ByteBuffer instead of String for Names and graal (#245)

* my implementation about 19.5 seconds on M1 MacBook Air

* final version

* use ByteBuffer instead of String; use graal

* add prepare_davecom script
This commit is contained in:
David Kopec 2024-01-11 04:24:25 -05:00 committed by GitHub
parent 999a4e9e10
commit 780ab9af3e
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 47 additions and 20 deletions

19
prepare_davecom.sh Normal file
View File

@ -0,0 +1,19 @@
#!/bin/bash
#
# Copyright 2023 The original authors
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
source "$HOME/.sdkman/bin/sdkman-init.sh"
sdk use java 21.0.1-graal 1>&2

View File

@ -25,9 +25,11 @@ import java.text.DecimalFormat;
import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentHashMap;
import java.util.HashMap; import java.util.HashMap;
import java.util.IntSummaryStatistics; import java.util.IntSummaryStatistics;
import java.util.List;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.concurrent.ExecutorService; import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; import java.util.concurrent.Executors;
import java.util.stream.Collectors;
public class CalculateAverage_davecom { public class CalculateAverage_davecom {
@ -41,19 +43,19 @@ public class CalculateAverage_davecom {
private static final String FILE = "./measurements.txt"; private static final String FILE = "./measurements.txt";
private static final ConcurrentHashMap<String, Integer> mins = new ConcurrentHashMap<>(); private static final ConcurrentHashMap<ByteBuffer, Integer> mins = new ConcurrentHashMap<>();
private static final ConcurrentHashMap<String, Integer> maxs = new ConcurrentHashMap<>(); private static final ConcurrentHashMap<ByteBuffer, Integer> maxs = new ConcurrentHashMap<>();
private static final ConcurrentHashMap<String, Integer> sums = new ConcurrentHashMap<>(); private static final ConcurrentHashMap<ByteBuffer, Integer> sums = new ConcurrentHashMap<>();
private static final ConcurrentHashMap<String, Integer> counts = new ConcurrentHashMap<>(); private static final ConcurrentHashMap<ByteBuffer, Integer> counts = new ConcurrentHashMap<>();
public static void processChunk(MappedByteBuffer chunk, long chunkSize) { public static void processChunk(MappedByteBuffer chunk, long chunkSize) {
// setup // setup
chunk.load(); chunk.load();
HashMap<String, IntSummaryStatistics> values = new HashMap<>(); HashMap<ByteBuffer, IntSummaryStatistics> values = new HashMap<>();
// do the actual processing // do the actual processing
long end = chunk.position() + chunkSize; long end = chunk.position() + chunkSize;
byte[] name = new byte[128]; // byte[] name = new byte[128];
int value = 0; int value = 0;
byte b = 0; byte b = 0;
boolean negate = false; boolean negate = false;
@ -70,9 +72,9 @@ public class CalculateAverage_davecom {
nameEnd = chunk.position() - 1; nameEnd = chunk.position() - 1;
nameLength = (int) (nameEnd - nameStart); nameLength = (int) (nameEnd - nameStart);
// generate byte array for name // generate byte array for name
chunk.get(chunk.position() - nameLength - 1, name, 0, nameLength); ByteBuffer nameBuffer = ByteBuffer.allocate(nameLength);
chunk.get(chunk.position() - nameLength - 1, nameBuffer.array(), 0, nameLength);
// convert name to string // convert name to string
String nameString = new String(name, 0, nameLength);
// read value // read value
value = 0; value = 0;
b = chunk.get(); b = chunk.get();
@ -94,22 +96,22 @@ public class CalculateAverage_davecom {
value = -value; value = -value;
} }
if (values.containsKey(nameString)) { if (values.containsKey(nameBuffer)) {
values.get(nameString).accept(value); values.get(nameBuffer).accept(value);
} }
else { else {
IntSummaryStatistics stats = new IntSummaryStatistics(); IntSummaryStatistics stats = new IntSummaryStatistics();
stats.accept(value); stats.accept(value);
values.put(nameString, stats); values.put(nameBuffer, stats);
} }
} }
for (String nameStr : values.keySet()) { for (ByteBuffer nameBfr : values.keySet()) {
IntSummaryStatistics stats = values.get(nameStr); IntSummaryStatistics stats = values.get(nameBfr);
mins.compute(nameStr, (k, v) -> v == null ? stats.getMin() : Math.min(v, stats.getMin())); mins.compute(nameBfr, (k, v) -> v == null ? stats.getMin() : Math.min(v, stats.getMin()));
maxs.compute(nameStr, (k, v) -> v == null ? stats.getMax() : Math.max(v, stats.getMax())); maxs.compute(nameBfr, (k, v) -> v == null ? stats.getMax() : Math.max(v, stats.getMax()));
sums.compute(nameStr, (k, v) -> v == null ? (int) stats.getSum() : (v + (int) stats.getSum())); sums.compute(nameBfr, (k, v) -> v == null ? (int) stats.getSum() : (v + (int) stats.getSum()));
counts.compute(nameStr, (k, v) -> v == null ? (int) stats.getCount() : (v + (int) stats.getCount())); counts.compute(nameBfr, (k, v) -> v == null ? (int) stats.getCount() : (v + (int) stats.getCount()));
} }
} }
@ -118,14 +120,20 @@ public class CalculateAverage_davecom {
// fast string concatenation starting with { and ending with } with a comma between each (no newlines) // fast string concatenation starting with { and ending with } with a comma between each (no newlines)
StringBuilder sb = new StringBuilder(); StringBuilder sb = new StringBuilder();
sb.append('{'); sb.append('{');
var sortedNames = mins.keySet().stream().sorted().toArray(String[]::new); // var sortedNames = mins.keySet().stream().sorted().toArray(String[]::new);
DecimalFormat df = new DecimalFormat("0.0"); DecimalFormat df = new DecimalFormat("0.0");
df.setRoundingMode(RoundingMode.HALF_UP); df.setRoundingMode(RoundingMode.HALF_UP);
for (String name : sortedNames) { List<String> sortedNames = mins.keySet().stream()
.map(b -> new String(b.array(), 0, b.limit()))
.sorted()
.collect(Collectors.toList());
for (String nameStr : sortedNames) {
ByteBuffer name = ByteBuffer.wrap(nameStr.getBytes());
double min = ((double) mins.get(name)) / 10; double min = ((double) mins.get(name)) / 10;
double max = ((double) maxs.get(name)) / 10; double max = ((double) maxs.get(name)) / 10;
double average = ((double) sums.get(name)) / ((double) counts.get(name)) / 10; double average = ((double) sums.get(name)) / ((double) counts.get(name)) / 10;
sb.append(name); sb.append(nameStr);
sb.append('='); sb.append('=');
sb.append(df.format(min)); sb.append(df.format(min));
sb.append('/'); sb.append('/');