2-SAT之完美塔防

小N最近喜歡玩一款塔防游戲。

題目描述

這款游戲的棋盤是一個?n×m?的網格,每個格子上會有以下類型物件:

  1. A 型炮臺:會向上下兩個方向同時發射激光,符號為?|;
  2. B 型炮臺:會向左右兩個方向同時發射激光,符號為?-;
  3. 空地:激光穿過該物件會保持方向前進,符號為?.;
  4. 障礙:激光到達該物件會消失,符號為?#;
  5. 正反射鏡:激光到達該物件后,會依物理定律改變方向,但仍繼續前進,符號為?\;
  6. 副反射鏡:激光到達該物件后,會依物理定律改變方向,但仍繼續前進,符號為?/;

注意激光之間可以互相穿過,但如果激光射出網格邊界,也會消失。
小 N 是一個強迫癥玩家,他想讓每一處空地都會被至少一束激光打到,但不能讓激光攻擊到自己的炮臺以致損壞。
小 N 可以將任意多的 A 型炮臺改造成 B 型炮臺,也可以把任意多的 B 型炮臺改造成 A 型炮臺。
你可以告訴他能否通過這些改造實現他的目標嗎?

輸入格式

第一行一個正整數?T,表示數據組數。
每組數據第一行有兩個正整數?n,m。
接下來?n?行,每行一個長度為?m?的字符串,表示棋盤。

輸出格式

對于每組數據,若無解則輸出一行?IMPOSSIBLE,否則輸出一行?POSSIBLE,再輸出改造后的棋盤。
如果有多組解,輸出任意一個即可。

思路:

首先?n,m≤50,暴力枚舉每個炮臺的朝向是不可取的,但可以分別計算每個炮臺每種朝向能打到的格子。不難想到,能打到自己炮臺的炮臺朝向一定是不可取的。

題目又說道,要求每一處空地都會被至少一束激光打到,那我們需要看每一處空地可能被哪些炮臺的哪些朝向打到。

接下來是題目的核心,每一處空地最多可能被哪些炮臺的哪些可取的朝向打到呢??首先,空地不會被兩束同向的激光打到,因為若可能,它們必有重合的一段路徑,然而其中一束激光會被路徑上的炮臺或反射鏡所影響。

舉個例子:

-.\.
....
-.\.
....

此圖中對于空地?(4,3),若上方除?(3,1)?的激光打來,一定會碰到反射鏡而改變路徑。

其次,空地也不會被兩束反向的激光打到,因為光路可逆,若可能,一束激光能打到空地,則一定能沿著反向打到該空地的激光的逆光路,打到另一個炮臺。

還是通過舉例說明:

-..\
....
.-./

此圖中兩束激光均能打到?(2,4)?的空地,且方向相反,不難發現它們必能互相打到。

綜上所述,再根據抽屜原理,每個空地最多可能被兩個炮臺的可取的朝向打到。?又因為至少被一束激光打到。所以,每個空地均可被看作?i 炮臺為橫/豎向?或?j 炮臺為橫/豎向?的約束條件。 這里就能看出是 2-SAT 問題了,根據前文所述,用 2-SAT 模板的處理方法即可。

代碼細節

