大二下第七周学习笔记

大二下课业真的太重了……机器学习课一周要精读4篇论文,掌握每篇论文的数学原理推导……

光是学校的课程就要花去我所有的时间……

现在好不容易挤点时间训练

周二

昨天vp了一场cfdiv2 然而打到一半被叫去做核酸……

算是做了一些水题吧 先找回打代码的感觉

A. Marin and Photoshoot(观察)

观察发现两个0之间必须有两个1

#include<bits/stdc++.h>
#define rep(i, a, b) for(int i = (a); i < (b); i++) 
#define _for(i, a, b) for(int i = (a); i <= (b); i++) 
using namespace std;

int main()
{
	int T; scanf("%d", &T);
	while(T--)
	{
		int n; 
		scanf("%d", &n);
		
		int ans = 0, pre = 0;
		_for(i, 1, n)
		{
			int x; scanf("%1d", &x);
			if(!x)
			{
				if(!pre) pre = i;
				else
				{
					int cur = i - pre;
					if(cur == 1) ans += 2;
					else if(cur == 2) ans++;
					pre = i;
				}
			}
		}
		printf("%d\n", ans);
	}
	
	return 0;
}

B. Marin and Anti-coprime Permutation(gcd)

考虑奇偶搭配即可

#include<bits/stdc++.h>
#define rep(i, a, b) for(int i = (a); i < (b); i++) 
#define _for(i, a, b) for(int i = (a); i <= (b); i++) 
using namespace std;

typedef long long ll;
const int mod = 998244353;

int main()
{
	int T; scanf("%d", &T);
	while(T--)
	{
		int n; 
		scanf("%d", &n);
		
		if(n % 2 == 1) puts("0");
		else
		{
			ll ans = 1;
			_for(i, 1, n / 2) ans = ans * i % mod;
			ans = ans * ans % mod;
			printf("%lld\n", ans);
		}
	}
	
	return 0;
}

C. Shinju and the Lost Permutation(找规律)

这种题无非就是找规律,看满足什么条件

c的值其实就是以第一个数为起点的最长上升子序列。

可以把数据移动一下,把1放到第一个

然后后面的值与前面的值,如果是增加的话,最多加1,减少的话不能减到1

用最长上升子序列的角度很好理解上面两个条件

我当时只是确定了这是必要条件,但是不清楚是不是充分条件

但没管太多,写了一发就AC了

#include<bits/stdc++.h>
#define rep(i, a, b) for(int i = (a); i < (b); i++) 
#define _for(i, a, b) for(int i = (a); i <= (b); i++) 
using namespace std;

const int N = 1e5 + 10;
int a[N], b[N], n;

int main()
{
	int T; scanf("%d", &T);
	while(T--)
	{
		scanf("%d", &n);
		
		int pos = 0;
		_for(i, 1, n) 
		{
			scanf("%d", &a[i]);
			if(a[i] == 1) pos = i;
		}
		if(!pos) 
		{
			puts("NO");
			continue;
		}
		
		int cnt = 0;
		_for(i, pos, n) b[++cnt] = a[i];
		_for(i, 1, pos - 1) b[++cnt] = a[i];
		
		bool ok = true;
		_for(i, 2, n)
			if(b[i] - b[i - 1] > 1 || b[i] < 2) 
			{
				ok = false;
				break;
			}
		puts(ok ? "YES" : "NO");
	}
	
	return 0;
}

D1. 388535(Easy Version) (异或)

这种位运算的题,尤其是涉及到异或的,常规套路都是按照一位一位考虑

果然挺久没做题了 都不敏感了 当时没有怎么意识到 而是尝试去推公式了

按照一位一位来考虑,在某一位下,0的个数小于等于1的个数

如果x的这一位为1,那么0和1全部颠倒,就变成0的个数大于等于1的个数

所以可以通过统计个数来判断x的这一位是否为1

一位一位判断即可

#include<bits/stdc++.h>
#define rep(i, a, b) for(int i = (a); i < (b); i++) 
#define _for(i, a, b) for(int i = (a); i <= (b); i++) 
using namespace std;

