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:
		
							
								
								
									
										19
									
								
								prepare_davecom.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								prepare_davecom.sh
									
									
									
									
									
										Normal 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 | ||||||
| @@ -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('/'); | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user