Codeforces 483 - A/B/C/D/E - (Done)

題目連接:http://codeforces.com/contest/483
node


A - Counterexample - [簡單構造題]

Your friend has recently learned about coprime numbers. A pair of numbers $(a, b)$ is called coprime if the maximum number that divides both $a$ and $b$ is equal to one.ios

Your friend often comes up with different statements. He has recently supposed that if the pair $(a, b)$ is coprime and the pair $(b, c)$ is coprime, then the pair $(a, c)$ is coprime.c++

You want to find a counterexample for your friend's statement. Therefore, your task is to find three distinct numbers $(a, b, c)$, for which the statement is false, and the numbers meet the condition $l \le a < b < c \le r$.數組

More specifically, you need to find three numbers $(a, b, c)$, such that $l \le a < b < c le r$, pairs $(a, b)$ and $(b, c)$ are coprime, and pair $(a, c)$ is not coprime.dom

Input
The single line contains two positive space-separated integers $l, r (1 \le l \le r \le 10^18; r - l \le 50)$.ide

Output
Print three positive space-separated integers $a, b, c$ — three distinct numbers $(a, b, c)$ that form the counterexample. If there are several solutions, you are allowed to print any of them. The numbers must be printed in ascending order.ui

If the counterexample does not exist, print the single number $-1$.this

Examples
Input
2 4
Output
2 3 4
Input
10 11
Output
-1
Input
900000000000000009 900000000000000029
Output
900000000000000009 900000000000000010 900000000000000021
Note
In the first sample pair (2, 4) is not coprime and pairs (2, 3) and (3, 4) are.spa

In the second sample you cannot form a group of three distinct integers, so the answer is -1.rest

In the third sample it is easy to see that numbers 900000000000000009 and 900000000000000021 are divisible by three.

題意:

在給定的 $[l,r]$ 裏找到三個數字 $a,b,c$,知足 $gcd(a,b) = 1, gcd(b,c) = 1, gcd(a,c) > 1$。

題解:

相鄰兩個整數必然互質,相鄰兩個奇數必然互質。所以只能找連續的三個爲偶奇偶的數。

AC代碼:

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
ll l,r;
int main()
{
    cin>>l>>r;
    ll a=(l%2==0)?l:l+1;
    ll b=a+1, c=b+1;
    if(c<=r) printf("%lld %lld %lld\n",a,b,c);
    else printf("-1\n");
}

 


B - Friends and Presents - [二分]

You have two friends. You want to present each of them several positive integers. You want to present $cnt_1$ numbers to the first friend and $cnt_2$ numbers to the second friend. Moreover, you want all presented numbers to be distinct, that also means that no number should be presented to both friends.

In addition, the first friend does not like the numbers that are divisible without remainder by prime number $x$. The second one does not like the numbers that are divisible without remainder by prime number $y$. Of course, you're not going to present your friends numbers they don't like.

Your task is to find such minimum number $v$, that you can form presents using numbers from a set $1, 2, ..., v$. Of course you may choose not to present some numbers at all.

A positive integer number greater than $1$ is called prime if it has no positive divisors other than $1$ and itself.

Input
The only line contains four positive integers $cnt_1, cnt_2, x, y (1 \le cnt1, cnt2 < 10^9; cnt1 + cnt2 \le 10^9; 2 \le x < y \le 3·10^4)$ — the numbers that are described in the statement. It is guaranteed that numbers $x, y$ are prime.

Output
Print a single integer — the answer to the problem.

Examples
input
3 1 2 3
output
5
input
1 3 2 3
output
4

Note
In the first sample you give the set of numbers {1, 3, 5} to the first friend and the set of numbers {2} to the second friend. Note that if you give set {1, 3, 5} to the first friend, then we cannot give any of the numbers 1, 3, 5 to the second friend.

In the second sample you give the set of numbers {3} to the first friend, and the set of numbers {1, 2, 4} to the second friend. Thus, the answer to the problem is 4.

題意:

如今要給兩我的贈送一些正整數,要給第一我的送 $cnt_1$ 個正整數,要給第二我的送 $cnt_2$ 個正整數,且第一我的不想要能被 $x$ 整除的數,第二我的不想要能被 $y$ 整除的數。

如今你要求出最小的正整數 $v$,意味着你送出的正整數所有屬於 $1 \sim v$。

題解:

二分卻是不難想到,就是算怎麼給兩人整數不太好想。

舉個栗子:假設我如今 $v=7,x=2,y=3,cnt_1=3,cnt_2=3$,顯然此時能給第一我的的數字有 $1,3,5,7$,能給第二我的的數字有 $1,2,4,5,7$;

