UVA - 11916 Emoogle Grid

题意:

给定 m × n m × n m×n 的 方格,其中有 b b b 个障碍物,除了障碍物以外的格子均需被染成 k k k 种颜色中的一种,且要求 ( x , y ) , ( x + 1 , y ) (x,y),(x+1,y) (x,y),(x+1,y) 不能染相同颜色,染色方案数模 1 e 8 + 7 1e8+7 1e8+7 r r r。现在给定 n , k , b , r n, k, b, r n,k,b,r,求最小的 m m m ( n , m ≤ 1 0 8 , 2 ≤ k ≤ 1 0 8 , b ≤ 500 ) (n, m \leq 10^8, 2 \leq k \leq 10^8, b \leq 500) (n,m108,2k108,b500)

链接:

https://vjudge.net/problem/UVA-11916

解题思路:

若没有障碍物,则除了第一行每个格子染 k k k 种外,其他格子均只有 k − 1 k-1 k1 种。有障碍物 ( x , y ) (x, y) (x,y) ( x + 1 , y ) (x + 1, y) (x+1,y) k k k 种染色可能。计算出能染 k k k k − 1 k-1 k1 的格子数量以后,问题转化为求 a x ≡ b   m o d   p a^x \equiv b~mod~p axb mod p 的最小的 x x x,用 B S G S BSGS BSGS 算法求解。

参考代码:
#include<bits/stdc++.h>
 
using namespace std;
typedef long long ll;
typedef pair<int, int> pii;
#define sz(a) ((int)a.size())
#define pb push_back
#define lson (rt << 1)
#define rson (rt << 1 | 1)
#define gmid (l + r >> 1)
const int maxn = 3e5 + 5;
const int inf = 0x3f3f3f3f;
const int mod = 1e8 + 7;

map<int, int> mp;
set<pii> st;
vector<pii> G;
int n, k, b, r, mx;

ll qpow(ll a, ll b){

    ll ret = 1;
    while(b){

        if(b & 1) ret = ret * a % mod;
        a = a * a % mod;
        b >>= 1;
    }
    return ret;
}

ll inv(ll a){

    return qpow(a, mod - 2);
}

ll BSGS(ll a, ll b){

    ll lim = sqrt(mod + 0.5);
    ll cur = 1, inva = 1;
    for(int i = 0; i < lim; ++i){

        if(!mp.count(cur)) mp[cur] = i;
        cur = cur * a % mod, inva = inva * a % mod;
    }
    inva = inv(inva);
    for(int i = 0; i <= lim; ++i){

        if(mp.count(b)) return i * lim + mp[b];
        b = b * inva % mod;
    }
    return -1;
}

int solve(){

    ll cntK = 0, cntK_1 = 0, cntB = 0, cntT = 0;
    for(auto &it : G){

        if(it.first == 1) ++cntT;
        if(it.first == mx) ++cntB;
        else if(st.find({it.first + 1, it.second}) == st.end()) ++cntK;
    }
    cntK += n - cntT;
    cntK_1 = n * 1ll * mx - b - cntK;
    ll ret = qpow(k, cntK) * qpow(k - 1, cntK_1) % mod;
    if(ret == r) return mx;
    ret = ret * qpow(k, cntB) % mod * qpow(k - 1, n - cntB) % mod;
    if(ret == r) return mx + 1;
    return BSGS(qpow(k - 1, n), r * inv(ret) % mod) + mx + 1;
}

int main(){
 
    // ios::sync_with_stdio(0); cin.tie(0);
    int t, cas = 0; scanf("%d", &t);
    while(t--){

        scanf("%d%d%d%d", &n, &k, &b, &r);
        G.clear(), st.clear(), mp.clear();
        mx = 1;
        for(int i = 1; i <= b; ++i){

            int x, y; scanf("%d%d", &x, &y);
            G.pb({x, y}), mx = max(mx, x);
            st.insert({x, y});
        }
        int ret = solve();
        printf("Case %d: %d\n", ++cas, ret);
    }
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值