zoukankan      html  css  js  c++  java
  • 20140705 总结

    说好的的赛区难度... →_→ “不完全是赛区难度”==“完全不是赛区难度”

    看完了题做好了爆0的准备,从testC开始写,结果有30,正解是dp...

    testB我是在最后十分钟才写的,感觉像最长公共子序列,但是时间不够了,我就直接printf("%d ", s/e);结果有80...

    testA写了个Floyd,然后就开始乱搞了,结果就wa0...

     -----------------------------------------------------------------------------

    testA

    输入文件: testA.in  输出文件: testA.out 时限: 2000ms

     

    问题描述:

             有一个城市拥有N个节点,被M条有权无向路径连接。现在你要在一个地方(可以在路径上当然也可以在节点上)开设一家咖啡馆,使得所有节点达到这个咖啡馆的最短路里面最大值最小(也就是说离咖啡馆最远的节点距离尽可能得小),求出这个最大值的最小值。

     

    输入描述:

    第一行N和M。

    第2至M+1行,每行三个整数U,V,W。表示从U到V有一条权值为W的无向边。

     

    输出描述:

    一行一个数表示答案。 四舍五入至2位小数

    数据范围 N<=200 , W<=100000 , M<=19900

     

    样例输入:

    3 2
    1 2 100
    2 3 1

     

    样例输出:

    50.50

     

    题解:

    二分答案L

    对于一个L,枚举每一条边(x, y, w),只要有一条边符合情况,L合法

    对于一条边,枚举每一个点k:(1)dis[x][k]>L && dis[y][k]>L ->这条边不和发

                   l = L-dis[x][k], r = L-dis[y][k]

                  (2) l>=w || r>=w || r+l>=w ->这个点对于这条边合法

                  (3) 求一个区间, 既对于点k在这条直线上标出不合法的安放区间

                                            -> 求并集U:

                         若U==[0,w]->此边不合法

                       else ->合法   

     1 #define NOMBRE "testA"
     2 #include <cstdio>
     3 #include <vector>
     4 #include <cstring>
     5 #include <algorithm>
     6 using namespace std;
     7 
     8 const int INF = 0x3f3f3f3f;
     9 const int MAXN = 200+10;
    10 
    11 double Pri;
    12 int N, M, dis[MAXN][MAXN];
    13 
    14 struct Edge{
    15     int to, d;
    16 };
    17 
    18 vector< pair<int, int> > cj;
    19 vector<Edge> e[MAXN];
    20 
    21 inline Edge MakeEdge(int to, int d){
    22     Edge ret;
    23     ret.to = to, ret.d = d;
    24     return ret;
    25 }
    26 
    27 inline void Floyd(){
    28     for (int i=1; i<=N; i++)
    29         dis[i][i] = 0;
    30     for (int k=1; k<=N; k++)
    31         for (int i=1; i<=N; i++){
    32             if (dis[i][k]==INF) continue;
    33             for (int j=1; j<=N; j++)
    34                 dis[i][j] = min(dis[i][j], dis[i][k]+dis[k][j]);
    35 
    36         }
    37 }
    38 
    39 inline bool OkayEdge(int L, int x, int y, int w){
    40     int dl, dr, Tail = 0, CJs;
    41     cj.clear();
    42     for (int k=1; k<=N; k++){
    43         dl = L-dis[x][k], dr = L-dis[y][k];
    44         if (dl<0 && dr<0) return false;
    45         if (dl>=w || dr>=w || dl+dr>=w) continue;
    46         cj.push_back(make_pair(dl+1, w-dr-1));
    47     }
    48     if (cj.empty()) return true;
    49     sort(cj.begin(), cj.end());
    50     CJs = cj.size();
    51     for (int i=0; i<CJs; i++){
    52         if (Tail<cj[i].first) return true;
    53         if (Tail<cj[i].second+1) Tail = cj[i].second+1;
    54     }
    55     return false;
    56 }
    57 
    58 inline bool Okay(int L){
    59     int Es;
    60     for (int i=1; i<=N; i++){
    61         Es = e[i].size();
    62         for (int j=0; j<Es; j++)
    63             if (OkayEdge(L, i, e[i][j].to, e[i][j].d)) return true;
    64     }
    65     return false;
    66 }
    67 
    68 int main(){
    69     freopen(NOMBRE ".in", "r", stdin);
    70     freopen(NOMBRE ".out", "w", stdout);
    71 
    72     memset(dis, 0x3f, sizeof(dis));
    73     scanf("%d %d", &N, &M);
    74     int u, v, d;
    75     for (int i=0; i<M; i++)
    76         scanf("%d %d %d", &u, &v, &d),
    77         d <<= 1, e[u].push_back(MakeEdge(v, d)),
    78         dis[u][v] = dis[v][u] = min(dis[u][v], d);
    79     Floyd();
    80     int l = 0, r = INF, Mid;
    81     while (l<=r){
    82         Mid = (l+r)>>1;
    83         if (Okay(Mid)) r = Mid-1, Pri = Mid;
    84         else l = Mid+1;
    85     }
    86     printf("%.2lf", Pri/=2.0);
    87 }

     

     -----------------------------------------------------------------------------

    testB

    输入文件: testB.in  输出文件testB.out 时限2000ms

     

    问题描述:

    方师傅有两个由数字组成的串 a1,a2,⋯,an 和 b1,b2,⋯,bm。 有一天,方师傅感到十分

    无聊因此他决定用这两个串来玩玩游戏。游戏规则十分简单,方师傅会进行一些操作,每个操作可能是以下两种操作之一:

     

    1.从a串选择一个a的非空前缀,再从b串选一个b的非空前缀。这两个前缀的最后一个元素必须相等,完成选择后把这两个前缀删除。

     

    2.删除两个串所有的元素。

     

    第一种操作会耗费e的能量值,并为方师傅增加一美分到他的电子账户中。第二种操作会耗费两个串的不完整度的能量。不完整度 = 两个串已经被删除的元素的数目。只有执行第二种操作后,方师傅才能从电子帐户中取出他的钱。 

    刚开始时,方师傅有一个空的电子账户和s的能量,请问方师傅最多可以赚多少美分?注意,由于乐警官偷吃光了方师傅的士力架,导致方师傅无法补充能量,因此方师傅的能量任何时候都不能小于0。

     

    输入描述: 

    第一行4个整数,n,m,s,e(1≤n,m≤10^5;1≤s≤3×10^5;10^3≤e≤10^4)。

    第二行n个整数,a1,a2⋯an.

    第三行m个整数,b1,b2⋯bm.

    1≤ai,bi≤10^5

     

    输出描述:

    输出一个整数,方师傅可以最多赚得的美分数目。

     

    样例输入1:

    5 5 100000 1000

    1 2 3 4 5

    3 2 4 5 1

     

    样例输出1:

    3
     

    样例输入2:

    3 4 3006 1000
    1 2 3
    1 2 4 3

     

    样例输出2:

    2

     

    dp[i][j]表示A匹配到i位置获得j美分是匹配到B的位置

     //看来标程才想到vector...多么美好vector啊

     1 #define NOMBRE "testB"
     2 #include <vector>
     3 #include <cstdio>
     4 #include <cstring>
     5 #include <algorithm>
     6 using namespace std;
     7 
     8 const int MAXN = 1e5+10;
     9 const int MAXJ = 300+10;
    10 
    11 int n, m, s, e, x, Best, Pri, a[MAXN], b[MAXN], dp[MAXN][MAXJ];
    12 vector<int> v[MAXN];
    13 
    14 int main(){
    15     freopen(NOMBRE ".in", "r", stdin);
    16     freopen(NOMBRE ".out", "w", stdout);
    17 
    18     memset(dp, 0x3f, sizeof(dp));
    19     scanf("%d %d %d %d", &n, &m, &s, &e), Best = s/e;
    20     for (int i=1; i<=n; i++)
    21         scanf("%d", &a[i]);
    22     for (int i=1; i<=m; i++)
    23         scanf("%d", &b[i]),
    24         v[b[i]].push_back(i);
    25     
    26     int temp;
    27     dp[0][0] = 0;
    28     for (int i=1; i<=n; i++){
    29         dp[i][0] = 0;
    30         for (int j=0; j<=Best; j++){
    31             dp[i][j] = dp[i-1][j];
    32             x = upper_bound(v[a[i]].begin(), v[a[i]].end(), dp[i-1][j-1])-v[a[i]].begin();
    33             temp = v[a[i]].size();
    34             if (x<temp) dp[i][j] = min(dp[i][j], v[a[i]][x]);
    35             if (j>Pri && i+dp[i][j]+j*e<=s) Pri = j; 
    36         }
    37     }    
    38     printf("%d
    ", Pri);
    39 }

     

    -----------------------------------------------------------------------------

    testC 

    输入文件: testC.in  输出文件testC.out 时限1000ms 

     

    问题描述:

    给你一组数,a1, a2a3,⋯,an。令:G=gcd(a1, a2a3,⋯,an)。

    现在从中任意删除一些数字,设剩下的数为:a1, a2a3,⋯,am。再令:g=gcd(a1, a2a3,⋯,am)

    现要求G=g,问最多能删除多少数?

     

    输入描述:

    第一行一个数n,第二行n个数a1,a2,a3,⋯,an

    1≤n≤700

    1≤ai≤10000

     

    输出描述:

    输出只有一个数,表示最多能删除多少数。

     

    样例输入:

    3
    4 6 8

    样例输出

    1

     

    题解:

    dp[i]表示使得gcd为i最少使用多少数

     1 #define NOMBRE "testc"
     2 #include <cstdio>
     3 #include <cstring>
     4 #include <algorithm>
     5 using namespace std;
     6 
     7 const int N = 10000; 
     8 const int MAXN = 700+10;
     9 
    10 int n, g, dp[N+10], a[MAXN], max;
    11 
    12 int gcd(int a, int b){
    13     if (!a && b) return b;
    14     if (!b && a) return a;
    15     while (b^=a^=b^=a%=b);
    16     return a;
    17 }
    18 
    19 int main(){
    20     freopen(NOMBRE ".in", "r", stdin);
    21     freopen(NOMBRE ".out", "w", stdout);
    22 
    23     memset(dp, 0x3f, sizeof(dp));
    24     scanf("%d", &n), dp[0] = Max = g = 0;
    25     for (int i=0; i<n; i++)
    26         scanf("%d", &a[i]), g = gcd(g, a[i]), Max = max(Max, a[i]);
    27     for (int i=0; i<n; i++)
    28         for (int j=0; j<=Max; j++)
    29             dp[gcd(a[i], j)] = min(dp[gcd(a[i], j)], dp[j]+1);
    30     printf("%d
    ", n-dp[g]);
    31 }

     

  • 相关阅读:
    AtCoder Beginner Contest 162 C~F
    题解 | 【CF896B】 Ithea Plays With Chtholly
    C# | C#快速入门
    Codeforces Round #618 (Div. 2) A~E
    Educational Codeforces Round 92 (Rated for Div. 2) A~C
    使用 Python 参与算法竞赛
    【网络爬虫学习】实战,爬取网页以及贴吧数据
    国内pip源提示“not a trusted or secure host”解决方案
    【网络爬虫学习】第一个Python爬虫程序 & 编码与解码详解 & Pythonの实现
    【网络爬虫学习】网页的基本构成
  • 原文地址:https://www.cnblogs.com/cjhahaha/p/3827026.html
Copyright © 2011-2022 走看看