Mercurial > dive4elements > river
changeset 4521:71e6fe870c1d
Added objects and factory for sediment densities.
author | Raimund Renkert <rrenkert@intevation.de> |
---|---|
date | Wed, 14 Nov 2012 17:17:02 +0100 (2012-11-14) |
parents | 0b330ed218f6 |
children | a9753f717b3d |
files | flys-artifacts/src/main/java/de/intevation/flys/artifacts/model/minfo/SedimentDensity.java flys-artifacts/src/main/java/de/intevation/flys/artifacts/model/minfo/SedimentDensityFactory.java flys-artifacts/src/main/java/de/intevation/flys/artifacts/model/minfo/SedimentDensityValue.java |
diffstat | 3 files changed, 296 insertions(+), 0 deletions(-) [+] |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/flys-artifacts/src/main/java/de/intevation/flys/artifacts/model/minfo/SedimentDensity.java Wed Nov 14 17:17:02 2012 +0100 @@ -0,0 +1,160 @@ +package de.intevation.flys.artifacts.model.minfo; + +import java.util.ArrayList; +import java.util.Collection; +import java.util.Collections; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Set; + +import org.apache.log4j.Logger; +import org.jfree.util.Log; + + +public class SedimentDensity +{ + private static final Logger logger = Logger + .getLogger(SedimentDensity.class); + + private Map<Integer, List<SedimentDensityValue>> densities; + private List<Integer> years; + + public SedimentDensity() { + this.densities = new HashMap<Integer, List<SedimentDensityValue>>(); + this.years = new ArrayList<Integer>(); + } + + public Map<Integer, List<SedimentDensityValue>> getDensities() { + return densities; + } + + public void setDensities(Map<Integer, List<SedimentDensityValue>> densities) { + this.densities = densities; + } + + public void addDensity(double km, double density, int year) { + logger.debug("adding " + year); + if (this.densities.containsKey(year)) { + List<SedimentDensityValue> list = this.densities.get(year); + list.add(new SedimentDensityValue(km, density, year)); + } + else { + List<SedimentDensityValue> list = + new ArrayList<SedimentDensityValue>(); + list.add(new SedimentDensityValue(km, density, year)); + densities.put(year, list); + } + if (!this.years.contains(new Integer(year))) { + logger.debug("new year"); + years.add(new Integer(year)); + } + } + + public List<Integer> getYears() { + return years; + } + + public void setYears(List<Integer> years) { + this.years = years; + } + + public double getDensity(double km, int year) { + Collections.sort(this.years); + if (this.years.size() == 1) { + return getDensityAtKm(densities.get(year), km); + } + else { + for (int i = 0; i < years.size() -1; i++) { + int y1 = years.get(i); + int y2 = years.get(i + 1); + int mid = Math.round((y1 + y2) / 2); + if (year < mid) { + return getDensityAtKm(densities.get(y1), km); + } + else if (i == years.size() -1) { + continue; + } + else { + return getDensityAtKm(densities.get(y2), km); + } + } + } + return -1d; + } + + private double getDensityAtKm( + List<SedimentDensityValue> values, + double km + ) { + boolean found = true; + SedimentDensityValue prev = null; + SedimentDensityValue next = null; + for (SedimentDensityValue sdv: values) { +logger.debug("year: " + sdv.getYear()); + if (sdv.getKm() == km) { + prev = sdv; + found = true; + break; + } + if (sdv.getKm() > km) { + next = sdv; + break; + } + prev = sdv; + } + if (found) { + return prev.getDensity(); + } + else { + return spline(prev, next, km); + } + } + + private double spline( + SedimentDensityValue prev, + SedimentDensityValue next, + double km + ) { + double lower = prev.getKm(); + double upper = next.getKm(); + double upperDensity = next.getDensity(); + double lowerDensity = prev.getDensity(); + + double m =(upperDensity - lowerDensity)/(upper - lower) * km; + double b = lowerDensity - + ((upperDensity - lowerDensity)/(upper - lower) * lower); + return (m * km) + b; + } + + public void cleanUp() { + Set<Integer> keys = densities.keySet(); + for (Integer key : keys) { + List<SedimentDensityValue> list = densities.get(key); + if (list.size() == 0) { + return; + } + List<SedimentDensityValue> cleaned = + new ArrayList<SedimentDensityValue>(); + double prevkm = list.get(0).getKm(); + int counter = 0; + double sum = 0d; + for (SedimentDensityValue value : list) { + if (value.getKm() == prevkm) { + sum += value.getDensity(); + counter++; + } + else { + cleaned.add(new SedimentDensityValue( + prevkm, + sum / counter, + value.getYear())); + sum = value.getDensity(); + counter = 1; + } + prevkm = value.getKm(); + } + this.densities.put(key, cleaned); + } + } +}
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/flys-artifacts/src/main/java/de/intevation/flys/artifacts/model/minfo/SedimentDensityFactory.java Wed Nov 14 17:17:02 2012 +0100 @@ -0,0 +1,90 @@ +package de.intevation.flys.artifacts.model.minfo; + +import java.util.List; + +import net.sf.ehcache.Cache; +import net.sf.ehcache.Element; + +import org.apache.log4j.Logger; +import org.hibernate.SQLQuery; +import org.hibernate.Session; +import org.hibernate.type.StandardBasicTypes; + +import de.intevation.flys.artifacts.cache.CacheFactory; +import de.intevation.flys.backend.SessionHolder; + + +public class SedimentDensityFactory +{ + /** Private logger to use here. */ + private static Logger log = Logger.getLogger(SedimentDensityFactory.class); + + private static final String DENSITY_CACHE_NAME = "sedimentdensity"; + + /**Query to get sediment density values and kms. */ + private static final String SQL_SELECT_DENSITY = + "SELECT sdv.station AS km, " + + " sdv.density AS density," + + " sdv.year AS year " + + " FROM sediment_density sd" + + " JOIN rivers r ON sd.river_id = r.id " + + " JOIN sediment_density_values sdv ON sd.id = sdv.sediment_density_id" + + " WHERE r.name = :name"; + + private SedimentDensityFactory() {} + + public static SedimentDensity getSedimentDensity( + String river, + double startKm, + double endKm, + int year + ) { + log.debug("getSedimentDensity"); + Cache cache = CacheFactory.getCache(DENSITY_CACHE_NAME); + + if (cache == null) { + log.debug("Cache not configured."); + return getSedimentDensityUncached(river, startKm, endKm, year); + } + + String key = river + startKm + endKm; + Element element = cache.get(key); + if (element != null) { + log.debug("SedimentDensity found in cache!"); + return (SedimentDensity)element.getValue(); + } + SedimentDensity value = + getSedimentDensityUncached(river, startKm, endKm, year); + + if (value != null && key != null) { + log.debug("Store sediment density values in cache."); + element = new Element(key, value); + cache.put(element); + } + return value; + } + + private static SedimentDensity getSedimentDensityUncached( + String river, + double startKm, + double endKm, + int year + ) { + log.debug("getSedimentDensityUncached"); + List<Object[]> results = null; + Session session = SessionHolder.HOLDER.get(); + SQLQuery sqlQuery = session.createSQLQuery(SQL_SELECT_DENSITY) + .addScalar("km", StandardBasicTypes.DOUBLE) + .addScalar("density", StandardBasicTypes.DOUBLE) + .addScalar("year", StandardBasicTypes.INTEGER); + sqlQuery.setString("name", river); + results = sqlQuery.list(); + SedimentDensity density = new SedimentDensity(); + for (Object[] row : results) { + density.addDensity((Double)row[0], (Double)row[1], (Integer)row[2]); + } + + density.cleanUp(); + return density; + } +}
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/flys-artifacts/src/main/java/de/intevation/flys/artifacts/model/minfo/SedimentDensityValue.java Wed Nov 14 17:17:02 2012 +0100 @@ -0,0 +1,46 @@ +package de.intevation.flys.artifacts.model.minfo; + + +public class SedimentDensityValue +{ + + private double km; + private double density; + private int year; + + public SedimentDensityValue() { + this.km = 0d; + this.density = 0d; + this.year = 0; + } + + public SedimentDensityValue(double km, double density, int year) { + this.km = km; + this.density = density; + this.year = year; + } + + public double getKm() { + return km; + } + + public void setKm(double km) { + this.km = km; + } + + public double getDensity() { + return density; + } + + public void setDensity(double density) { + this.density = density; + } + + public int getYear() { + return year; + } + + public void setYear(int year) { + this.year = year; + } +}