那麼咱們知道,爲了儘可能使得 $v$ 小,咱們應該儘可能把第二我的不要的數字塞給第一我的,好比第二我的是不要 $3$ 的,但第一我的要,因此能夠塞給他;同理,第一我的不要 $2,4$,可是第二我的要,能夠塞給他;

假設 $1 \sim v$ 中能被 $x$ 整除的數有 $cnt_x$ 個,能被 $y$ 整除的數有 $cnt_y$ 個,能被 $xy$ 整除的數有 $cnt_xy$ 個;所以,第二我的不要而第一我的要的數字,其數量爲 $cnt_y - cnt_xy$,第一我的不要而第二我的要的數字,其數量爲 $cnt_x - cnt_xy$;

那麼剩下來再要送出去的數字,就既不能被 $x$ 整除,也不能被 $y$ 整除了,若是這些數夠分,那麼 $v$ 就是可行的,不夠分的話 $v$ 就不可行。

AC代碼:

#include<bits/stdc++.h>
using namespace std;
int cnt1,cnt2,x,y;

int judge(int v)
{
    int cnt_x=v/x, cnt_y=v/y, cnt_xy=v/(x*y);
    int rest1=max(cnt1-(cnt_y-cnt_xy),0);
    int rest2=max(cnt2-(cnt_x-cnt_xy),0);
    return rest1+rest2<=v-cnt_x-cnt_y+cnt_xy;
}

int main()
{
    ios::sync_with_stdio(0);
    cin.tie(0);

    cin>>cnt1>>cnt2>>x>>y;
    int l=1, r=2*(cnt1+cnt2);
    while(l<r)
    {
        int mid=l+(r-l)/2;
        if(judge(mid)) r=mid;
        else l=mid+1;
    }
    cout<<l<<'\n';
}

 


C - Diverse Permutation - [構造題]

Permutation $p$ is an ordered set of integers $p_1, p_2, \cdots, p_n$, consisting of $n$ distinct positive integers not larger than $n$. We'll denote as n the length of permutation $p_1, p_2, \cdots, p_n$.

Your task is to find such permutation $p$ of length $n$, that the group of numbers $|p_1 - p_2|, |p_2 - p_3|, \cdots, |p_{n-1} - p_n|$ has exactly $k$ distinct elements.

Input
The single line of the input contains two space-separated positive integers $n, k (1 \le k < n \le 10^5)$.

Output
Print $n$ integers forming the permutation. If there are multiple answers, print any of them.

Examples
Input
3 2
Output
1 3 2
Input
3 1
Output
1 2 3
Input
5 2
Output
1 3 2 4 5

題意:

讓你給出一個 $1 \sim n$ 的某個排列 $p$,使得 $|p_1 - p_2|, |p_2 - p_3|, \cdots, |p_{n-1} - p_n|$ 中包含 $k(1 \le k \le n-1)$ 個不一樣的數字。

題解:

實際上,要 $1 \sim n$ 的排列 $p$ 最多可讓 $k=n-1$,換句話說,只須要 $1 \sim k+1$ 的排列就能知足條件。

而後,第一步,把 $1$ 放到 $k$ 和 $k+1$ 之間,便可產生 $k-1,k$ 這兩個不一樣的數字,然後,把 $2$ 放到 $k-1$ 和 $k$ 之間可產生 $k-3,k-2$ 這兩個不一樣的數字。依次類推便可。

AC代碼:

#include<bits/stdc++.h>
using namespace std;
const int maxn=1e5+10;
int n,k;
int num[maxn];
void fill(int v,int k)
{
    if(k<=0) return;
    num[k]=v;
    num[k-1]=num[k+1]-1;
    fill(v+1,k-2);
}
int main()
{
    cin>>n>>k;
    for(int i=k+1;i<=n;i++) num[i]=i;
    fill(1,k);
    for(int i=1;i<=n;i++) printf("%d%c",num[i],i<n?' ':'\n');
}

 


D - Interesting Array - [線段樹]

We'll call an array of $n$ non-negative integers $a[1], a[2], ..., a[n]$ interesting, if it meets $m$ constraints. The $i$-th of the $m$ constraints consists of three integers $l_i, r_i, q_i (1 \le l_i \le r_i \le n)$ meaning that value $a[l_i] \& a[l_{i+1}] \& \cdots \& a[r_i]$ should be equal to $q_i$.

Your task is to find any interesting array of $n$ elements or state that such array doesn't exist.

Expression $x$&$y$ means the bitwise AND of numbers $x$ and $y$. In programming languages C++, Java and Python this operation is represented as "&", in Pascal — as "and".

Input
The first line contains two integers $n, m (1 \le n \le 10^5, 1 \le m \le 10^5)$ — the number of elements in the array and the number of limits.