思路是簡明且重要的,并且代碼細節難度也不容小覷?(畢竟是黑題

認為自己的代碼實現思路是清晰且完備的,下面進行展示。

  • 遍歷激光的路徑使用 dfs,除了建立方向數組,再建立兩個數組?lf[],rf[]?代表每種激光方向經過副/主反射鏡后改變成的方向,這樣能簡便地處理反射的問題。
const int dx[]={0,-1,0,1},dy[]={-1,0,1,0},lf[]={3,2,1,0},rf[]={1,0,3,2};
...
bool dfs(int x,int y,int dir){int xx=x+dx[dir],yy=y+dy[dir];if(mp[xx][yy]=='#'||xx<1||xx>n||yy<1||yy>m)return 1;if(mp[xx][yy]=='/')return dfs(xx,yy,lf[dir]);if(mp[xx][yy]=='\\')return dfs(xx,yy,rf[dir]);if(mp[xx][yy]=='.')return dfs(xx,yy,dir);return 0;
}
  • 對于每個朝向,先 dfs 一次判斷是否能打到其他炮臺,能則說明朝向不合法,建一條選該朝向指向選另外朝向的邊,體現選該朝向就能推出矛盾。否則朝向合法,再進行一次 dfs 尋找能打到的空地。這里用?i?表示橫向,用?i+tcnt?表示豎向。
	for(int i=1;i<=tcnt;i++){if(!dfs(towx[i],towy[i],0)||!dfs(towx[i],towy[i],2))add(i,i+tcnt);else{dfs2(i,towx[i],towy[i],0);dfs2(i,towx[i],towy[i],2);}if(!dfs(towx[i],towy[i],1)||!dfs(towx[i],towy[i],3))add(i+tcnt,i);else{dfs2(i+tcnt,towx[i],towy[i],1);dfs2(i+tcnt,towx[i],towy[i],3);}}
  • 接下來遍歷每處空地建邊,注意分類討論:

  • 若空地不能被任何激光打到,直接無解。

  • 若空地只能被一個炮臺的朝向打到,建一條不選該朝向指向選該朝向的邊,體現該朝向不得不選。

  • 若空地能被兩個炮臺的朝向打到,就是常規的 2-SAT,分別建??i→j?和??j→i?的邊。

這里定義了一個函數?opp?表示相反朝向對應的編號。

		for(int i=1;i<=n;i++){for(int j=1;j<=m;j++){if(mp[i][j]=='.'){int sz=S[i][j].size();if(sz==0)add(1,tcnt+1),add(tcnt+1,1);if(sz==1)add(opp(S[i][j][0]),S[i][j][0]);if(sz==2){if(opp(S[i][j][0])!=S[i][j][1])add(opp(S[i][j][0]),S[i][j][1]),add(opp(S[i][j][1]),S[i][j][0]);}}}}

至此,本題的所有代碼難點都過去了,接下來正常地跑 Tarjan,通過強連通分量編號確定每個炮臺的朝向即可

?代碼
?

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<iostream>
#include<vector>
#define N 5010
#define M N<<1
using namespace std;
int T,n,m;
int low[N],dfn[N],scc[N],num,cnt,towx[N],towy[N],tcnt,flag;
int h[N],nxt[M],ver[M],tot,st[N],top;
const int dx[]={0,-1,0,1},dy[]={-1,0,1,0},lf[]={3,2,1,0},rf[]={1,0,3,2};
char mp[55][55];
vector<int>S[55][55];
void init(){num=cnt=tcnt=top=0,tot=-1,flag=1;memset(h,-1,sizeof(h)); memset(dfn,0,sizeof(dfn));memset(scc,0,sizeof(scc));for(int i=1;i<=n;i++){for(int j=1;j<=m;j++)S[i][j].clear();}
}
void add(int x,int y){ver[++tot]=y;nxt[tot]=h[x];h[x]=tot;
}
int opp(int x){return x<=tcnt?x+tcnt:x-tcnt;
}
bool dfs(int x,int y,int dir){int xx=x+dx[dir],yy=y+dy[dir];if(mp[xx][yy]=='#'||xx<1||xx>n||yy<1||yy>m)return 1;if(mp[xx][yy]=='/')return dfs(xx,yy,lf[dir]);if(mp[xx][yy]=='\\')return dfs(xx,yy,rf[dir]);if(mp[xx][yy]=='.')return dfs(xx,yy,dir);return 0;
}
void dfs2(int id,int x,int y,int dir){int xx=x+dx[dir],yy=y+dy[dir];if(mp[xx][yy]=='#'||xx<1||xx>n||yy<1||yy>m)return ;if(mp[xx][yy]=='/')dfs2(id,xx,yy,lf[dir]);if(mp[xx][yy]=='\\')dfs2(id,xx,yy,rf[dir]);if(mp[xx][yy]=='.')S[xx][yy].push_back(id),dfs2(id,xx,yy,dir);
}
void tarjan(int u){st[++top]=u;low[u]=dfn[u]=++num;for(int i=h[u];~i;i=nxt[i]){int v=ver[i];if(!dfn[v]){tarjan(v);low[u]=min(low[v],low[u]);}else if(!scc[v])low[u]=min(dfn[v],low[u]);}if(low[u]==dfn[u]){++cnt;while(1){int v=st[top--];scc[v]=cnt;if(v==u)break;}}
}
int main(){scanf("%d",&T);while(T--){scanf("%d%d",&n,&m);init();for(int i=1;i<=n;i++){scanf("%s",mp[i]+1);for(int j=1;j<=m;j++){if(mp[i][j]=='-'||mp[i][j]=='|')towx[++tcnt]=i,towy[tcnt]=j;}}for(int i=1;i<=tcnt;i++){if(!dfs(towx[i],towy[i],0)||!dfs(towx[i],towy[i],2))add(i,i+tcnt);else{dfs2(i,towx[i],towy[i],0);dfs2(i,towx[i],towy[i],2);}if(!dfs(towx[i],towy[i],1)||!dfs(towx[i],towy[i],3))add(i+tcnt,i);else{dfs2(i+tcnt,towx[i],towy[i],1);dfs2(i+tcnt,towx[i],towy[i],3);}}for(int i=1;i<=n;i++){for(int j=1;j<=m;j++){if(mp[i][j]=='.'){int sz=S[i][j].size();if(sz==0)add(1,tcnt+1),add(tcnt+1,1);if(sz==1)add(opp(S[i][j][0]),S[i][j][0]);if(sz==2){if(opp(S[i][j][0])!=S[i][j][1])add(opp(S[i][j][0]),S[i][j][1]),add(opp(S[i][j][1]),S[i][j][0]);}}}}for(int i=1;i<=2*tcnt;i++){if(!dfn[i])tarjan(i);}for(int i=1;i<=tcnt;i++){if(scc[i]==scc[i+tcnt]){printf("IMPOSSIBLE\n");flag=0;break;}}if(flag){printf("POSSIBLE\n");for(int i=1;i<=tcnt;i++){if(scc[i]>scc[i+tcnt])mp[towx[i]][towy[i]]='|';else mp[towx[i]][towy[i]]='-';}for(int i=1;i<=n;i++){printf("%s\n",mp[i]+1);}}}return 0;
}

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/903474.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/903474.shtml
英文地址,請注明出處:http://en.pswp.cn/news/903474.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

【android bluetooth 案例分析 03】【PTS 測試 】【PBAP/PCE/SSM/BV-02-C】

1. 測試介紹 PBAP/PCE/SSM/BV-02-C [PCE Closes a PBAP Session] 1. Test Purpose Verify that the PCE can terminate a PBAP session. 2. Initial Condition IUT: The IUT is engaged in a PBAP session with the Lower Tester.Lower Tester: The Lower Tester is engag…

ArcGIS:開啟洪水災害普查、評估與制圖新征程

技術點目錄 一、洪水普查技術規范解讀二、ArcGIS介紹及數據管理三、空間數據的轉換與處理四、洪水淹沒專題地圖制作五、矢量數據的采集與處理六、柵格數據的下載與處理七、ArcGIS水文分析八、ArcGIS洪水分析九、ArcGIS淹沒分析了解更多 ———————————————————…

【系統參數合法性校驗】spring-boot-starter-validation

JSR303校驗 統一校驗的需求 前端請求后端接口傳輸參數&#xff0c;是在controller中校驗還是在Service中校驗&#xff1f; 答案是都需要校驗&#xff0c;只是分工不同。 Contoller中校驗請求參數的合法性&#xff0c;包括&#xff1a;必填項校驗&#xff0c;數據格式校驗&…

[零基礎]內網ubuntu映射到云服務器上,http訪問(frp內網穿透)

阿里云服務器&#xff0c;高校教師可以半價&#xff0c; frp下載地址&#xff1a;https://github.com/fatedier/frp/releases&#xff0c;選amd64&#xff0c; 云服務器開放端口 選擇網絡與安全–>安全組->管理規則 配置開放端口&#xff0c;7000為支持frp開放的端口&…

第十六屆藍橋杯 2025 C/C++組 破解信息

目錄 題目&#xff1a; 題目描述&#xff1a; 題目鏈接&#xff1a; 思路&#xff1a; 思路詳解&#xff1a; 代碼&#xff1a; 代碼詳解&#xff1a; 題目&#xff1a; 題目描述&#xff1a; 題目鏈接&#xff1a; P12344 [藍橋杯 2025 省 B/Python B 第二場] 破解信息…

OpenAI Embedding 和密集檢索(如 BERT/DPR)進行語義相似度搜索有什么區別和聯系

OpenAI Embedding 和密集檢索&#xff08;如 BERT/DPR&#xff09;其實是“同一種思想的不同實現”&#xff0c;它們都屬于Dense Retrieval&#xff08;密集向量檢索&#xff09;&#xff0c;只不過使用的模型、部署方式和調用方式不同。 &#x1f9e0; 首先搞清楚&#xff1a;…

Linux電源管理(3)_關機和重啟的過程

原文&#xff1a;Linux電源管理&#xff08;3&#xff09;_Generic PM之重新啟動過程 1.前言 在使用計算機的過程中&#xff0c;關機和重啟是最先學會的兩個操作。同樣&#xff0c;這兩個操作在Linux中也存在&#xff0c;可以關機和重啟。這就是這里要描述的對象。在Linux Ke…

C# 繼承詳解

繼承是面向對象程序設計&#xff08;OOP&#xff09;中的核心概念之一&#xff0c;它極大地增強了代碼的重用性、擴展性和維護性。本篇文章將詳細講解C#中的繼承機制&#xff0c;包括基礎概念、語法特法、多重繼承&#xff08;通過接口實現&#xff09;、繼承的規則和實際應用示…

SQLAlchemy 2.x 異步查詢方法比較

SQLAlchemy 2.x 異步查詢中常用的 結果處理方法速查表&#xff0c;包含方法說明、使用場景、返回類型及典型用途。 SQLAlchemy 查詢結果處理方法速查表&#xff08;適用于 AsyncSession&#xff09; 方法 說明 返回類型 示例 SQL 示例輸出 scalars().all() 獲取單列所有…

極客天成參與”AI助力智慧城市構建”主題演講暨招商引智專題推介活動

4月7日下午&#xff0c;北京極客天成科技有限公司參加了天津市河東區數據局舉辦的“AI賦能智慧城市構建”主題演講暨招商引智專題推介活動。 活動中&#xff0c;華為&#xff08;天津&#xff09;有限公司數字政府解決方案總監姜華庚圍繞“政務大模型賦能智慧城市建設”&#x…

理解 EKS CloudWatch Pod CPU Utilization 指標:與 `kubectl top` 及節點 CPU 的關系

在使用 AWS EKS 時&#xff0c;CloudWatch Container Insights 提供了豐富的容器級別監控指標&#xff0c;幫助我們深入了解應用的運行狀態。如下截圖中的 ContainerInsights pod_cpu_utilization 指標就是一個非常重要的維度。本文將詳細解釋這個指標的含義&#xff0c;并將其…

使用pip3安裝軟件包報錯`externally-managed-environment`的幾種解決方式

1、pip3安裝軟件包報錯 報錯externally-managed-environment的原因&#xff1a; 從 Python 3.11 開始引入了 PEP 668 規范&#xff0c;該規范限制了在系統級 Python 環境中使用 pip 安裝第三方包&#xff0c;以避免與系統包管理器&#xff08;如 apt&#xff09;產生沖突。 如…

spring security用戶退出

Spring security默認實現了用戶退出的功能&#xff0c;用戶退出主要考慮退出后會話如何管理以及跳轉到哪個頁面。HttpSecurity類提供了logout()方法開啟退出登錄的支持&#xff0c;默認觸發用戶退出操作的URL為“/logout”&#xff0c;用戶退出時同時也會清除Session等默認用戶…

愛普生SG2520HHN晶振數據中心服務器的理想解決方案

在當今數字化時代&#xff0c;數據中心作為海量數據存儲、處理與傳輸的核心樞紐&#xff0c;其服務器的高效穩定運行至關重要。服務器作為其核心設備&#xff0c;對時鐘信號的精度和穩定性提出了嚴苛要求——微小的時序誤差可能導致數據傳輸失敗或系統宕機。愛普生 SG2520HHN 差…

LeetCode 155題解 | 最小棧

最小棧 一、題目鏈接二、題目三、算法原理思路1&#xff1a;用一個變量存儲最小元素思路2&#xff1a;雙棧普通棧和最小棧 四、編寫代碼五、時間復雜度 一、題目鏈接 最小棧 二、題目 三、算法原理 棧用數組、鏈表實現都行&#xff0c;最主要的就是在能在常數時間內檢索到最…

es+kibana---集群部署

其實一般es要跑3個節點的&#xff0c;這樣才能做高可用&#xff0c;處理并發大&#xff0c;但是我這里只是一個pod mkdir -p /stroe/data/es es搭建&#xff1a; #【拉取鏡像】 #docker pull elasticsearch:6.8.7 #docker pull busybox:1.28 【導入鏡像】 docker load -i es.…

WPF常用技巧匯總 - Part 2

WPF常用技巧匯總-CSDN博客 主要用于記錄工作中發現的一些問題和常見的解決方法。 目錄 WPF常用技巧匯總-CSDN博客 1. DataGrid Tooltip - Multiple 2. DataGrid Tooltip - Cell值和ToolTip值一樣 3. DataGrid Tooltip - Cell值和ToolTip值不一樣 4. DataGrid - Ctrl A /…

uniapp打包apk如何實現版本更新

我們做的比較簡單&#xff0c;在后端設置版本號&#xff0c;并在uniapp的config.js中定義版本號&#xff0c;每次跟后端的進行對比&#xff0c;不一致的話就更新。 一、下載apk 主要代碼&#xff08;下載安裝包&#xff0c;并進行安裝&#xff0c;一般得手動同意安裝&#xf…

局部和整體的關系

Prompt&#xff1a;為什么要研究局部和整體的關系呢&#xff1f;是因為很多情況下&#xff0c;局部就能表達出整體&#xff1f; 這是一個非常本質的問題&#xff0c;其實你已經接近數學和物理中“幾何本質”的核心了。我們研究局部和整體的關系&#xff0c;是因為&#xff1a;…

企業辦公協同平臺安全一體化生態入住技術架構與接口標準分析報告

全球組織數字化與智能化背景下 企業辦公協同平臺安全一體化生態入住技術架構與接口標準分析報告 一、背景與市場需求 市場規模與增量 根據Statista數據&#xff0c;全球協同辦公平臺市場規模預計從2023年的$480億增長至2027年的$900億&#xff0c;年復合增長率&#xff08;CAG…