[ Code Plus #4 ] 最短路

\(\\\)

\(Description\)


一张点的编号为\(1\text~N\)的有向图,初始给出 \(M\) 条边,有边权。其次还存在大量的边,其形式是:

对于任意两个编号分别为 \(i,j\) 的两个点,其之前存在一条边权为 \((i\oplus j)\times C\) 的边。

\(A\) 号点到 \(B\) 号点的最短路长度。

  • \(N\le10^5,M\le 5\times10^5,C\le 100\)

\(\\\)

\(Solution\)


\(N^2\) 建边显然空间时间都承受不住,考虑能否优化这个过程。

假如如此构造现在有一条从 \(2\)\(5\) 的边,显然这个边权是
\[ (111)_2\times C \]
那么我们观察图中的边,显然存在如下几条:
\[ 2\rightarrow 0:w=(010)_2\times C\\ 0\rightarrow 4:w=(100)_2\times C\\ 4\rightarrow 5:w=(001)_2\times C \]
有没有发现,走这三条路的权值和其实等价于走原来那一条路?

于是有了一个简单的做法:按照二进制位分开考虑。

也就是说,我们对于每一个点,除去给出的 \(M\) 条边外,只连出去边长为 \(2\) 的整次幂的边。

为什么这样建边等价于原来的图?因为我们每次只走二进制位里的一个 \(1\) ,最后走的总路径长度不变。

然后边的级别就是 \(NlogN+M\),存的下,跑个\(Dij\) 开个 \(O2\) 能过。

注意这个转移过程中我们经过的点的编号可能会超出\(N\),所以我们要把 \(N\) 加大到第一个 \(\ge N\)\(2\) 的幂,同时注意这些新加的点也要把对应的 \(log\) 条边建出来,否则就可能无法转移了。

\(\\\)

\(Code\)


#include<cmath>
#include<queue>
#include<cstdio>
#include<cctype>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#define N 200010
#define M 3000010
#define R register
#define gc getchar
using namespace std;
typedef long long ll;

int n,m,c,s,t,tot,bit,hd[N],dis[N];

bool vis[N];

struct edge{int to,nxt,w;}e[M];

inline void add(int u,int v,int w){
  e[++tot].to=v; e[tot].w=w;
  e[tot].nxt=hd[u]; hd[u]=tot;
}

inline int rd(){
  int x=0; bool f=0; char c=gc();
  while(!isdigit(c)){if(c=='-')f=1;c=gc();}
  while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=gc();}
  return f?-x:x;
}

priority_queue<pair<int,int> > q;

inline void dij(){
  memset(dis,0x3f,sizeof(dis));
  dis[s]=0; q.push(make_pair(0,s));
  R int u;
  while(!q.empty()){
    u=q.top().second; q.pop();
    if(vis[u]) continue; vis[u]=1;
    for(R int i=hd[u],v;i;i=e[i].nxt)
      if(dis[v=e[i].to]>dis[u]+e[i].w){
        dis[v]=dis[u]+e[i].w;
        q.push(make_pair(-dis[v],v));
      }
  }
  printf("%d\n",dis[t]);
}

int main(){
  n=rd(); m=rd(); c=rd();
  bit=log2(n)+1; n=(1<<bit)-1;
  for(R int i=1,u,v,w;i<=m;++i){
    u=rd(); v=rd(); w=rd(); add(u,v,w);
  }
  for(R int i=0;i<=n;++i)
    for(R int j=0;j<bit;++j) add(i,i^(1<<j),c*(1<<j));
  s=rd(); t=rd(); dij();
  return 0;
}

转载于:https://www.cnblogs.com/SGCollin/p/9810841.html