Each of the next m lines contains three integers $l_i, r_i, q_i (1 \le l_i \le r_i \le n, 0 \le qi < 2^{30})$ describing the $i$-th limit.

Output
If the interesting array exists, in the first line print "YES" (without the quotes) and in the second line print n integers $a[1], a[2], ..., a[n] (0 \le a[i] < 2^{30})$ decribing the interesting array. If there are multiple answers, print any of them.

If the interesting array doesn't exist, print "NO" (without the quotes) in the single line.

Examples
Input
3 1
1 3 3
Output
YES
3 3 3
Input
3 2
1 3 3
1 3 2
Output
NO

題意:

給出 $n,m$,若數組 $a[1 \sim n]$ 知足 $m$ 項條件中的任意一項,則稱它爲有趣的。

第 $i$ 項條件表述爲 $l_i, r_i, q_i (1 \le l_i \le r_i \le n, 0 \le q_i < 2^{30})$,即要求 $a[l_i] \& a[l_{i+1}] \& \cdots \& a[r_i] = q_i$。

如今要你構造一個有趣的數組 $a[1 \sim n]$。

題解:

初始所有 $a[i]=0$;考慮每一組 $l_i, r_i, q_i$,既然要 $a[l_i] \& a[l_{i+1}] \& \cdots \& a[r_i] = q_i$,那麼就應當對區間 $[l,r]$ 內全部的 $a[i]|=q[i]$。

最後 $m$ 組要求所有完成後,再從新跑一遍 $m$ 個 $l_i, r_i, q_i$ 進行驗證便可。

AC代碼:

#include<bits/stdc++.h>
using namespace std;
const int maxn=1e5+10;
const int maxm=1e5+10;
int n,m;
int l[maxm],r[maxm],q[maxm];

/********************************* Segment Tree - st *********************************/
struct Node{
    int l,r;
    int val,lazy;
    void update(int x)
    {
        val|=x;
        lazy|=x;
    }
}node[4*maxn];
void pushdown(int root)
{
    if(node[root].lazy)
    {
        node[root*2].update(node[root].lazy);
        node[root*2+1].update(node[root].lazy);
        node[root].lazy=0;
    }
}
void pushup(int root)
{
    node[root].val=node[root*2].val&node[root*2+1].val;
}
void build(int root,int l,int r) //對區間[l,r]建樹
{
    node[root].l=l; node[root].r=r;
    node[root].val=0; node[root].lazy=0;
    if(l==r) node[root].val=0;
    else
    {
        int mid=l+(r-l)/2;
        build(root*2,l,mid);
        build(root*2+1,mid+1,r);
        pushup(root);
    }
}
void update(int root,int st,int ed,int val)
{
    if(st>node[root].r || ed<node[root].l) return;
    if(st<=node[root].l && node[root].r<=ed) node[root].update(val);
    else
    {
        pushdown(root);
        update(root*2,st,ed,val);
        update(root*2+1,st,ed,val);
        pushup(root);
    }
}
int query(int root,int st,int ed)
{
    if(st>node[root].r || ed<node[root].l) return (1<<30)-1;
    if(st<=node[root].l && node[root].r<=ed) return node[root].val;
    else
    {
        pushdown(root);
        int ls=query(root*2,st,ed);
        int rs=query(root*2+1,st,ed);
        pushup(root);
        return ls&rs;
    }
}
/********************************* Segment Tree - ed *********************************/

int main()
{
    scanf("%d%d",&n,&m);
    build(1,1,n);
    for(int i=1;i<=m;i++)
    {
        scanf("%d%d%d",&l[i],&r[i],&q[i]);
        update(1,l[i],r[i],q[i]);
    }

    bool ok=1;
    for(int i=1;i<=m;i++)
    {
        if(query(1,l[i],r[i])!=q[i]) {
            ok=0; break;
        }
    }
    printf("%s\n",ok?"YES":"NO");
    if(ok) for(int i=1;i<=n;i++) printf("%d%c",query(1,i,i),i<n?' ':'\n');
}

 


E - Game with Strings - [指望+狀壓DP]

You play the game with your friend. The description of this game is listed below.

Your friend creates $n$ distinct strings of the same length $m$ and tells you all the strings. Then he randomly chooses one of them. He chooses strings equiprobably, i.e. the probability of choosing each of the $n$ strings equals $\frac{1}{n}$. You want to guess which string was chosen by your friend.

In order to guess what string your friend has chosen, you are allowed to ask him questions. Each question has the following form: «What character stands on position $pos$ in the string you have chosen?» A string is considered guessed when the answers to the given questions uniquely identify the string. After the string is guessed, you stop asking questions.

