博弈论nim^|sg函数|涂色dp

acwing过过一遍,不用就会淡忘,好消息是再看一眼就能想起来了😇

lc1908

nim游戏:把所有堆的数量异或,结果非零则当前玩家能赢

非零先手玩家只用将其变为0,然后镜像后手玩家操作,后手必败

class Solution {

public:

bool nimGame(vector<int>& piles) {

int t = 0;

for(int p : piles){

t ^= p;

}

++return t != 0;++

}

};

下面三题都是

枚举出规则 + dp/memo统计的思路

lc1931

三进制数表示m列网格的单行合法涂色方案,筛选出相邻列颜色不同的方案

memo统计n行相邻行同列颜色不同的总合法涂色数

class Solution {

public:

int f[1005][255];

int mod = 1e9 + 7, M;

bool check(int S) {

int last = -1;

for(int i = 0; i < M; ++i){

if(S%3==last)return false;

last = S%3;

S /= 3;

}

return true;

}

bool check_n(int x, int y) {

for(int i = 0; i < M; ++i) {

if(x%3==y%3)return false;

x/=3,y/=3;

}

return true;

}

int colorTheGrid(int m, int n) {

M = m;

int tot = 1;

for(int i = 1; i <= m; ++i)tot*=3;

for(int i = 0; i < tot; ++i)

if(check(i))f[1][i] = 1;

for(int i = 2; i <= n; ++i)

for(int j = 0; j < tot; ++j)

if(check(j))

for(int k = 0;k < tot; ++k)

if(check(k)) {

if(!check_n(j,k))continue;

f[i][j] = (f[i][j] + f[i - 1][k]) % mod;

}

int ans = 0;

for(int i = 0; i < tot; ++i)

ans = (ans + f[n][i]) % mod;

return ans;

}

};

lc2184

++dfs构造+check+dp memo计算++

DFS枚举所有单行长宽符合要求的砌砖间隙排列

dp统计高度达标且++相邻行间隙不重合++的合法砌墙方案总数

class Solution {

typedef long long ll;

const int mod=1e9+7;

public:

vector<vector<int>> vec;//所有排列

bool check(int i, int j){

// 判断第i种排列与第j种排列 能否相邻

auto& v1 = vec[i];

auto& v2 = vec[j];

int x = 1, y = 1;

while(x < v1.size() - 1 && y < v2.size() - 1){

if(v1[x] == v2[y]) return false;

if(v1[x] < v2[y]) x++;

else y++;

}

return true;

}

int buildWall(int height, int width, vector<int>& bricks) {

++vector<int> cur{0};//back init++

int n=bricks.size();

auto dfs=[&](this auto&& dfs,int need)

{

if(need == 0){

vec.push_back(cur);

return;

}

for(int i = 0; i < n; ++i)

{

if(need >= bricks[i]){

cur.push_back(++cur.back()+bricks[i]);++

++dfs(need - bricks[i]++ );

cur.pop_back();

}

}

};

dfs(width); //排列保存间隙的横坐标位置

int m = vec.size();

vector<vector<ll>> dp(height, vector<ll>(m, 0));

//dp 行排列组合种数

vector<vector<int>> last(m); //能与第i种排列 相邻的排列

for(int i = 0; i < m; ++i){

if(check(i,i))

++last[i].push_back(i);
//单块砖排列可重复用
++

for(int j = i + 1; j < m; ++j){

if(check(i, j)){

last[i].push_back(j);

last[j].push_back(i);

}

}

}

for(int i = 0; i < m; ++i)

dp[0][i] = 1; // init

for(int i = 1; i < height; ++i){

for(int j = 0; j < m; ++j){

for(int k: last[j]){

++dp[i][j] += dp[i - 1][k++ ];

}

dp[i][j] %= mod;

}

}

ll ret = 0;

for(auto ev: dp.back()) ret += ev;

ret %= mod;

return (int)ret;

}

};

相关推荐
l1t1 天前
郭其先生利用DeepSeek实现的PostgreSQL递归CTE实现DFS写法
sql·算法·postgresql·深度优先
橘颂TA1 天前
【剑斩OFFER】算法的暴力美学——力扣 227 题:基本计算机Ⅱ
c++·算法·leetcode·职场和发展·结构于算法
信奥卷王1 天前
2025年12月GESPC++二级真题解析(含视频)
算法
Z1Jxxx1 天前
输入n个数进行排序,要求先按奇偶后按从小到大的顺序排序
数据结构·算法
乐迪信息1 天前
乐迪信息:船体AI烟火检测,24小时火灾自动预警
人工智能·物联网·算法·目标检测·语音识别
Z1Jxxx1 天前
整除整除整除
开发语言·c++·算法
Swift社区1 天前
LeetCode 471 编码最短长度的字符串
算法·leetcode·职场和发展
没有天赋那就反复1 天前
JAVA length
java·开发语言·算法
Tisfy1 天前
LeetCode 0712.两个字符串的最小ASCII删除和:反向思维保留最大(动态规划)
算法·leetcode·动态规划·字符串·dp·子序列
ohoy1 天前
RedisTemplate 使用之Hash
redis·算法·哈希算法