本次难度:
202006-1 线性分类器
题目:202006-1
题目分析:
给定n个点,并标记为AB两类,问给定直线是否能将其分为两个点集。
简单数学知识,点在直线上满足ax+by+c=0,点在直线割平面所得的上下其值会正负相反。
AC代码:
// -*- coding:utf-8 -*-// File : 202006-1.cpp
// Time : 2024/03/26
// Author : wolf#include <iostream>
#include <vector>
const int N = 1e8 + 10;
using namespace std;
vector<pair<int, int>> A, B;
#define ll long long// 判断该点在直线的哪一侧
int func(ll t0, ll t1, ll t2, ll x, ll y)
{return (t0 + t1 * x + t2 * y) > 0 ? 1 : -1;
}int main()
{int n, m;cin >> n >> m;// 读入n个点,使用两个vector进行存储for (int i = 0; i < n; i++){ll x, y;char type;cin >> x >> y >> type;if (type == 'A')A.push_back(make_pair(x, y));else if (type == 'B')B.push_back(make_pair(x, y));}while (m--){ll t0, t1, t2;cin >> t0 >> t1 >> t2;bool ans = true; // 标记,假设是满足的int A_std, B_std;// 设定A标准 B标准,就是在A集合和B集合各自随便抓一个点,代表它们集合中的点应该在直线哪边// 当且仅当A集合内的点都和A标准相等,B集合内的点都和B标准相等,再A标准不等于B标准,才满足题意A_std = func(t0, t1, t2, A[0].first, A[0].second);for (unsigned int i = 0; i < A.size(); i++){if (func(t0, t1, t2, A[i].first, A[i].second) != A_std){ans = false;break;}}B_std = func(t0, t1, t2, B[0].first, B[0].second);for (unsigned int i = 0; i < B.size(); i++){if (func(t0, t1, t2, B[i].first, B[i].second) != B_std){ans = false;break;}}if (A_std != B_std && ans == true)cout << "Yes" << endl;elsecout << "No" << endl;}return 0;
}
202006-2 稀疏矩阵(100)
题目:202006-2
题目分析:
如题,给定两个向量,但是是稀疏的。稀疏就是该向量大多数位置都是0,只有少数有值。
最简单的想法就是用数组模拟这两个向量,然后for循环模拟相加,但这样显然超时间,考虑负数的话要达到10^12,应该是0分。
再进一步,想到两重循环遍历这两个向量,index相等时相乘,30分,在10^5时就会超时,只能过前三个点,因此30分。
要完成10^9的数据规模,应该是O(n),即只能遍历一次。那么就遍历一个向量A,对于这个向量A的每一个,用while使得B不超过A的当前index但尽可能增大,这样大概就是2O(n)的复杂度,因此能100分。
AC代码:
// -*- coding:utf-8 -*-// File : 202006-2.cpp
// Time : 2024/03/25
// Author : wolf#include <algorithm>
#include <iostream>
#include <vector>
#define ll long longusing namespace std;
vector<pair<int, int>> v1, v2;
bool cmp(pair<int, int> a, pair<int, int> b)
{return a.first < b.first;
}int main()
{int n, a, b;cin >> n >> a >> b;for (int i = 0; i < a; i++){int index, value;cin >> index >> value;v1.push_back(make_pair(index, value));}for (int i = 0; i < b; i++){int index, value;cin >> index >> value;v2.push_back(make_pair(index, value));}ll ans = 0;// 30分答案// for (unsigned int i = 0; i < v1.size(); i++)// {// for (unsigned int j = 0; j < v2.size(); j++)// {// if (v1[i].first == v2[j].first)// ans += v1[i].second * v2[j].second;// }// }// 100分答案sort(v1.begin(), v1.end(), cmp);sort(v2.begin(), v2.end(), cmp);unsigned int v2i = 0;for (unsigned int v1i = 0; v1i < v1.size(); v1i++){while (v2[v2i].first < v1[v1i].first && v2i < v2.size())v2i++;if (v1[v1i].first == v2[v2i].first)ans += v1[v1i].second * v2[v2i].second;}cout << ans << endl;return 0;
}