You do not have a particular strategy, so as each question you equiprobably ask about a position that hasn't been yet mentioned. Your task is to determine the expected number of questions needed to guess the string chosen by your friend.

Input
The first line contains a single integer $n (1 \le n \le 50)$ — the number of strings your friend came up with.

The next $n$ lines contain the strings that your friend has created. It is guaranteed that all the strings are distinct and only consist of large and small English letters. Besides, the lengths of all strings are the same and are between $1$ to $20$ inclusive.

Output
Print the single number — the expected value. Your answer will be considered correct if its absolute or relative error doesn't exceed $10^{-9}$.

Examples
Input
2
aab
aac
Output
2.000000000000000
Input
3
aaA
aBa
Caa
Output
1.666666666666667
Input
3
aca
vac
wqq
Output
1.000000000000000

題意:

給出 $n$ 個等長 $m$ 的字符串,對方會等可能地選擇其中一個字符串,我不知道他選擇了哪一個,因此每次會等可能地詢問對方某個位置上是什麼字符。

一旦可以惟一肯定是哪一個字符串,我就中止詢問。求我詢問次數的指望。

題解:

假設 $dp[sta]$ 表示進行 $sta$ 狀態詢問的機率,這樣一來,要完成狀態轉移,必需要知道在進行該狀態的詢問後是否已經肯定對方選定的串。

若是已經肯定對方選定的串,那麼咱們就不須要再詢問了;不然還要再詢問一個位置,那麼這個多一次的詢問,對於整個詢問數目的指望的貢獻就爲 $dp[sta] \times 1$。

咱們可使用 $f[same]$ 來表示 $same$ 狀態的位置上字符所有相同的字符串有哪些。而後咱們發現,$f[same]$ 也是能夠用dp求出來的,一開始咱們 $O(n^2)$ 地求出初始條件:對於任意兩個字符串 $i,j(i<j)$,他們全部的同位置上相同的字符可表示成狀態 $same$($1$ 表示相同,$0$ 表示不一樣),即有初始條件 $f[same]=(1<<i)|(1<<j)$。其後咱們能夠dp求出所有的 $f[same]$,若是 $same_1$ 狀態包含$same_2$ 狀態,那麼 $same_1$ 狀態下不能區分的字符串在 $same_2$ 狀態下同樣沒法區分。

接下來考慮如何根據 $dp[sta]$ 求指望,$f[sta]$ 裏有 $k$ 個 $1$ 表明有 $k$ 個串尚不能分辨,那麼對於詢問這 $k$ 個串其中一個,詢問指望就是 $dp[sta]×1$,其餘0的位置就是已經不須要詢問了,1的位置此次詢問是此次必須的,以後還詢問不詢問不知道,還詢問不詢問是後面的狀態所決定的,對於 $k$ 個詢問次數指望就是 $dp[sta] \times k$,對於最後的答案天然是加 $dp[sta] \times k / n$。其實主要思想就是利用指望的可加性,分開來算。

AC代碼:

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn=53;
const int maxm=23;

int n,m;
char s[maxn][maxm];
double dp[1<<20];
ll f[1<<20];
inline int cntbit(int x){return x?(x&1)+cntbit(x>>1):0;}
int main()
{

    scanf("%d",&n);
    for(int i=0;i<n;i++) scanf("%s",s[i]);
    m=strlen(s[1]);

    //f[same]初始化
    memset(f,0,sizeof(f));
    for(int i=0;i<n;i++)
    {
        for(int j=i+1;j<n;j++)
        {
            int same=0;
            for(int k=0;k<m;k++) {
                if(s[i][k]==s[j][k]) same|=(1<<k);
            }
            f[same]|=(1LL<<j)|(1LL<<i);
        }
    }

    //dp求全部的f[same]
    for(int same=(1<<m)-1;same>=0;same--)
    {
        for(int i=0;i<m;i++) {
            if(same&(1<<i)) f[same^(1<<i)]|=f[same];
        }
    }

    double ans=0;
    memset(dp,0,sizeof(dp)); dp[0]=1;
    for(int sta=0;sta<(1<<m);sta++)
    {
        if(f[sta]==0) continue;

        int cnt=cntbit(sta);
        for(int i=0;i<m;i++)
        {
            if(sta&(1<<i)) continue;
            dp[sta|(1<<i)]+=dp[sta]/(m-cnt);
        }

        for(int i=0;i<n;i++) {
            if(f[sta]&(1LL<<i)) ans+=dp[sta];
        }
    }

    printf("%.10f\n",ans/n);
}

說實話,這題爲何這麼求指望我仍是雲裏霧裏的……應該是我飄了,指望和機率題也敢碰了……

相關文章
相關標籤/搜索