7-1 欢迎词 (5 分)

这个就没什么好说的了,直接输出就可以了。

#include<bits/stdc++.h>
using namespace std;
int main()
{
    cout << "Hello!\nWelcom to LinYi University!" << endl;
    return 0;
}

7-2 落单的数 (10 分)

只有两个数是出现一次,其它都是出现两次,那么只要用一个集合:对于每个读入的数,集合里没有就放进去,集合里有就抹掉。那么最后剩下的两个数就是要输出的答案。

#include<bits/stdc++.h>
using namespace std;
set<int>s;
int main()
{
    int n;
    while(cin >> n)
    {
        if(s.find(n) != s.end())
            s.erase(n);
        else
            s.insert(n);
    }
    int flag = 0;
    for(auto x : s)
    {
        if(flag == 0)
            flag = 1;
        else
            cout << " ";
        cout << x;
    }
    return 0;
}

7-3 哥德巴赫猜想 (10 分)

$2^{15}$大概是三万多,因为是多次询问,所以我们先把这个范围内的素数处理出来,然后对于每一次询问只需要跑一个$O(n)$即可

#include<bits/stdc++.h>
using namespace std;
const int maxn = 40000;
bool is_prime[maxn];
void Init()
{
    for(int i = 0;i < maxn; i++)
        is_prime[i] = true;
    is_prime[0] = is_prime[1] = false;
    for(int i = 2;i <= maxn; i++)
    {
        if(is_prime[i])
        {
            for(int j = i * 2;j < maxn; j += i)
                is_prime[j] = false;
        }
    }
}
int main()
{
    Init();
    int n;
    while(cin >> n)
    {
        if(n == 0)
            break;
        int ans = 0;
        for(int i = 2;i <= n/2; i++)
        {
            if(is_prime[i] && is_prime[n-i])
                ans++;
        }
        cout << ans << endl;
    }
    return 0;
}

7-4 过河 (10 分)

就是一个基本的dp,前三步先算出来,然后后边的每一步都可以从前边三步走过来,所以状态转移方程就是

dp[i] = dp[i-1] + dp[i-2] + dp[i-3];
#include<bits/stdc++.h>
using namespace std;
const int maxn = 1e6+10;
int dp[maxn];
int main()
{
    dp[1] = 1;
    dp[2] = 2;
    dp[3] = 4;
    int n;
    cin >> n;
    for(int i = 4;i <= n; i++)
        dp[i] = dp[i-1] + dp[i-2] + dp[i-3];
    cout << (n > 0 ? dp[n] : 0) << endl;
    return 0;
}

7-5 分数位置 (15 分)

这个题手动模拟一下除法就可以了。

#include<bits/stdc++.h>
using namespace std;
int main()
{
    int a,b,c;
    cin >> a >> b >> c;
    a %= b;
    for(int i = 1;i <= 10000; i++)
    {
        a *= 10;
        int tmp = a / b;
        a %= b;
        if(tmp == c)
        {
            cout << i << endl;
            return 0;
        }
    }
    puts("-1");
    return 0;
}

7-6 两个蛋糕 (15 分)

因为是两个蛋糕,所以每一层只出现两次,把这两次找出来把相邻间的差的绝对值加起来就ok了。

#include<bits/stdc++.h>
#define int long long
using namespace std;
const int maxn = 1e5+10;
int arr[2][maxn];
bool al[maxn];
signed main()
{
    int n, k;
    cin >> n;
    memset(al, false, sizeof(al));
    int ne = n * 2;
    for(int i = 0; i < ne; ++i)
    {
        cin >> k;
        if(al[k])
            arr[1][k] = i;
        else
        {
            arr[0][k] = i;
            al[k] = true;
        }
    }
    arr[1][0] = arr[0][0] = 0;
    int ans = 0;
    for(int i = 1; i <= n; ++i)
        ans += abs(arr[0][i]-arr[0][i-1]) + abs(arr[1][i]-arr[1][i-1]);
    cout << ans << endl;
    return 0;
}

7-7 报数字 (15 分)

写出sg函数后找出循环节,然后循环节内对应输出就可以了。

#include<bits/stdc++.h>
#define int long long
using namespace std;
signed main()
{
    int t,n,m;
    cin >> t;
    while(t--)
    {
        cin >> n >> m;
        n %= (m + 1);
        if(n == 0)
            puts("second");
        else
            puts("first");
    }
    return 0;
}

7-8 与零交换 (25 分)

一开始看到此题大喜,还25分,前两天刚学了置换群,感觉就是板子题。直到看到了那个最少的次数,好吧,还是我太年轻了。闲话到此结束,此题的正解是:

设置个map作为值x所在位置的映射,拿0所在的下标作为交换的值,去找对应的值交换,如果当前0的位置是0本身就遍历去找不在本身位置的数字,直到所有都符合,

#include<bits/stdc++.h>
using namespace std;
const int maxn = 1e5+100;
int x[maxn],n;
map<int,int> mp;
int main()
{
    int p;
    cin >> n;
    for (int i = 0; i < n; i++)
    {
        cin >> x[i];
        if (x[i] == 0)
            p = i;
        mp[x[i]] = i;
    }
    int cnt = 0;
    int q = 0;
    while (1)
    {
        if (x[p] != p)
        {
            int p1 = mp[p];
            mp[x[p]] = p1;
            mp[x[p1]] = p;
            swap(x[p], x[p1]);
            p = p1;
            cnt++;
        }
        else
        {
            int flag = 0,p1;
            for ( ;q < n; q++)
            {
                if (x[q] != q)
                {
                    p1 = q;
                    flag = 1;
                    break;
                }
            }
            if (!flag)
                break;
            else
            {
                cnt++;
                p = p1;
                mp[x[p1]] = p;
                mp[x[p]] = p1;
                swap(x[p], x[p1]);
                p = p1;
                cnt++;
            }
        }
    }
    cout << cnt << endl;
    return 0;
}

7-9 看电影 (15 分)

这个题就是贪心的板子题了,先把电影按结束时间从小到大排序,然后遍历开始时间非小于正在观看的那一部电影的结束时间就累加1,最后输出即为答案。

#include<bits/stdc++.h>
using namespace std;
const int maxn = 120;
pair<int,int>pq[maxn];
bool cmp(pair<int,int> p,pair<int,int> q)
{
    return p.second < q.second;
}
int main()
{
    int n;
    while(cin >> n)
    {
        if(n == 0)
            break;
        for(int i = 0;i < n; i++)
            cin >> pq[i].first >> pq[i].second;
        sort(pq,pq + n,cmp);
        int ans = 0,last = 0;
        for(int i = 0;i < n; i++)
        {
            if(pq[i].first >= last)
            {
                ans++;
                last = pq[i].second;
            }
        }
        cout << ans << endl;
    }
    return 0;
}

7-10 迎风一刀斩 (30 分)

emmmm,先膜一下刘汝佳大神,话说看到这题作者是我大神的名字的时候极为震惊,我第一本算法入门书就是大神写的,那么这个题呢就:咕咕咕咕咕。