zoukankan      html  css  js  c++  java
  • ZOJ 3644 Kitty's Game(数论+DP)

    Description

    Kitty is a little cat. She is crazy about a game recently.

    There arenscenes in the game(mark from 1 ton). Each scene has a numberpi. Kitty's score will become least_common_multiple(x,pi) when Kitty enter theithscene.xis the score that Kitty had previous. Notice that Kitty will become mad If she go to another scene but the score didn't change.

    Kitty is staying in the first scene now(withp1score). Please find out how many paths which can arrive at thenthscene and haskscores at there. Of course, you can't make Kitty mad.

    We regard two paths different if and only if the edge sequence is different.

    Input

    There are multiple test cases. For each test case:

    The first line contains three integern(2 ≤n≤ 2000),m(2 ≤m≤ 20000),k(2 ≤k≤ 106). Then followed bymlines. Each line contains two integeru,v(1 ≤u,v≤ n, u ≠ v) indicate we can go tovthscene fromuthscene directly. The last line of each case contains n integerpi(1 ≤pi≤ 106).

    Process to the end of input.

    Output

    One line for each case. The number of paths module 1000000007.

    题目大意:有n个点m条无向边,每个点有一个分值p[i],设x为当前分值,每经过一个点,分值就会变成LCM(x, p[i]),若经过一个点的时候分值没有变化,那么这个点不能走。现在要从点1走到点n,要得到k的分值,问有多少种方法。

    思路:首先走的每一步都必须是k的约数(不然到达终点的时候不可能得到k),那么走到每个点上至多有sum{k的约数}的分值,把k离散化,k最多有2*sqrt(k)个约数,即2000个。然后开始DP,dp[i][j]表示走到第i个点,得到分值j(离散值),并且能走到终点的方案数。然后记忆化搜索即可。原图有环也无所谓,因为分值每走一步必须要变化,而且只会越来越大,不会走出环来。

    代码(120MS):

     1 #include <cstdio>
     2 #include <cstring>
     3 #include <iostream>
     4 #include <map>
     5 using namespace std;
     6 
     7 typedef long long LL;
     8 
     9 const int MAXN = 2010;
    10 const int MAXE = 20010;
    11 const int MOD = 1000000007;
    12 
    13 map<int, int> mp;
    14 
    15 int n, m, k, ecnt;
    16 int head[MAXN];
    17 int to[MAXE], next[MAXE];
    18 int dp[MAXN][MAXN], val[MAXN];
    19 
    20 void init() {
    21     memset(head, 0, sizeof(head));
    22     ecnt = 1;
    23 }
    24 
    25 inline void add_edge(int u, int v) {
    26     to[ecnt] = v; next[ecnt] = head[u]; head[u] = ecnt++;
    27 }
    28 
    29 LL gcd(LL a, LL b) {
    30     return b ? gcd(b, a % b) : a;
    31 }
    32 
    33 LL lcm(LL a, LL b) {
    34     return a * b / gcd(a, b);
    35 }
    36 
    37 inline void get_app() {
    38     mp.clear();
    39     int cnt = 0;
    40     for(int i = 1; i * i <= k; ++i) {
    41         if(k % i != 0) continue;
    42         mp[i] = ++cnt;
    43         if(i * i != k) mp[k / i] = ++cnt;
    44     }
    45 }
    46 
    47 int dfs(int u, LL x) {
    48     if(u == n && x == k) return 1;
    49     int now = mp[x];
    50     if(dp[u][now] != -1) return dp[u][now];
    51     dp[u][now] = 0;
    52     for(int p = head[u]; p; p = next[p]) {
    53         int &v = to[p];
    54         if(k % val[v] != 0) continue;
    55         LL tmp = lcm(x, val[v]);
    56         if(tmp == x || tmp > k) continue;
    57         dp[u][now] = (dp[u][now] + dfs(v, tmp)) % MOD;
    58     }
    59     return dp[u][now];
    60 }
    61 
    62 int main() {
    63     while(scanf("%d%d%d", &n, &m, &k) != EOF) {
    64         init();
    65         while(m--) {
    66             int u, v;
    67             scanf("%d%d", &u, &v);
    68             add_edge(u, v);
    69         }
    70         for(int i = 1; i <= n; ++i) scanf("%d", &val[i]);
    71         get_app();
    72         memset(dp, 255, sizeof(dp));
    73         int ans = (k % val[1]) ? 0 : dfs(1, val[1]);
    74         cout<<ans<<endl;
    75     }
    76 }
    View Code
  • 相关阅读:
    JS 获取浏览器窗口大小方面的
    可以直接拿来用的15个jQuery代码片段
    JS 键盘方面的
    JAVA 实现DES MD5加密
    Spring SpringMvc Hibernate整合
    easyUI comboselector使用
    页面中嵌套html代码显示
    在IntelliJ上操作GitHub
    Maven项目配置第三方jar包
    JSP自定义标签
  • 原文地址:https://www.cnblogs.com/oyking/p/3255576.html
Copyright © 2011-2022 走看看