const int N = 2e5 + 10;
int a[N], l, r;

int main()
{
	int T; scanf("%d", &T);
	while(T--)
	{
		scanf("%d%d", &l, &r);
		_for(i, l, r) scanf("%d", &a[i]);
		
		int ans = 0;
		_for(j, 0, 16)
		{
			int cnt0 = 0, cnt1 = 0;
			_for(i, l, r)
			{
				if(a[i] & (1 << j)) cnt1++;
				else cnt0++;
			}
			if(cnt0 < cnt1) ans |= 1 << j;
		}
		printf("%d\n", ans);	
	}
	
	return 0;
}

D1. 388535(Hard Version) (字典树)

这个想法非常妙啊

我当时想题的时候就想枚举每一个可能的x,然后迅速判断它是否是答案。后一步并没有想出来

正解就是这个思路。

首先怎么枚举可能的x呢

l ^ x = ai

x = l ^ ai

所以枚举每一个ai   然后l^ai就一定是可能的x,x一定是其中的一个

那么怎么迅速判断这个x是否是答案呢 这里很妙

[a1……an] ^ x = [l……r]

因为l到r各不相同,所以a1到an各部相同

如果这个等式成立,必然有x与[a1……an]的异或最小值等于l,最大值等于r

又因为[a1……an]各不相同,异或x后也各不相同,所以异或后的结果必然是[l, r]

所以只需要判断异或最小值和最大值就行了,这就是字典树模板题了

#include<bits/stdc++.h>
#define rep(i, a, b) for(int i = (a); i < (b); i++) 
#define _for(i, a, b) for(int i = (a); i <= (b); i++) 
using namespace std;

const int N = 3e6 + 10;
int trie[N][2], a[N], l, r, cnt;

void insert(int x)
{
	int p = 0;
	for(int j = 16; j >= 0; j--)
	{
		int cur = (x >> j) & 1;
		if(!trie[p][cur]) trie[p][cur] = ++cnt;
		p = trie[p][cur];
	}
}

int max_xor(int x)
{
	int res = 0, p = 0;
	for(int j = 16; j >= 0; j--)
	{
		int cur = (x >> j) & 1;
		if(trie[p][cur ^ 1])
		{
			p = trie[p][cur ^ 1];
			res |= 1 << j;
		}
		else p = trie[p][cur];
	}
	return res;
}

int min_xor(int x)
{
	int res = 0, p = 0;
	for(int j = 16; j >= 0; j--)
	{
		int cur = (x >> j) & 1;
		if(trie[p][cur]) p = trie[p][cur];
		else
		{
			res |= 1 << j;
			p = trie[p][cur ^ 1];
		}
	}
	return res;
}

int main()
{
	int T; scanf("%d", &T);
	while(T--)
	{
		cnt = 0;
		scanf("%d%d", &l, &r);
		_for(i, l, r) scanf("%d", &a[i]), insert(a[i]);
		
		_for(i, l, r)
		{
			int x = a[i] ^ l;
			if(min_xor(x) == l && max_xor(x) == r)
			{
				printf("%d\n", x);
				break;
			}
		}
		_for(i, 0, cnt) trie[i][0] = trie[i][1] = 0;
	}
	
	return 0;
}

周三

过几天是蓝桥杯省赛,随便挑了几道题做了做

回文日期(模拟)

模拟即可。

用string的substr函数以及stoi函数string转int会很方便

注意闰年判断条件

#include<bits/stdc++.h>
#define rep(i, a, b) for(int i = (a); i < (b); i++) 
#define _for(i, a, b) for(int i = (a); i <= (b); i++) 
using namespace std;

string s;
int year, month, day;

void add()
{
    day++;
    int m[20] = {0};
    _for(i, 1, 12) m[i] = 31;
    m[4] = m[6] = m[9] = m[11] = 30;
    if(year % 4 == 0 && year % 100 != 0 || year % 400 == 0) m[2] = 29;
    else m[2] = 28;
    
    if(day == m[month] + 1)
    {
        month++;
        day = 1;
        if(month == 13)
        {
            month = 1;
            year++;
        }
    }
}

