-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrose.cpp
403 lines (348 loc) · 11.4 KB
/
rose.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
#include <bits/stdc++.h>
#include <vector>
#include <queue>
#include <limits>
#include <cmath>
#include <unordered_map>
#include <set>
#define rep(i,a,b) for(int i=a;i<=b;i++)
#define pb push_back
using namespace std;
const int INF = 1e9;
// 图的结构:使用邻接表表示
struct Edge {
int to;
int weight;
};
int n;
class Graph {
public:
int n; // 节点数量
vector<vector<Edge>> adjList; // 邻接表
Graph(int nodes) : n(nodes) {
adjList.resize(n);
}
void addEdge(int u, int v, int w) {
adjList[u].push_back({v, w});
adjList[v].push_back({u, w});
}
// Dijkstra 算法,用于计算单源最短路径
int dijkstra(int start, int end){//cout<<"!!";
vector<int> dist(n, INF);
priority_queue<pair<int, int>, vector<pair<int, int>>, greater<pair<int, int>>> pq;
dist[start] = 0;pq.push({0,start});
while (!pq.empty()) {
int d = pq.top().first;
int u = pq.top().second;
//cout<<u<<"?"<<endl;
pq.pop();
if(d > dist[u]) continue;
if(u == end)
{
//cout<<dist[u]<<"!@!@#@!#"<<endl;
return dist[u];
}
for (Edge& edge : adjList[u]) {
int v = edge.to;
int w = edge.weight;
if (dist[u] + w < dist[v]) {
dist[v] = dist[u] + w;
pq.push({dist[v], v});
}
}
}
}
vector<int> dijkstra(int start) {
vector<int> dist(n, INF);
priority_queue<pair<int, int>, vector<pair<int, int>>, greater<pair<int, int>>> pq;
dist[start] = 0;
pq.push({0, start});
while (!pq.empty()) {
int d = pq.top().first;
int u = pq.top().second;
pq.pop();
if (d > dist[u]) continue;
for (Edge& edge : adjList[u]) {
int v = edge.to;
int w = edge.weight;
if (dist[u] + w < dist[v]) {
dist[v] = dist[u] + w;
pq.push({dist[v], v});
}
}
}
return dist;
}
};
// RNII索引
class RNII {
public:
vector<int> referenceNodes;
unordered_map<int, vector<int>> poiDistances;
RNII() {}
void GetNodes(Graph& graph, const vector<int>& pois) {
// 假设你希望随机选择 50 个参考节点
int numReferences = min(n-2,10);
vector<int> allNodes = pois; // 可以从 POI 列表中选择参考节点
random_device rd;
mt19937 g(rd());
shuffle(allNodes.begin(), allNodes.end(), g);
// 选择前 50 个作为参考节点
referenceNodes.assign(allNodes.begin(), allNodes.begin() + numReferences);
}
void buildIndex(Graph& graph, const vector<int>& pois) {
for (int ref : referenceNodes) {
vector<int> dist = graph.dijkstra(ref);
for (int poi : pois) {
poiDistances[poi].push_back(dist[poi]);
}
}
}
int tot=0;
int estimateDistance(int poi1, int poi2) {
int lowerBound = 0;
// cout<<poiDistances[1120].size()<<" "<<++tot<<endl;
for (int i = 0; i < referenceNodes.size(); i++) {
//cout<<poi1<<" "<<poi2<<endl;
// if(tot>=55137)
// cout<<"sd"<<" "<<tot<<" "<<poi1<<" "<<poi2<<" "<<i<<" "<<referenceNodes.size()<<endl;
int dist1 = poiDistances[poi1][i];
// if(tot>=55137)
//cout<<poiDistances[poi1][i]<<" "<<referenceNodes[i]<<" "<<poiDistances[poi2].size()<<" "<<poi2<<endl; if(tot==55137)cout<<tot<<" "<<poiDistances[poi2][i]<<endl;
int dist2 = poiDistances[poi2][i];//cout<<"Sd!";
lowerBound = max(lowerBound, abs(dist1 - dist2));
}//cout<<"sd!";
return lowerBound;
}
};
const int TT=25,CC=155,KK=15;
int k;
int T;
struct Compare {
bool operator()(const pair<int, vector<int>>& a, const pair<int, vector<int>>& b) {
if(a.first!=b.first)return a.first < b.first; return a.second.size()<b.second.size();
}
};
// KROSE算法
int start, ed;int K;vector<int>c[CC];//装包含某关键词的所有点
class KROSE {
public:
Graph& graph;
RNII& rnii;
int costThresh; // 剪枝阈值
KROSE(Graph& g, RNII& r) : graph(g), rnii(r), costThresh(INF) {}
map<pair<int,int> ,int>diss;
int getdis(int u,int v){
if(u>v)swap(u,v);
if(diss.count({u,v}))return diss[{u,v}];
return rnii.estimateDistance(u,v);
}
pair<int,vector<int> > OS[TT][CC][KK];
int n;
// 动态规划扩展Top-k路径,带POI顺序约束
vector<pair<int, vector<int> > > Top_K_ose(const vector<int>& poiOrder) {
vector<int> dist = graph.dijkstra(start);
vector<int> nullpath,nowpath;
nullpath.clear();
vector<pair<int, vector<int>> > paths;
//vector<vector<int>> paths = greedyPaths(start, end, poiOrder);
rep(i,1,T){
rep(j,0,c[poiOrder[i]].size()-1){
if(i!=1){
//cout<<i<<"?"<<endl;
priority_queue<pair<int, vector<int>>, vector<pair<int, vector<int>>>, Compare> Q;
rep(l,0,c[poiOrder[i-1]].size()-1){
rep(k,1,K)
{
nowpath=OS[i-1][l][k].second;
nowpath.pb(c[poiOrder[i]][j]);//if(i==2)cout<<nowpath.size()<<"?"<<endl;
Q.push({OS[i-1][l][k].first+getdis(c[poiOrder[i-1]][l],c[poiOrder[i]][j]),nowpath});
}
while(Q.size()>K)
{
// exit(0);
Q.pop();//将大于第k个的排除
}
}
int nowk=K;
/*
if(i==2){
cout<<Q.size()<<endl;
while(Q.size()){
cout<<Q.top().first<<"!!"<<endl;
for(int u:Q.top().second)cout<<u<<"?"<<endl;
Q.pop();
}
exit(0);
//exit(0);
}*/
while(!Q.empty()){
OS[i][j][nowk--]=Q.top();
/* if(i==2){
cout<<Q.top().first<<"!!"<<" "<<i<<" "<<j<<" "<<nowk+1<<endl;
cout<<Q.top().second.size()<<"!!!"<<endl;//exit(0);
for(int u:Q.top().second)cout<<u<<"?"<<endl;
//exit(0);
}*/
Q.pop();//从大到小
}
}
else{
nowpath=nullpath;
if(c[poiOrder[i]][j]!=start)
nowpath.pb(start);
nowpath.pb(c[poiOrder[i]][j]);
rep(k,1,K){
if(k==1){
OS[i][j][k]={getdis(start,c[poiOrder[i]][j]),nowpath};
}
else
OS[i][j][k]={1e8,nowpath};
}
}
}
}
priority_queue<pair<int, vector<int>>, vector<pair<int, vector<int>>>, Compare> QQ;
rep(j,0,c[poiOrder[T]].size()-1)
{
rep(k,1,K)
{
nowpath=OS[T][j][k].second;
//if(ed!=c[poiOrder[T]][j])nowpath.pb(ed);
// cout<<OS[T][j][k].first<<"?"<<c[T][j]<<" "<<ed<<endl;
QQ.push({OS[T][j][k].first//+getdis(c[poiOrder[T]][j],ed)
,nowpath});//cout<<"sd";
while(QQ.size()>K)QQ.pop();//用堆把大的踢出去
}
}
while(!QQ.empty()){
paths.pb(QQ.top());QQ.pop();//从大到小装前k个路径
}
return paths;
}
// 计算Top-k路径的精确距离
vector<int> computeExactDistances(const vector<pair<int,vector<int>> >& paths) {
vector<int> totalCosts(k, 0);
for (int i = 0; i < k; i++) {
//cout<<paths[i].second.size()<<endl;
for (size_t j = 1; j < paths[i].second.size(); j++) {
int disss = graph.dijkstra(paths[i].second[j - 1], paths[i].second[j]);
diss[{paths[i].second[j-1],paths[i].second[j]}]=diss[{paths[i].second[j],paths[i].second[j-1]}]=disss;
totalCosts[i] += disss;
}
}
return totalCosts;
}
// 执行KROSE算法
int tot=0;
vector<pair<int,vector<int>> > run(const vector<int>& poiOrder) {
vector<pair<int,vector<int>> > newPaths;
while (true) {
newPaths = Top_K_ose(poiOrder);
vector<int> newCosts = computeExactDistances(newPaths);
bool updated = false;
for (int i = 0; i < k; i++) {
// cout<<i<<" "<<newCosts[i]<<" "<<newPaths[i].first<<endl;
//for(auto u:newPaths[i].second)cout<<u<<endl;
//if(++tot>=10)exit(0);
if (newCosts[i]!=newPaths[i].first) {//预估不等于精确
updated = true;
}
}
if (!updated) {
break; // 如果没有更新任何路径,则结束迭代
}
}
return newPaths;
}
};
void get_Graph(Graph &graph) {
string filename = "C:\\Users\\hp\\Desktop\\USA-road-t.NY-stripped-1000.gr";
ifstream file(filename); // 打开文件
if (!file) { // 文件打开失败
cerr << "无法打开文件:" << filename << endl;
return;
}
string line;
// 跳过开头的注释部分
while (getline(file, line)) {
if (line[0] != 'a') continue; // 只处理以 'a' 开头的行
int u, v, w; // u: 起点, v: 终点, w: 权重
sscanf(line.c_str(), "a %d %d %d", &u, &v, &w); // 读取边的起点、终点和权重
graph.addEdge(u, v, w);
//edges.push_back(make_tuple(u, v, w)); // 将边存储到边的列表
}
file.close(); // 关闭文件
}
int val[10005];
vector<int>p;
void gettype(){
string filename = "C:\\Users\\hp\\Desktop\\USA-road-t.NY-stripped-1000.poi";
ifstream file(filename); // 打开文件
if (!file.is_open()) { // 检查文件是否成功打开
cerr << "Error: Could not open the file!" << endl;
return;
}
string line;
vector<int> thirdColumn, lastColumn;
//int totrr=0;
// 按行读取文件
while (getline(file, line)) {
stringstream s(line);
vector<int> values;
int value;
// 将每行的数据读取到values向量中
while (s >> value) {
values.push_back(value);
}
// 提取第3列和最后一列数据
if (values.size() >= 4) {
// cout<<"sd";
// if(c[values[2]].size()>=8)continue;
p.pb(values[1]);//加入关键编号
val[values[1]]=values[4];
c[values[2]].pb(values[1]);
//if(values[2]==2)totrr++;
}
}
//cout<<totrr<<"!!"<<endl;
file.close(); // 关闭文件
//rep(i,1,200)_Poi[i].r=16000+rand(),_Poi[i].key=ss[(i%15+1)]; //给点的积分赋值
//rep(i,1,5)cout<<_Poi[i].key<<endl;
}
int main() {
// 创建图
n = 10005;
Graph graph(n);gettype();
get_Graph(graph);
// 添加边
// getchar();
// 定义POI顺序
vector<int> poiOrder = {0, 1, 2, 5,6}; // POI访问顺序:1 -> 2 -> 3
// 构建RNII索引
RNII rnii;
rnii.GetNodes(graph,p); // 参考节点
rnii.buildIndex(graph,p);
// 运行KROSE算法-
K = 10; // Top-k值
k= 10 ;
start = 810;
// ed = 1103;
T=4;
KROSE krose(graph, rnii);
vector<pair<int,vector<int>> > result = krose.run(poiOrder);
sort(result.begin(),result.end(),Compare());
cout << "Top-k optimal paths (with POI order):" << endl;
for (int i = 0; i < k; i++) {
long long res=0;
rep(j,1,T){
res+=val[result[i].second[j]];
}
cout << "Path " << i + 1 << ": "<<result[i].first/1000<<" "<<res<<" "<<0.2*res-0.0008*result[i].first<<endl;
for (auto node : result[i].second) {
cout << node << " ";
}
cout << endl;
}
return 0;
}