@Override public List<MyBatteryInfoEntity> queryInfo2Export(Map<String, Object> params) { String key = params.get("key")!=null?String.valueOf(params.get("key")):""; String deviceID = params.get("deviceId")!=null?String.valueOf(params.get("deviceId")):""; String starttime = params.get("starttime")!=null?String.valueOf(params.get("starttime")):""; String endtime = params.get("endtime")!=null?String.valueOf(params.get("endtime")):""; String startTime = StringUtils.isNotBlank(starttime)&&!"null".equals(starttime.trim())? DateUtils.format2(DateUtils.transferDateFormat(starttime),"yyyy-MM-dd HH:mm:ss"):""; String endTime =StringUtils.isNotBlank(endtime)&&!"null".equals(endtime.trim())?DateUtils.format2(DateUtils.transferDateFormat(endtime),"yyyy-MM-dd HH:mm:ss"):""; String sn = params.get("sn")!=null?String.valueOf(params.get("sn")):""; int project_id = params.get("projectId")!=null&&StringUtils.isNotBlank(String.valueOf(params.get("projectId")))? Integer.parseInt(String.valueOf(params.get("projectId"))):-1; int flag = params.get("flag")!=null&&StringUtils.isNotBlank(String.valueOf(params.get("flag")))? Integer.parseInt(String.valueOf(params.get("flag"))):-1; List<MyBatteryInfoEntity> result_list = myBatteryInfoDao.selectList(new QueryWrapper<MyBatteryInfoEntity>() .and(StringUtils.isNotBlank(key), qw->qw.like(StringUtils.isNotBlank(key), "battery_id", key) .or() .like(StringUtils.isNotBlank(key), "topic", key) .or() .like(StringUtils.isNotBlank(key), "battery_soft_version", key) ) .like(StringUtils.isNotBlank(deviceID),"battery_id",deviceID) .like(StringUtils.isNotBlank(sn), "sn", sn) .and(StringUtils.isNotBlank(startTime)||StringUtils.isNotBlank(endTime), qw->qw.ge(StringUtils.isNotBlank(startTime),"upload_time", startTime) .le(StringUtils.isNotBlank(endTime),"upload_time", endTime)) .eq(flag>0,"flag",flag) .eq(project_id>0,"project_id",project_id) .orderByAsc("upload_time")); // 状态转换等 if (result_list != null && result_list.size() > 0) { for (MyBatteryInfoEntity mtBatteryInfo : result_list) { // 电池状态 int btStatus = StringUtils.isNotBlank(mtBatteryInfo.getBatteryWorkMode())?Integer.parseInt(mtBatteryInfo.getBatteryWorkMode().trim()):0; String sysStatusValue = ""; if(btStatus>0) { if (btStatus == 16) { sysStatusValue = "充电("+btStatus+")"; } else if (btStatus == 1) { sysStatusValue = "放电("+btStatus+")"; } else if (btStatus == 33) { sysStatusValue = "搁置(保护-"+btStatus+")"; } else if (btStatus == 48) { sysStatusValue = "搁置(待机无输出-"+btStatus+")"; } else if (btStatus == 49) { sysStatusValue = "搁置(待机预放电-"+btStatus+")"; } else if (btStatus == 255) { sysStatusValue = "搁置(故障需返厂-"+btStatus+")"; } } mtBatteryInfo.setBatteryWorkModeValue(sysStatusValue); // 故障信息 if(StringUtils.isNotBlank(mtBatteryInfo.getBatteryErrorCode())){ StringBuffer errCodeValue = new StringBuffer(); // mtBatteryInfo.getBatteryErrorCode()分成3段 long errCode1 = Long.parseLong(mtBatteryInfo.getBatteryErrorCode().substring(0, 8), 16); long errCode2 = Long.parseLong(mtBatteryInfo.getBatteryErrorCode().substring(8, 16), 16); long errCode3 = Long.parseLong(mtBatteryInfo.getBatteryErrorCode().substring(16, 24), 16); String fault = "正常"; if(errCode1>0){ fault = "故障中"; if(Utility.analyseCode(errCode1,0)){ errCodeValue.append("电芯过压1级&"); } if(Utility.analyseCode(errCode1,1)){ errCodeValue.append("电芯过压2级&"); } if(Utility.analyseCode(errCode1,2)){ errCodeValue.append("电芯过压3级&"); } if(Utility.analyseCode(errCode1,3)){ errCodeValue.append("电芯过压4级&"); } if(Utility.analyseCode(errCode1,4)){ errCodeValue.append("整包过压1级&"); } if(Utility.analyseCode(errCode1,5)){ errCodeValue.append("整包过压2级&"); } if(Utility.analyseCode(errCode1,6)){ errCodeValue.append("整包过压3级&"); } if(Utility.analyseCode(errCode1,7)){ errCodeValue.append("充电器过压1级&"); } if(Utility.analyseCode(errCode1,8)){ errCodeValue.append("电芯欠压1级&"); } if(Utility.analyseCode(errCode1,9)){ errCodeValue.append("电芯欠压2级&"); } if(Utility.analyseCode(errCode1,10)){ errCodeValue.append("电芯欠压3级&"); } if(Utility.analyseCode(errCode1,11)){ errCodeValue.append("电芯欠压4级&"); } if(Utility.analyseCode(errCode1,12)){ errCodeValue.append("整包欠压1级&"); } if(Utility.analyseCode(errCode1,13)){ errCodeValue.append("整包欠压2级&"); } if(Utility.analyseCode(errCode1,14)){ errCodeValue.append("整包欠压3级&"); } if(Utility.analyseCode(errCode1,15)){ errCodeValue.append("整包欠压4级&"); } if(Utility.analyseCode(errCode1,16)){ errCodeValue.append("充电过流1级&"); } if(Utility.analyseCode(errCode1,17)){ errCodeValue.append("充电过流2级&"); } if(Utility.analyseCode(errCode1,18)){ errCodeValue.append("充电过流3级&"); } // if(Utility.analyseCode(errCode1,19)){ // errCodeValue.append("回馈过流1级&"); // } if(Utility.analyseCode(errCode1,20)){ // errCodeValue.append("回馈过流2级&"); // } if(Utility.analyseCode(errCode1,21)){ // errCodeValue.append("回馈过流3级&"); // } if(Utility.analyseCode(errCode1,22)){ // errCodeValue.append("回馈过流4级&"); // } if(Utility.analyseCode(errCode1,19)){ errCodeValue.append("充电过流4级&"); } if(Utility.analyseCode(errCode1,20)){ errCodeValue.append("回馈过流1级&"); } if(Utility.analyseCode(errCode1,21)){ errCodeValue.append("回馈过流2级&"); } if(Utility.analyseCode(errCode1,22)){ errCodeValue.append("回馈过流3级&"); } if(Utility.analyseCode(errCode1,23)){ errCodeValue.append("充电过流锁死1级&"); } if(Utility.analyseCode(errCode1,24)){ errCodeValue.append("放电过流1级&"); } if(Utility.analyseCode(errCode1,25)){ errCodeValue.append("放电过流2级&"); } if(Utility.analyseCode(errCode1,26)){ errCodeValue.append("放电过流3级&"); } if(Utility.analyseCode(errCode1,27)){ errCodeValue.append("放电过流4级&"); } if(Utility.analyseCode(errCode1,28)){ errCodeValue.append("放电过流5级&"); } if(Utility.analyseCode(errCode1,29)){ errCodeValue.append("放电过流6级&"); } if(Utility.analyseCode(errCode1,30)){ errCodeValue.append("放电过流7级&"); } if(Utility.analyseCode(errCode1,31)){ errCodeValue.append("放电过流锁死1级&"); } } if(errCode2>0){ fault = "故障中"; if(Utility.analyseCode(errCode2,0)){ errCodeValue.append("辅源短路1级&"); } if(Utility.analyseCode(errCode2,1)){ errCodeValue.append("预放超时1级&"); } if(Utility.analyseCode(errCode2,2)){ errCodeValue.append("预放锁死1级&"); } // if(Utility.analyseCode(errCode2,3)){ // errCodeValue.append("电芯过压4级&"); // } if(Utility.analyseCode(errCode2,4)){ // errCodeValue.append("整包过压1级&"); // } if(Utility.analyseCode(errCode2,5)){ // errCodeValue.append("整包过压2级&"); // } if(Utility.analyseCode(errCode2,6)){ // errCodeValue.append("整包过压3级&"); // } if(Utility.analyseCode(errCode2,7)){ // errCodeValue.append("整包过压4级&"); // } if(Utility.analyseCode(errCode2,8)){ errCodeValue.append("电芯充电高温1级&"); } if(Utility.analyseCode(errCode2,9)){ errCodeValue.append("电芯充电高温2级&"); } if(Utility.analyseCode(errCode2,10)){ errCodeValue.append("电芯充电低温1级&"); } if(Utility.analyseCode(errCode2,11)){ errCodeValue.append("电芯充电低温2级&"); } if(Utility.analyseCode(errCode2,12)){ errCodeValue.append("充电MOS高温1级&"); } if(Utility.analyseCode(errCode2,13)){ errCodeValue.append("充电MOS高温2级&"); } if(Utility.analyseCode(errCode2,14)){ errCodeValue.append("充电防反MOS高温1级&"); } if(Utility.analyseCode(errCode2,15)){ errCodeValue.append("充电防反MOS高温2级&"); } if(Utility.analyseCode(errCode2,16)){ errCodeValue.append("电芯放电高温1级&"); } if(Utility.analyseCode(errCode2,17)){ errCodeValue.append("电芯放电高温2级&"); } if(Utility.analyseCode(errCode2,18)){ errCodeValue.append("电芯放电低温1级&"); } if(Utility.analyseCode(errCode2,19)){ errCodeValue.append("电芯放电低温2级&"); } if(Utility.analyseCode(errCode2,20)){ errCodeValue.append("放电MOS高温1级&"); } if(Utility.analyseCode(errCode2,21)){ errCodeValue.append("放电MOS高温2级&"); } // if(Utility.analyseCode(errCode2,22)){ // errCodeValue.append("回馈过流4级&"); // } if(Utility.analyseCode(errCode2,23)){ // errCodeValue.append("充电过流锁死1级&"); // } if(Utility.analyseCode(errCode2,24)){ errCodeValue.append("均衡温度保护1级&"); } if(Utility.analyseCode(errCode2,25)){ errCodeValue.append("均衡温度保护2级&"); } if(Utility.analyseCode(errCode2,26)){ errCodeValue.append("辅源高温保护1级&"); } if(Utility.analyseCode(errCode2,27)){ errCodeValue.append("辅源低温保护1级&"); } // if(Utility.analyseCode(errCode2,28)){ // errCodeValue.append("放电过流5级&"); // } if(Utility.analyseCode(errCode2,29)){ // errCodeValue.append("放电过流6级&"); // } if(Utility.analyseCode(errCode2,30)){ // errCodeValue.append("放电过流7级&"); // } if(Utility.analyseCode(errCode2,31)){ // errCodeValue.append("放电过流锁死1级&"); // } } if(errCode3>0){ fault = "故障中"; if(Utility.analyseCode(errCode3,0)){ errCodeValue.append("充电MOS短路1级&"); } if(Utility.analyseCode(errCode3,1)){ errCodeValue.append("放电MOS短路1级&"); } if(Utility.analyseCode(errCode3,2)){ errCodeValue.append("NTC失效1级&"); } if(Utility.analyseCode(errCode3,3)){ errCodeValue.append("电芯采集断线1级&"); } if(Utility.analyseCode(errCode3,4)){ errCodeValue.append("电芯失衡1级&"); } if(Utility.analyseCode(errCode3,5)){ errCodeValue.append("电芯欠压失效1级&"); } if(Utility.analyseCode(errCode3,6)){ errCodeValue.append("AFE故障1级&"); } if(Utility.analyseCode(errCode3,7)){ errCodeValue.append("放电MOS断路1级&"); } if(Utility.analyseCode(errCode3,8)){ errCodeValue.append("FUSE熔断1级&"); } } if(fault.length()>0) { mtBatteryInfo.setBatteryErrorCodeValue(fault); } if(errCodeValue.length()>0) { mtBatteryInfo.setBatteryErrorCodeDetails(errCodeValue.toString().substring(0, errCodeValue.length() - 1)+"("+mtBatteryInfo.getBatteryErrorCode()+")"); }else{ mtBatteryInfo.setBatteryErrorCodeDetails("正常"); } } // Mos管状态 // bit4:均衡管状态( 0关闭, 1 有均衡管开启 // bit3:预放电 MOS 状态(0 关闭, 1 开启) // bit2:充电 MOS 状态0 关闭, 1 开启) // bit1:放电 MOS 状态0 关闭, 1 开启) // bit0:备用 long mosCode = mtBatteryInfo.getMosStatus()!=null?mtBatteryInfo.getMosStatus():0; StringBuffer mosCodeValue = new StringBuffer(); if(mosCode>=0){ if(Utility.analyseCode(mosCode,1)){ mosCodeValue.append("放电MOS状态:开启&"); mtBatteryInfo.setDischargeMosStatusValue("开启"); }else{ mosCodeValue.append("放电MOS状态:关闭&"); mtBatteryInfo.setDischargeMosStatusValue("关闭"); } if(Utility.analyseCode(mosCode,2)){ mosCodeValue.append("充电MOS状态:开启&"); mtBatteryInfo.setChargeMosStatusValue("开启"); }else{ mosCodeValue.append("充电MOS状态:关闭&"); mtBatteryInfo.setChargeMosStatusValue("关闭"); } if(Utility.analyseCode(mosCode,3)){ mosCodeValue.append("预放电MOS状态:开启&"); mtBatteryInfo.setPreDischargeMosStatusValue("开启"); }else{ mosCodeValue.append("预放电MOS状态:关闭&"); mtBatteryInfo.setPreDischargeMosStatusValue("关闭"); } if(Utility.analyseCode(mosCode,4)){ mosCodeValue.append("均衡管状态:开启&"); mtBatteryInfo.setBalanceTubeStateValue("开启"); }else{ mosCodeValue.append("均衡管状态:关闭&"); mtBatteryInfo.setBalanceTubeStateValue("关闭"); } } if(mosCodeValue.length()>0) { mtBatteryInfo.setMosStatusValue(mosCodeValue.toString().substring(0, mosCodeValue.length()-1)); } // 1:GPS标准坐标(wgs84); // 2:搜狗地图坐标; // 3:火星坐标(gcj02),即高德地图、腾讯地图和MapABC等地图使用的坐标; // 4:3中列举的地图坐标对应的墨卡托平面坐标; // 5:百度地图采用的经纬度坐标(bd09ll); // 6:百度地图采用的墨卡托平面坐标(bd09mc); // 7:图吧地图坐标; // 8:51地图坐标; String locationMode = ""; if(StringUtils.isNotBlank(mtBatteryInfo.getLocationMode())){ int modeNum = Integer.parseInt(mtBatteryInfo.getLocationMode().trim()); if(modeNum==1){ locationMode = "GPS标准坐标(wgs84)"; }else if (modeNum==2){ locationMode = "搜狗地图坐标"; }else if (modeNum==3){ locationMode = "火星坐标(gcj02)"; }else if (modeNum==4){ locationMode = "3中列举的地图坐标对应的墨卡托平面坐标"; }else if (modeNum==5){ locationMode = "百度地图采用的经纬度坐标(bd09ll)"; }else if (modeNum==6){ locationMode = "百度地图采用的墨卡托平面坐标(bd09mc)"; }else if (modeNum==7){ locationMode = "图吧地图坐标"; }else if (modeNum==8){ locationMode = "51地图坐标"; }else{ locationMode = ""; } } mtBatteryInfo.setLocationModeValue(StringUtils.isNotBlank(locationMode)?locationMode:"GPS标准坐标(wgs84)"); } } return result_list; } 使用多线程分批查询,循环也要优化,帮我优化整个代码,太慢了
07-15
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值