string get(int year, int month, int day)
{
    string a, b, c;
    a = to_string(year);
    if(month < 10) b = "0" + to_string(month);
    else b = to_string(month);
    if(day < 10) c = "0" + to_string(day);
    else c = to_string(day);
    return " " + a + b + c;
}

bool check1()
{
    string s = get(year, month, day);
    _for(i, 1, 4)
        if(s[i] != s[8 - i + 1])
            return false;
    return true;
}

bool check2()
{
    if(!check1()) return false;
    string s = get(year, month, day);
    return s[1] == s[3] && s[2] == s[4];
}

int main()
{
    cin >> s; s = " " + s;
    year = stoi(s.substr(1, 4));
    month = stoi(s.substr(5, 2));
    day = stoi(s.substr(7, 2));
    
    int flag = 1;
    while(1)
    {
        add();
        if(flag && check1())
        {
            printf("%d%02d%02d\n", year, month, day);
            flag = 0;
        }
        if(check2())
        {
            printf("%d%02d%02d\n", year, month, day);
            break;
        }
    }
            
    return 0;
}

子串分值(思维)

读完题就做了

枚举一个点的贡献即可

#include<bits/stdc++.h>
#define rep(i, a, b) for(int i = (a); i < (b); i++) 
#define _for(i, a, b) for(int i = (a); i <= (b); i++) 
using namespace std;

typedef long long ll;
vector<int> pos[30];
string s;

int main()
{
    cin >> s;
    int len = s.size();
    
    _for(i, 0, 25) pos[i].push_back(-1);
    rep(i, 0, len) pos[s[i] - 'a'].push_back(i);
    _for(i, 0, 25) pos[i].push_back(len);
        
    ll ans = 0;
    rep(i, 0, len)
    {
        int id = s[i] - 'a';
        int x = lower_bound(pos[id].begin(), pos[id].end(), i) - pos[id].begin();
        int l = pos[id][x - 1], r = pos[id][x + 1];
        ans += 1LL * (i - l) * (r - i); 
    }
    printf("%lld\n", ans);
    
    return 0;
}

周四

F. Remainder Problem(分块)

这题妙啊,分块

记录ans[x][y]  表示模x余y的答案

每次增加,就更新ans和a数组

分类讨论,如模数大于700,则暴力遍历a数组

否则直接输出ans[x][y]

#include<bits/stdc++.h>
#define rep(i, a, b) for(int i = (a); i < (b); i++)
#define _for(i, a, b) for(int i = (a); i <= (b); i++)
using namespace std;

const int N = 5e5 + 10;
int a[N], ans[705][705], q;

int main()
{
	scanf("%d", &q);
	while(q--)
	{
		int op, x, y;
		scanf("%d%d%d", &op, &x, &y);
		if(op == 1) 
		{
			a[x] += y;
			_for(i, 1, 700) ans[i][x % i] += y;
		}
		else
		{
			if(x <= 700) printf("%d\n", ans[x][y]);
			else
			{
				int res = 0;
				for(int i = y; i <= 5e5; i += x)
					res += a[i];
				printf("%d\n", res);
			}
		}
	}
	
	return 0;
}

C. Money Transfers(思维)

这道题妙啊

首先,如果存在一个和为0的区间,那么操作可以少一次

那么就尽可能的找多的和为0的区间

和为0意味着前缀和里面有两个相同的数

因此做前缀和,看有多少个相同的数,就意味着有多少个和为0的区间

注意首尾也是一个区间,因为中间和为0,总和为0,得出首尾和也为0

#include<bits/stdc++.h>
#define rep(i, a, b) for(int i = (a); i < (b); i++)
#define _for(i, a, b) for(int i = (a); i <= (b); i++)
using namespace std;

typedef long long ll;
map<ll, int> mp;
int ans, n;

int main()
{
	scanf("%d", &n);
	ll sum = 0;
	_for(i, 1, n)
	{
		int x; scanf("%d", &x);
		sum += x;
		ans = max(ans, ++mp[sum]);	
	} 
	printf("%d\n", n - ans);
	
	return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值