using SqlSugar; using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; using WIDESEA_Common.CommonEnum; using WIDESEA_Common.LocationEnum; using WIDESEA_Core.BaseServices; using WIDESEA_Core.Helper; using WIDESEA_IBasicRepository; using WIDESEA_IBasicService; using WIDESEA_Model.Models; namespace WIDESEA_BasicService { public partial class LocationInfoService { double weightValue_CSJ = 0.25; private readonly static object _locker_CSJ = new object(); static List locationCaches_CSJ = new List(); /// /// 测试架仓货位分配
/// 托盘类型:
/// 1:短
/// 2:中等
/// 3:长
/// 4:特长
/// 1和2都是3深,且可以混放。
/// 3和4都是2深,且可以混放。
/// 移库时只能是同侧货位。
/// 6排43列13层---13层不可用 ///
/// 巷道号 /// /// 托盘类型
/// 1:短
/// 2:中等
/// 3:长
/// 4:特长 /// /// public Dt_LocationInfo? AssignLocation_CSJ(string roadwayNo, PalletTypeEnum palletType, string beRelocationCode = "") { lock (_locker_CSJ) { List removeItems = locationCaches_CSJ.Where(x => (DateTime.Now - x.DateTime).TotalMinutes > 5).ToList(); int count = removeItems.Count; for (int i = 0; i < count; i++) { locationCaches_CSJ.Remove(removeItems[i]); } List lockLocationCodes = locationCaches_CSJ.Select(x => x.LocationCode).ToList(); List locationInfos = Repository.QueryData(x => x.RoadwayNo == roadwayNo); if (locationInfos == null || locationInfos.Count == 0) { throw new Exception($"未找到该巷道的货位信息,巷道号:{roadwayNo}"); } if (!string.IsNullOrEmpty(beRelocationCode)) { Dt_LocationInfo? beRelocation = locationInfos.FirstOrDefault(x => x.LocationCode == beRelocationCode); if (beRelocation == null) { throw new Exception($"未找到货位信息"); } int maxDepth = locationInfos.Max(x => x.Depth); int mathCurrentRow = beRelocation.Row - Convert.ToInt32(Math.Ceiling(beRelocation.Row / maxDepth / 2.0)) * maxDepth * 2; if (mathCurrentRow <= maxDepth) { locationInfos = locationInfos.Where(x => x.Row - Convert.ToInt32(Math.Ceiling(x.Row / maxDepth / 2.0)) * maxDepth * 2 <= maxDepth).ToList(); } else { locationInfos = locationInfos.Where(x => x.Row - Convert.ToInt32(Math.Ceiling(x.Row / maxDepth / 2.0)) * maxDepth * 2 > maxDepth).ToList(); } } //已定义货位类型的货位 List definedTypeLocations = locationInfos.Where(x => x.LocationType == palletType.ObjToInt()).ToList(); //未定义类型的空货位 List undefinedTypeEmptyLocations = locationInfos.Where(x => (x.EnableStatus == EnableStatusEnum.Normal.ObjToInt() || x.EnableStatus == EnableStatusEnum.OnlyIn.ObjToInt()) && x.LocationStatus == LocationStatusEnum.Free.ObjToInt() && x.LocationType == LocationTypeEnum.Undefined.ObjToInt() && !lockLocationCodes.Contains(x.LocationCode)).OrderByDescending(x => x.Depth).ThenBy(x => x.Layer).ThenBy(x => x.Column).ThenBy(x => x.Row).ToList(); List definedTypeEmptyLocations = locationInfos.Where(x => (x.EnableStatus == EnableStatusEnum.Normal.ObjToInt() || x.EnableStatus == EnableStatusEnum.OnlyIn.ObjToInt()) && x.LocationStatus == LocationStatusEnum.Free.ObjToInt() && x.LocationType == palletType.ObjToInt() && !lockLocationCodes.Contains(x.LocationCode)).OrderByDescending(x => x.Depth).ThenBy(x => x.Layer).ThenBy(x => x.Column).ThenBy(x => x.Row).ToList(); if ((locationInfos.Count * weightValue_CSJ >= definedTypeLocations.Count && undefinedTypeEmptyLocations.Any()) || !definedTypeEmptyLocations.Any())//如果已定义类型货位未超过比例,且有未定义类型的货位 { if (palletType == PalletTypeEnum.LargePallet || palletType == PalletTypeEnum.LargestPallet) { undefinedTypeEmptyLocations = undefinedTypeEmptyLocations.Where(x => x.Depth % 2 == 1).ToList(); } for (int i = 0; i < undefinedTypeEmptyLocations.Count; i++) { Dt_LocationInfo undefinedTypeEmptyLocation = undefinedTypeEmptyLocations[i]; Dt_LocationInfo? locationInfo = GetUsableLocation_CSJ(locationInfos, undefinedTypeEmptyLocation, palletType); if (locationInfo != null) { locationCaches_CSJ.Add(new LocationCache { LocationCode = locationInfo.LocationCode, DateTime = DateTime.Now }); return locationInfo; } } } else { for (int i = 0; i < definedTypeEmptyLocations.Count; i++) { Dt_LocationInfo definedTypeEmptyLocation = definedTypeEmptyLocations[i]; Dt_LocationInfo? locationInfo = GetUsableLocation_CSJ(locationInfos, definedTypeEmptyLocation, palletType); if (locationInfo != null) { locationCaches_CSJ.Add(new LocationCache { LocationCode = locationInfo.LocationCode, DateTime = DateTime.Now }); return locationInfo; } } } return null; } } /// /// 获取可用货位(测试架仓) /// /// /// /// /// private Dt_LocationInfo? GetUsableLocation_CSJ(List locationInfos, Dt_LocationInfo emptyLocation, PalletTypeEnum palletType) { switch (palletType) { case PalletTypeEnum.LargestPallet: case PalletTypeEnum.LargePallet: if (emptyLocation.Depth % 2 == 0) { return null; } break; } Dt_LocationInfo? locationInfo = DepthLocationIsEmpty_CSJ(locationInfos, emptyLocation, palletType); if (locationInfo != null) { return locationInfo; } return null; } private bool LittleDepthLocationIsEmpty_CSJ(Dt_LocationInfo locationInfo, PalletTypeEnum palletType) { List locations = GetGroupLocations(locationInfo); switch (palletType) { case PalletTypeEnum.LargestPallet: case PalletTypeEnum.LargePallet: { List littleDepth = locations.Where(x => x.Depth <= locationInfo.Depth).ToList(); return littleDepth.FirstOrDefault(x => x.LocationStatus != LocationStatusEnum.Free.ObjToInt() && x.EnableStatus != EnableStatusEnum.OnlyIn.ObjToInt() && x.EnableStatus != EnableStatusEnum.Normal.ObjToInt()) == null; //查询小于当前货位深度的集合里是否有状态不为空,且禁用状态不为禁用以及只入的货位,如果是true,则表示浅货位被使用或者被禁用的情况 } case PalletTypeEnum.MediumPallet: case PalletTypeEnum.SmallPallet: { List littleDepth = locations.Where(x => x.Depth <= locationInfo.Depth).ToList(); return littleDepth.FirstOrDefault(x => x.LocationStatus != LocationStatusEnum.Free.ObjToInt() && x.EnableStatus != EnableStatusEnum.OnlyIn.ObjToInt() && x.EnableStatus != EnableStatusEnum.Normal.ObjToInt()) == null; //查询小于当前货位深度的集合里是否有状态不为空,且禁用状态不为禁用以及只入的货位,如果是true,则表示浅货位被使用或者被禁用的情况 } } return false; } /// /// 判断不同深度的同组货位状态是否为空闲空位(测试架仓) /// /// /// /// private Dt_LocationInfo? DepthLocationIsEmpty_CSJ(List locationInfos, Dt_LocationInfo emptyLocation, PalletTypeEnum palletType) { List locations = GetGroupLocations(locationInfos, emptyLocation); bool moreDepthFlag = false; bool littleDepthFlag = false; switch (palletType) { case PalletTypeEnum.LargestPallet: case PalletTypeEnum.LargePallet: { List moreDepth = locations.Where(x => x.Depth > emptyLocation.Depth).ToList(); moreDepthFlag = moreDepth.FirstOrDefault(x => x.LocationStatus != LocationStatusEnum.InStock.ObjToInt() && x.LocationStatus != LocationStatusEnum.PalletLock.ObjToInt() && (x.EnableStatus == EnableStatusEnum.OnlyIn.ObjToInt() || x.EnableStatus == EnableStatusEnum.Normal.ObjToInt())) == null;//查询大于当前货位深度的集合里是否有状态不为有货的货位,如果是true,则表示深货位有未被使用的情况 List littleDepth = locations.Where(x => x.Depth <= emptyLocation.Depth).ToList(); littleDepthFlag = littleDepth.FirstOrDefault(x => x.LocationStatus != LocationStatusEnum.Free.ObjToInt() && (x.EnableStatus == EnableStatusEnum.OnlyIn.ObjToInt() || x.EnableStatus == EnableStatusEnum.Normal.ObjToInt())) == null; //查询小于当前货位深度的集合里是否有状态不为空,且禁用状态不为禁用以及只入的货位,如果是true,则表示浅货位被使用或者被禁用的情况 } break; case PalletTypeEnum.MediumPallet: case PalletTypeEnum.SmallPallet: { List moreDepth = locations.Where(x => x.Depth > emptyLocation.Depth).ToList(); moreDepthFlag = moreDepth.FirstOrDefault(x => x.LocationStatus != LocationStatusEnum.InStock.ObjToInt() && (x.EnableStatus == EnableStatusEnum.OnlyIn.ObjToInt() || x.EnableStatus == EnableStatusEnum.Normal.ObjToInt())) == null;//查询大于当前货位深度的集合里是否有状态不为有货的货位,如果是true,则表示深货位有未被使用的情况 List littleDepth = locations.Where(x => x.Depth <= emptyLocation.Depth).ToList(); littleDepthFlag = littleDepth.FirstOrDefault(x => x.LocationStatus != LocationStatusEnum.Free.ObjToInt() && (x.EnableStatus == EnableStatusEnum.OnlyIn.ObjToInt() || x.EnableStatus == EnableStatusEnum.Normal.ObjToInt())) == null; //查询小于当前货位深度的集合里是否有状态不为空,且禁用状态不为禁用以及只入的货位,如果是true,则表示浅货位被使用或者被禁用的情况 } break; default: throw new Exception($"托盘类型错误"); } if (moreDepthFlag && littleDepthFlag) { return emptyLocation; } return null; } } }