Projektteil 2
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

EpizentrumRechnung.java 11KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. package de.edotzlaff.detection.detektion.berechnung;
  2. import android.location.Location;
  3. import android.nfc.Tag;
  4. import android.os.Build;
  5. import android.util.Log;
  6. import androidx.annotation.RequiresApi;
  7. import com.google.common.collect.ImmutableSet;
  8. import com.google.common.collect.Sets;
  9. import com.lemmingapex.trilateration.NonLinearLeastSquaresSolver;
  10. import com.lemmingapex.trilateration.TrilaterationFunction;
  11. import org.apache.commons.math3.fitting.leastsquares.LeastSquaresOptimizer;
  12. import org.apache.commons.math3.fitting.leastsquares.LevenbergMarquardtOptimizer;
  13. import de.edotzlaff.detection.detektion.Detektionssignal;
  14. import de.edotzlaff.detection.detektion.Epizentrum;
  15. import de.edotzlaff.detection.detektion.berechnung.mathObjekte.Koordinaten;
  16. import de.edotzlaff.detection.detektion.berechnung.mathObjekte.Kreis;
  17. import de.edotzlaff.detection.detektion.berechnung.mathObjekte.Schnittpunkt;
  18. import de.edotzlaff.detection.detektion.berechnung.mathObjekte.UTM2Deg;
  19. import java.math.BigDecimal;
  20. import java.math.RoundingMode;
  21. import java.time.LocalDateTime;
  22. import java.util.ArrayList;
  23. import java.util.Set;
  24. public class EpizentrumRechnung {
  25. //ToDo Schrittweite und Abstand auf Szenario anpassen
  26. private static final double SCHRITTWEITE = 5;
  27. private static final double MAX_ABSTAND = 5;
  28. private static final int DOUBLESTELLEN = 15;
  29. private static final int SCHLEIFENDURCHLAUFTE = 5000;
  30. private static final String TAG = "EPIZENTRUMRECHNUNG";
  31. @RequiresApi(api = Build.VERSION_CODES.O)
  32. public ArrayList<Endpunkt> initEndpunkte(ArrayList<Detektionssignal> detektionssignal, Detektionssignal ersteMessung) {
  33. ArrayList<Endpunkt> arrBerechnungsObjekte = new ArrayList<>();
  34. for (Detektionssignal aufzeichnung : detektionssignal) {
  35. int zeitunterschied = aufzeichnung.getAnkunftsZeit().getHour() * 3600 + aufzeichnung.getAnkunftsZeit().getMinute() * 60 + aufzeichnung.getAnkunftsZeit().getSecond()
  36. - ersteMessung.getAnkunftsZeit().getHour() * 3600 - ersteMessung.getAnkunftsZeit().getMinute() * 60 - ersteMessung.getAnkunftsZeit().getSecond();
  37. double startradius = zeitunterschied * aufzeichnung.getGeschwErdbeben();
  38. Log.i(TAG, "Init Daten: " + aufzeichnung.getDeviceId() + "Startradius: " + startradius);
  39. Endpunkt tempEndpunkt = new Endpunkt(aufzeichnung.getDeviceId());
  40. Kreis k = new Kreis(aufzeichnung.getKoordinaten().getX(), aufzeichnung.getKoordinaten().getY(), startradius);
  41. tempEndpunkt.setKreis(k);
  42. arrBerechnungsObjekte.add(tempEndpunkt);
  43. }
  44. Log.i(TAG, "Daten Erfolgreich initialisiert");
  45. return arrBerechnungsObjekte;
  46. }
  47. @RequiresApi(api = Build.VERSION_CODES.O)
  48. public Detektionssignal ermittleErstesDetektionssignal(ArrayList<Detektionssignal> detektionssignale)
  49. {
  50. Detektionssignal erstesDetektionssignal = detektionssignale.get(0);
  51. for (int i = 1; i < detektionssignale.size(); i++) {
  52. if (detektionssignale.get(i).getAnkunftsZeit().isBefore(erstesDetektionssignal.getAnkunftsZeit())) {
  53. erstesDetektionssignal = detektionssignale.get(i);
  54. }
  55. }
  56. Log.i(TAG, "Test");
  57. return erstesDetektionssignal;
  58. }
  59. public ArrayList<Schnittpunkt> berechneKreisSchnittpunkte(Kreis kreis1, Kreis kreis2) {
  60. ArrayList<Schnittpunkt> schnittpunkte = new ArrayList<>();
  61. double dx = kreis2.getMittelPunkt().getX() - kreis1.getMittelPunkt().getX();
  62. double dy = kreis2.getMittelPunkt().getY() - kreis1.getMittelPunkt().getY();
  63. double c = Math.sqrt(Math.pow(dx, 2) + Math.pow(dy, 2));
  64. double x = ((Math.pow(kreis1.getRadius(), 2) + Math.pow(c, 2) - Math.pow(kreis2.getRadius(), 2))) / (2 * c);
  65. double y = Math.pow(kreis1.getRadius(), 2) - Math.pow(x, 2);
  66. //Kein Schnittpunkt vorhanden
  67. if (y < 0) {
  68. return schnittpunkte;
  69. }
  70. if (y > 0) {
  71. y = Math.sqrt(y);
  72. }
  73. double ex0 = dx / c;
  74. double ex1 = dy / c;
  75. double ey0 = -ex1;
  76. Schnittpunkt s1 = new Schnittpunkt(kreis1.getMittelPunkt().getX() + x * ex0,
  77. kreis1.getMittelPunkt().getY() + x * ex1);
  78. //Genau ein Schnittpunkt vorhanden
  79. if (y == 0) {
  80. schnittpunkte.add(s1);
  81. return schnittpunkte;
  82. }
  83. Schnittpunkt s2 = new Schnittpunkt(s1.getKoordinaten().getX() - y * ey0, s1.getKoordinaten().getY() - y * ex0);
  84. s1.getKoordinaten().setLocation(s1.getKoordinaten().getX() + y * ey0, s1.getKoordinaten().getY() + y * ex0);
  85. schnittpunkte.add(s1);
  86. schnittpunkte.add(s2);
  87. return schnittpunkte;
  88. }
  89. public boolean genuegendSchnittpunktevorhanden(ArrayList<EndpunktVerbindung> endpunktVerbindungen) {
  90. //Log.i(TAG,"Überprüfe ob genügend Schnittpunkte vorhanden sind");
  91. //Log.i(TAG,"Radius: "+ Double.toString(endpunktVerbindungen.get(0).getEndpunkt1().getKreis().getRadius()));
  92. for (EndpunktVerbindung endpunktVerbindung : endpunktVerbindungen) {
  93. ArrayList<Schnittpunkt> templist;
  94. templist = berechneKreisSchnittpunkte(endpunktVerbindung.getEndpunkt1().getKreis(), endpunktVerbindung.getEndpunkt2().getKreis());
  95. if (templist.isEmpty()) {
  96. return false;
  97. }
  98. endpunktVerbindung.setSchnittpunkte(templist);
  99. }
  100. return true;
  101. }
  102. public Epizentrum ueberpruefeObEpizentrum(ArrayList<EndpunktVerbindung> endpunktVerbindungen) {
  103. int counter;
  104. ArrayList<Schnittpunkt> schnittpunkte = new ArrayList<>();
  105. for (Schnittpunkt s1 : endpunktVerbindungen.get(0).getSchnittpunkte()) {
  106. counter = 0;
  107. for (int i = 1; i < endpunktVerbindungen.size(); i++) {
  108. boolean bereitsSchnittpunktErmittelt = false;
  109. for (Schnittpunkt s2 : endpunktVerbindungen.get(i).getSchnittpunkte()) {
  110. double abstand = Math.sqrt(
  111. Math.pow(s2.getKoordinaten().getX() - s1.getKoordinaten().getX(), 2) +
  112. Math.pow(s2.getKoordinaten().getY() - s1.getKoordinaten().getY(), 2));
  113. if (abstand < MAX_ABSTAND && !bereitsSchnittpunktErmittelt) {
  114. bereitsSchnittpunktErmittelt = true;
  115. counter++;
  116. schnittpunkte.add(s2);
  117. }
  118. if (counter == endpunktVerbindungen.size() - 1) {
  119. schnittpunkte.add(s1);
  120. Epizentrum tempEpizentrum = new Epizentrum();
  121. Koordinaten koordinaten = (berechneEpizentrumKoordinaten(schnittpunkte));
  122. tempEpizentrum.setKoordinaten(koordinaten);
  123. Log.i(TAG, "Epizentrumkoord: " + tempEpizentrum.getKoordinaten().getX() + " " + tempEpizentrum.getKoordinaten().getY());
  124. return tempEpizentrum;
  125. }
  126. }
  127. }
  128. }
  129. return null;
  130. }
  131. public Koordinaten berechneEpizentrumKoordinaten(ArrayList<Schnittpunkt> schnittpunkte) {
  132. Log.i(TAG, "Berechne Epizentrum Koordinaten");
  133. double x = 0;
  134. double y = 0;
  135. for (Schnittpunkt s : schnittpunkte) {
  136. Log.i(TAG, "Koordinaten der Schnittpunkte: "+ s.getKoordinaten().getX() + " " + s.getKoordinaten().getY());
  137. x+= s.getKoordinaten().getX();
  138. y+= s.getKoordinaten().getY();
  139. }
  140. x = new BigDecimal(x/schnittpunkte.size()).setScale(DOUBLESTELLEN, RoundingMode.HALF_UP).doubleValue();
  141. y = new BigDecimal(y/schnittpunkte.size()).setScale(DOUBLESTELLEN, RoundingMode.HALF_UP).doubleValue();
  142. return new Koordinaten(x,y);
  143. }
  144. public ArrayList<EndpunktVerbindung> ermittleEndpunktKonbinationen(ArrayList<Endpunkt> tempBerechungsObjekte) {
  145. ArrayList<EndpunktVerbindung> verbindungen = new ArrayList<>();
  146. Set<Set<Endpunkt>> combinations = Sets.combinations(ImmutableSet.copyOf(tempBerechungsObjekte), 2);
  147. for (Set<Endpunkt> tempEndpunkt : combinations) {
  148. ArrayList<Endpunkt> helpArrayToGetValues = new ArrayList<>(tempEndpunkt);
  149. EndpunktVerbindung endpunktVerbindung = new EndpunktVerbindung(helpArrayToGetValues.get(0), helpArrayToGetValues.get(1));
  150. verbindungen.add(endpunktVerbindung);
  151. }
  152. Log.i(TAG, "Endpunkteverbindungen Erfolgreich festgelegt, Anzahl: " + verbindungen.size());
  153. return verbindungen;
  154. }
  155. @RequiresApi(api = Build.VERSION_CODES.O)
  156. public LocalDateTime berechneEntstehungsZeitpunkt(Detektionssignal detektionssignal, Koordinaten koordEpi) {
  157. double abstand = Math.sqrt(
  158. Math.pow(detektionssignal.getKoordinaten().getX() - koordEpi.getX(), 2) +
  159. Math.pow(detektionssignal.getKoordinaten().getY() - koordEpi.getY(), 2));
  160. int sec = (int) (abstand / detektionssignal.geschwErdbeben);
  161. return detektionssignal.getAnkunftsZeit().minusSeconds(sec);
  162. }
  163. @RequiresApi(api = Build.VERSION_CODES.O)
  164. public LocalDateTime berechneEntstehungsZeitpunktLongLatt(Detektionssignal detektionssignal, Koordinaten koordEpi) {
  165. float[] results = new float[1];
  166. Location.distanceBetween(detektionssignal.getKoordinaten().getX(), detektionssignal.getKoordinaten().getY(), koordEpi.getX(), koordEpi.getY(), results);
  167. float distanceInMeters = results[0];
  168. int sec = (int) (distanceInMeters / detektionssignal.geschwErdbeben);
  169. return detektionssignal.getAnkunftsZeit().minusSeconds(sec);
  170. }
  171. public void erhoheRadius(ArrayList<Endpunkt> tempEndpunktListe) {
  172. for (Endpunkt b : tempEndpunktListe) {
  173. b.erhoeheRadius(SCHRITTWEITE);
  174. }
  175. }
  176. public Epizentrum trilateration(ArrayList<Endpunkt> tempEndpunktListe) {
  177. double[][] positions = new double[tempEndpunktListe.size()][2];
  178. double[] distances = new double[tempEndpunktListe.size()];
  179. for (int i = 0; i < tempEndpunktListe.size(); i++) {
  180. positions[i][0] = tempEndpunktListe.get(i).getKreis().getMittelPunkt().getX();
  181. positions[i][1] = tempEndpunktListe.get(i).getKreis().getMittelPunkt().getY();
  182. distances[i] = tempEndpunktListe.get(i).getKreis().getRadius();
  183. }
  184. NonLinearLeastSquaresSolver solver = new NonLinearLeastSquaresSolver(new TrilaterationFunction(positions, distances), new LevenbergMarquardtOptimizer());
  185. LeastSquaresOptimizer.Optimum optimum = solver.solve();
  186. double[] centroid = optimum.getPoint().toArray();
  187. if (centroid.length == 2) {
  188. UTM2Deg utm2Deg = new UTM2Deg();
  189. utm2Deg.convertToDeg(32, 'U', centroid[0], centroid[1]);
  190. Log.i(TAG, "Center: " + utm2Deg.getLatitude() + " " + utm2Deg.getLongitude());
  191. Epizentrum epizentrum = new Epizentrum();
  192. Koordinaten koordinaten = new Koordinaten(centroid[0], centroid[1]);
  193. epizentrum.setKoordinaten(koordinaten);
  194. return epizentrum;
  195. }
  196. return null;
  197. }
  198. }