From 81f55825df260fd7931f3c1ccd5a11bdde0228bb Mon Sep 17 00:00:00 2001 From: Peter Alexander <peter@blackhillock.co.uk> Date: Tue, 24 Jul 2018 11:49:16 +0100 Subject: [PATCH] More changes to match feed amounts --- GAMS/IntExtOpt.gms | 21 ++++++++++++------- src/ac/ed/lurg/ModelConfig.java | 6 +++--- src/ac/ed/lurg/country/CountryAgent.java | 8 ++++++- .../lurg/country/gams/GamsCountryInput.java | 11 ++++++++-- .../country/gams/GamsLocationOptimiser.java | 1 + 5 files changed, 33 insertions(+), 14 deletions(-) diff --git a/GAMS/IntExtOpt.gms b/GAMS/IntExtOpt.gms index 1001ce4d..4afcee4d 100644 --- a/GAMS/IntExtOpt.gms +++ b/GAMS/IntExtOpt.gms @@ -42,13 +42,14 @@ SCALAR pastureDecCost price for decreasing pasture; SCALAR meatEfficency efficiency of converting feed and pasture into animal products; + SCALAR animalFeedFromOtherSources animal feed from source other than modelled crops; SCALAR fertiliserUnitCost fert cost at max fert rate; SCALAR otherIParam yield response to other intensity; SCALAR otherICost cost of other intensity; SCALAR unhandledCropRate rate of fruit veg and other crops not modelled; SCALAR setAsideRate rate of set aside and failed crop; SCALAR domesticPriceMarkup factor price increased from cost of production; - SCALAR maxLandExpansionRate max rate of country land expansion + SCALAR maxLandExpansionRate max rate of country land expansion; *$gdxin "/Users/peteralexander/Documents/R_Workspace/UNPLUM/temp/GamsTmp/t1.gdx" $gdxin %gdxincname% @@ -56,11 +57,15 @@ $load location, suitableLandArea, demand, agriExpansionCost, cropIncCost, pastur $load previousArea, previousFertIntensity, previousIrrigIntensity, previousOtherIntensity, previousRuminantFeed, previousMonogastricFeed, previousImportAmount, previousExportAmount $load yieldNone, yieldFertOnly, yieldIrrigOnly, yieldBoth $load fertParam, irrigParam, otherIParam, exportPrices, importPrices, maxNetImport, minNetImport, unhandledCropRate, setAsideRate, maxLandExpansionRate -$load meatEfficency, otherICost, irrigCost, irrigMaxRate, irrigConstraint, fertiliserUnitCost, domesticPriceMarkup, minDemandPerCereal, seedAndWasteRate +$load meatEfficency, otherICost, irrigCost, irrigMaxRate, irrigConstraint, fertiliserUnitCost, domesticPriceMarkup, minDemandPerCereal, seedAndWasteRate, animalFeedFromOtherSources $gdxin SCALAR delta use to smooth power function see 7.5 www.gams.com dd docs solversconopt.pdf / 0.00000000001 /; - + + SCALAR ruminantOtherFeed; + SCALAR monogastricOtherFeed; + ruminantOtherFeed = animalFeedFromOtherSources * 0.25; + monogastricOtherFeed = animalFeedFromOtherSources * 0.75; previousArea(crop_less_pasture, location) = previousArea(crop_less_pasture, location) * (1.0 - unhandledCropRate); @@ -76,10 +81,10 @@ $gdxin PARAMETER prodCost(crop) cost per ha before intensity values not including fertiliser or irrigation i.e. seed spray and other. In 1000 $ per ha / wheat 0.32 maize 0.31 - rice 0.4 + rice 0.36 oilcrops 0.2 pulses 0.31 - starchyRoots 3.64 + starchyRoots 3.14 energyCrops 0.34 / ; PARAMETER baseCost(crop); @@ -164,12 +169,12 @@ $gdxin TOTAL_CEREAL_DEMAND_CONSTRAINT .. sum(cereal_crop, net_supply(cereal_crop)) =G= demand('cereals'); - RUMINANT_DEMAND_CONSTRAINT .. meatEfficency*sum(feed_crop, ruminantFeed(feed_crop) * cropDM(feed_crop)) * (1 - seedAndWasteRate('ruminants')) =G= (demand('ruminants') - importAmount('ruminants') + exportAmount('ruminants')) * (1.0 - unhandledCropRate); + RUMINANT_DEMAND_CONSTRAINT .. meatEfficency*(sum(feed_crop, ruminantFeed(feed_crop) * cropDM(feed_crop)) + ruminantOtherFeed) * (1 - seedAndWasteRate('ruminants')) =G= (demand('ruminants') - importAmount('ruminants') + exportAmount('ruminants')); - MONOGASTRICS_DEMAND_CONSTRAINT .. meatEfficency*sum(feed_crop_less_pasture, monogastricFeed(feed_crop_less_pasture) * cropDM(feed_crop_less_pasture)) * (1 - seedAndWasteRate('monogastrics')) =G= (demand('monogastrics') - importAmount('monogastrics') + exportAmount('monogastrics')) * (1.0 - unhandledCropRate); + MONOGASTRICS_DEMAND_CONSTRAINT .. meatEfficency*(sum(feed_crop_less_pasture, monogastricFeed(feed_crop_less_pasture) * cropDM(feed_crop_less_pasture)) + monogastricOtherFeed) * (1 - seedAndWasteRate('monogastrics')) =G= (demand('monogastrics') - importAmount('monogastrics') + exportAmount('monogastrics')); TOTAL_NON_PASTURE_FEED_DM_CALC .. totalFeedDM =E= sum(feed_crop_less_pasture, (ruminantFeed(feed_crop_less_pasture) + monogastricFeed(feed_crop_less_pasture)) * cropDM(feed_crop_less_pasture)); - FEED_MIX_CONSTRAINT(feed_crop_less_pasture) .. (ruminantFeed(feed_crop_less_pasture) + monogastricFeed(feed_crop_less_pasture)) * cropDM(feed_crop_less_pasture) =L= totalFeedDM * 0.8; + FEED_MIX_CONSTRAINT(feed_crop_less_pasture) .. (ruminantFeed(feed_crop_less_pasture) + monogastricFeed(feed_crop_less_pasture)) * cropDM(feed_crop_less_pasture) =L= totalFeedDM * 0.7; MAX_FERT_INTENSITY_CONSTRAINT(crop, location) .. fertI(crop, location) =L= 1; MAX_IRRIG_INTENSITY_CONSTRAINT(crop, location) .. irrigI(crop, location) =L= 1; diff --git a/src/ac/ed/lurg/ModelConfig.java b/src/ac/ed/lurg/ModelConfig.java index 6ca610da..44720cc7 100644 --- a/src/ac/ed/lurg/ModelConfig.java +++ b/src/ac/ed/lurg/ModelConfig.java @@ -216,7 +216,7 @@ public class ModelConfig { public static final boolean LIMIT_CEREAL_FRACTION = getBooleanProperty("LIMIT_CEREAL_FRACTION", true); public static final double PASTURE_HARVEST_FRACTION = getDoubleProperty("PASTURE_HARVEST_FRACTION", 0.5); - public static final double ANIMAL_FEED_FROM_WASTE_AND_OTHER_SOURCES = getDoubleProperty("ANIMAL_FEED_FROM_WASTE_AND_OTHER_SOURCES", 0.15); // animal nutrition coming from sources other than from crops purpose grown as feed + public static final double ANIMAL_FEED_FROM_OTHER_SOURCES_RATE = getDoubleProperty("ANIMAL_FEED_FROM_OTHER_SOURCES_RATE", 0.127); // animal nutrition coming from sources other crops modelled public static final double MEAT_EFFICIENCY = getDoubleProperty("MEAT_EFFICIENCY", 1.0); // 'meat' is includes feed conversion ratio already, this is tech. change or similar public static final double IRRIGIATION_EFFICIENCY = getDoubleProperty("IRRIGIATION_EFFICIENCY", 0.5); public static final int ELLIOTT_BASEYEAR = 2010; @@ -289,11 +289,11 @@ public class ModelConfig { public static final double UNHANDLED_CROP_RATE = getDoubleProperty("UNHANDLED_CROP_RATE", 0.177); // includes fruit veg forage crops set aside and failed crop public static final double SETASIDE_RATE = getDoubleProperty("SETASIDE_RATE", 0.103); // includes aside, fallow and failed cropland areas - public static final double OTHER_INTENSITY_COST = getDoubleProperty("OTHER_INTENSITY_COST", 0.6); + public static final double OTHER_INTENSITY_COST = getDoubleProperty("OTHER_INTENSITY_COST", 0.7); public static final double OTHER_INTENSITY_PARAM = getDoubleProperty("OTHER_INTENSITY_PARAM", 3.22); public static final double IRRIG_COST_SCALE_FACTOR = getDoubleProperty("IRRIG_COST_SCALE_FACTOR", 0.00035); - public static final double FERTILISER_COST_PER_T = getDoubleProperty("FERTILISER_COST_PER_T", 1.6); // $500/t, 18% N/t + public static final double FERTILISER_COST_PER_T = getDoubleProperty("FERTILISER_COST_PER_T", 1.5); // $500/t, 18% N/t public static final double FERTILISER_MAX_COST = FERTILISER_COST_PER_T * MAX_FERT_AMOUNT/1000; public static final double DOMESTIC_PRICE_MARKUP = getDoubleProperty("DOMESTIC_PRICE_MARKUP", 1.0); diff --git a/src/ac/ed/lurg/country/CountryAgent.java b/src/ac/ed/lurg/country/CountryAgent.java index 5df2fc4a..ea961dcf 100644 --- a/src/ac/ed/lurg/country/CountryAgent.java +++ b/src/ac/ed/lurg/country/CountryAgent.java @@ -42,6 +42,7 @@ public class CountryAgent { private Map<CropType, CountryPrice> currentCountryPrices; private Map<CropType, Double> tradeBarriers; private RasterSet<IntegerRasterItem> yieldClusters; + private double animalFeedFromOtherSources; public CountryAgent(AbstractDemandManager demandManager,CompositeCountry country, RasterSet<LandUseItem> cropAreaRaster, Map<CropType, CropUsageData> cropUsageData, Map<CropType, Double> tradeBarriers, RasterSet<IntegerRasterItem> yieldClusters) { @@ -108,6 +109,11 @@ public class CountryAgent { // get projected demand currentProjectedDemand = demandManager.getDemand(country, timestep.getYear()); + if (currentTimestep.isInitialTimestep()) { + double totalAnimalProductDemand = currentProjectedDemand.get(CommodityType.MONOGASTRICS) + currentProjectedDemand.get(CommodityType.RUMINANTS); + animalFeedFromOtherSources = totalAnimalProductDemand * ModelConfig.ANIMAL_FEED_FROM_OTHER_SOURCES_RATE; + } + currentCountryPrices = calculateCountryPrices(worldPrices); if(ModelConfig.LIMIT_MAIZE_YIELD_SAMERICA & country.getRegion().equals("Latin America & Caribbean")){ @@ -263,7 +269,7 @@ public class CountryAgent { Map<CropType, Double> minCerealFract = getMinCerealFraction(currentTimestep); GamsCountryInput countryLevelInputs = new GamsCountryInput(country, currentProjectedDemand, currentCountryPrices, importConstraints, - previousGamsRasterOutput.getCropUsageData(), minCerealFract); + previousGamsRasterOutput.getCropUsageData(), minCerealFract, animalFeedFromOtherSources); GamsRasterInput input = new GamsRasterInput(currentTimestep, countryYieldSurfaces, previousGamsRasterOutput.getLandUses(), irrigData, countryLevelInputs); return input; diff --git a/src/ac/ed/lurg/country/gams/GamsCountryInput.java b/src/ac/ed/lurg/country/gams/GamsCountryInput.java index 22814d2e..6272692d 100644 --- a/src/ac/ed/lurg/country/gams/GamsCountryInput.java +++ b/src/ac/ed/lurg/country/gams/GamsCountryInput.java @@ -19,9 +19,11 @@ public class GamsCountryInput { private Map<CropType, CountryPrice> countryPrices; private Map<CropType, CropUsageData> previousCropUsageData; private Map<CropType, Double> minCerealFraction; + private double animalFeedFromOtherSources; public GamsCountryInput(CompositeCountry country, Map<CommodityType, Double> projectedDemand, Map<CropType, CountryPrice> countryPrices, - Map<CropType, TradeOrProductionConstraint> importConstraints, Map<CropType, CropUsageData> previousCropUsageData, Map<CropType, Double> minCerealFraction) { + Map<CropType, TradeOrProductionConstraint> importConstraints, Map<CropType, CropUsageData> previousCropUsageData, Map<CropType, Double> minCerealFraction, + double animalFeedFromOtherSources) { super(); this.country = country; this.projectedDemand = projectedDemand; @@ -29,6 +31,7 @@ public class GamsCountryInput { this.countryPrices = countryPrices; this.previousCropUsageData = previousCropUsageData; this.minCerealFraction = minCerealFraction; + this.animalFeedFromOtherSources = animalFeedFromOtherSources; } public CompositeCountry getCountry() { @@ -52,7 +55,7 @@ public class GamsCountryInput { } public double getMeatEfficiency() { - return ModelConfig.MEAT_EFFICIENCY * (1 - ModelConfig.ANIMAL_FEED_FROM_WASTE_AND_OTHER_SOURCES); // this is already handled by the feed conversion efficiency for each animal product + return ModelConfig.MEAT_EFFICIENCY; // this is already handled by the feed conversion efficiency for each animal product } public Map<CropType, Double> getMinTradeOrProd() { @@ -74,4 +77,8 @@ public class GamsCountryInput { public Map<CropType, Double> getMinCerealFraction() { return minCerealFraction; } + + public double getAnimalFeedFromOtherSources() { + return animalFeedFromOtherSources; + } } \ No newline at end of file diff --git a/src/ac/ed/lurg/country/gams/GamsLocationOptimiser.java b/src/ac/ed/lurg/country/gams/GamsLocationOptimiser.java index e3dac733..c29ab364 100644 --- a/src/ac/ed/lurg/country/gams/GamsLocationOptimiser.java +++ b/src/ac/ed/lurg/country/gams/GamsLocationOptimiser.java @@ -285,6 +285,7 @@ public class GamsLocationOptimiser { addScalar(inDB, "unhandledCropRate", ModelConfig.UNHANDLED_CROP_RATE, 3); addScalar(inDB, "setAsideRate", ModelConfig.SETASIDE_RATE, 5); addScalar(inDB, "domesticPriceMarkup", ModelConfig.DOMESTIC_PRICE_MARKUP, 3); + addScalar(inDB, "animalFeedFromOtherSources", countryInput.getAnimalFeedFromOtherSources(), 2); double maxExpansion = 1.0; if (!ModelConfig.IS_CALIBRATION_RUN && countryInput.getCountry().getName().equals("China")) { maxExpansion = ModelConfig.MAX_CHINA_LAND_EXPANSION_RATE; -- GitLab