zoukankan      html  css  js  c++  java
  • Uva_11916 Emoogle Grid

    题目链接

    题意:

      有个N X M的棋盘, 有K种颜色, 有B个不可涂色的位置, 共有R种涂色方案。

      1)每个可涂色的位置必须涂上一种颜色

      2)不可涂色位置不能涂色

      3)每个位置必须从K种颜色中选出一种颜色进行涂色

      4)当前格子(x,y) 上面的那个格子(x+1,y)不能同色

      

      现在已知N, K, B, R, 求满足条件的最小的M

      

    思路:

      B个不可涂色位置设为(x1, y1), (x2, y2), (x3, y3), ... , (xb, yb)

      1)M必然 ≥ max(x[i])

      2)设前max(x[i]) 行 与 max(x[i]) + 1 行涂色方案为 cnt, 则 max(x[i]) + 1之后的每一行, 涂色方案都是 (K-1)^N。

      3)设P = (K - 1) ^ N

      存在一个j 使得:

              cnt * P^j = R

      右乘cnt的逆元 cnt ^ -1 得:

              P^j = R * cnt ^ -1

      即求一个最小的j满足上式。

              P^j ≡ R * cnt ^ -1 (mod MOD) , MOD = 100000007

      利用对数取余进行求解

      

      代码如下:

      

      1 #include <cmath>
      2 #include <cstdio>
      3 #include <cstring>
      4 #include <cstdlib>
      5 #include <ctime>
      6 #include <set>
      7 #include <map>
      8 #include <list>
      9 #include <queue>
     10 #include <string>
     11 #include <vector>
     12 #include <fstream>
     13 #include <iterator>
     14 #include <iostream>
     15 using namespace std;
     16 #define LL long long
     17 #define MAXN 550
     18 #define MOD 100000007
     19 int n, b, r, k, m;
     20 int x[MAXN], y[MAXN];
     21 set<pair<int, int> > best;
     22 
     23 void ex_gcd(LL a, LL b, LL& d, LL& x, LL& y)
     24 {
     25     if(!b) {d = a; x = 1; y = 0;}
     26     else 
     27     {ex_gcd(b, a%b, d, y, x); y -= x*(a/b);}
     28 }
     29 
     30 LL inv(LL a)
     31 {
     32     LL d, x, y;
     33     ex_gcd(a, MOD, d, x, y);
     34     return d == 1?(x + MOD) % MOD : -1;
     35 }
     36 
     37 LL mul_mod(LL a, LL b)
     38 {
     39     return a * b % MOD;
     40 }
     41 
     42 LL pow_mod(LL a, LL p)
     43 {
     44     if(p == 0) return 1;
     45     LL ans = pow_mod(a, p/2);
     46     ans = ans * ans % MOD;
     47     if(p % 2 == 1) ans = ans * a % MOD;
     48     return ans;
     49 }
     50 
     51 int log_mod(int a, int b) 
     52 {
     53     int m, v, e = 1, i;
     54     m = (int)sqrt(MOD+0.5);
     55     v = inv(pow_mod(a, m));
     56     map<int, int> x;
     57     x[1] = 0;
     58     for (int i = 1; i < m; i++) 
     59     {
     60         e = mul_mod(e, a);
     61         if (!x.count(e))
     62             x[e] = i;
     63     }
     64     for (int i = 0; i < m; i++) 
     65     {
     66         if (x.count(b))
     67             return i*m + x[b];
     68         b = mul_mod(b, v);
     69     }
     70     return -1;
     71 }
     72 
     73 int solve()
     74 {
     75     int cur = 0;
     76     int cnt = 0;
     77 
     78     for(int i = 0; i < b; i ++)
     79         if(x[i] != m && !best.count(make_pair(x[i] + 1, y[i]))) cur ++;
     80     cur += n;
     81     for(int i = 0; i < b; i ++)
     82         if(x[i] == 1) cur --;
     83 
     84     // ans = k^cur * (k-1) ^ (n*m - b - cur);
     85     cnt = mul_mod(pow_mod(k, cur), pow_mod(k - 1, (LL)n * m - b - cur));
     86 
     87     if(cnt == r) return m;
     88 
     89     cur = 0;
     90     for(int i = 0; i < b; i ++)
     91         if(x[i] == m) cur ++;
     92 
     93     //ans = cnt * k^cur * (k - 1)^(n - cur);
     94     cnt = mul_mod(cnt, pow_mod(k, cur));
     95     cnt = mul_mod(cnt, pow_mod(k - 1, n - cur));
     96     m ++;
     97 
     98     if(cnt == r) return m;
     99 
    100     //printf("%d %d %d
    ", pow_mod(k - 1, n), inv(cnt), log_mod(pow_mod(k - 1, n), mul_mod(r, inv(cnt))));
    101     // P = (k - 1) ^ n,  P^x = r * cnt^-1;
    102     return log_mod(pow_mod(k - 1, n), mul_mod(r, inv(cnt))) + m;
    103 }
    104 
    105 int main()
    106 {
    107     int T;
    108     scanf("%d", &T);
    109     for(int kcase = 1; kcase <= T; kcase ++)
    110     {
    111         scanf("%d %d %d %d", &n, &k, &b, &r);
    112         best.clear();
    113         m = 1;
    114         for(int i = 0; i < b; i ++)
    115         {
    116             scanf("%d %d",&x[i], & y[i]);
    117             m = max(m, x[i]);
    118             best.insert(make_pair(x[i], y[i]));
    119         }
    120         printf("Case %d: %d
    ", kcase, solve());
    121     }
    122     return 0;
    123 }
    View Code

      这题WA了一天, 反复找错都没找到, 重写第N + 1次的时候终于找到错误 , 求逆函数写错了 囧

  • 相关阅读:
    ubuntu 14.04搭建PHP项目基本流程
    linux下 lvm 磁盘扩容
    LVM基本介绍与常用命令
    Linux LVM逻辑卷配置过程详解
    mysql 5.7中的用户权限分配相关解读!
    linux系统维护时的一些小技巧,包括系统挂载新磁盘的方法!可收藏!
    linux系统内存爆满的解决办法!~
    源、更新源时容易出现的问题解决方法
    NV显卡Ubuntu14.04更新软件导致登录死循环,不过可以进入tty模式
    一些要注意的地方
  • 原文地址:https://www.cnblogs.com/By-ruoyu/p/4665470.html
Copyright © 2011-2022 走看看