cf 是什么
具体来说就是
穿越火线是一款非常受欢迎的第一人称射击网络游戏,它也被称作"CrossFire"。它最早由韩国SmileGate公司开发,后来在中国由腾讯公司运营。这个游戏在中国有着庞大的玩家基础,在亚洲其他地区以及全球也有相当多的粉丝。
游戏提供了多种模式,包括经典的团队对战、爆破模式、生化模式等。玩家能选择成为全球反恐精英的一员,或者加入恐怖分子阵营。在游戏中,玩家可以购买和升级各种武器装备,与队友协同作战,完成不同的任务和挑战。
Codeforces 是一个在线编程和算法竞赛平台,广受全球程序员和算法爱好者的欢迎。它提供了一个平台,用户可以通过参加定期举办的编程比赛来提高自己的编程技能和算法知识。这些比赛通常分为几个不同的难度等级,适合从初学者到高级程序员的各个水平。
在Codeforces上,参赛者需要在限定时间内解决一系列编程问题。这些问题覆盖了数据结构、算法、数学、字符串处理、图论等众多领域。参赛者的表现根据解决问题的速度和正确性来评分,并在全球范围内进行排名。 – chatgpt 生成
网站: Codeforces
rust 是什么
具体来说就是
《Rust》是一款多人生存游戏,玩家需收集资源,建造庇护所,与环境和其他玩家互动以求生存。游戏以其开放世界、真实的生存挑战和玩家之间的复杂互动闻名。
Rust是一种开源的系统编程语言,以安全性、速度和并发性为设计目标。它旨在帮助开发者构建高效、可靠的软件,同时避免常见的内存安全错误,如缓冲区溢出。Rust通过一套严格的编译时检查机制实现这些目标,而无需依赖于传统的垃圾回收机制或大量的运行时检查。
Rust的设计充分考虑了现代硬件的特性,提供了零成本抽象、安全的并发编程模型,以及对内存布局的精细控制。这些特性使Rust成为开发操作系统、游戏引擎、浏览器组件以及需要高性能和高可靠性的应用程序的理想选择。 – chatgpt 生成
网站: Rust Programming Language
正文
rust 标准库只提供了一些基本和常用的数据结构和一套"抽象特性"(或者说规范),所以可能需要手动造一些轮子,比如处理输入和随机数生成等等,下面将介绍对应的实现代码。
处理输入
相较于 c++ 的 scanf
、cin
,rust 的标准库中缺少一种方便地从标准输入中读取并构造数据的方法,网上也存在一些讨论:
- Why is it so difficult to get user input in Rust? - help - The Rust Programming Language Forum
- Why is it so painful to read user inputs in Rust
所以使用 rust 解决 codeforces 中的问题所遇到的第一个困难就是如何处理输入,所幸的是这个问题在这里有所讨论,具体的解决方法参考这个回复。根据讨论中的代码修改,笔者使用的模板代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50
| pub struct Scanner<B> { reader: B, buf_str: Vec<u8>, buf_iter: std::str::SplitWhitespace<'static>, } impl<B: BufRead> Scanner<B> { pub fn new(reader: B) -> Self { Self { reader, buf_str: Vec::new(), buf_iter: "".split_whitespace(), } } pub fn sc<T: std::str::FromStr>(&mut self) -> T { loop { if let Some(token) = self.buf_iter.next() { return token.parse().ok().expect("Failed parse"); } self.buf_str.clear(); self.reader .read_until(b'\n', &mut self.buf_str) .expect("Failed read"); self.buf_iter = unsafe { let slice = std::str::from_utf8_unchecked(&self.buf_str); std::mem::transmute(slice.split_whitespace()) } } } }
static mut IN: *mut Scanner<StdinLock<'static>> = std::ptr::null_mut(); #[allow(unused_macros)] macro_rules! i { () => {{ i!(i32) }}; ($t:ty) => {{ unsafe { (*IN).sc::<$t>() } }}; }
fn main() { unsafe { IN = Box::leak(Box::new(Scanner::new(io::stdin().lock()))) as *mut Scanner<StdinLock<'_>>; } let t = i!(String); let a = i!(); }
|
下面大致解释一下代码的实现,首先核心的是一个名为 Scanner
的结构体,其中的 Reader
保存了数据的来源,它是一个 trait ,可以认为是任意实现了 std::io::BufRead
这一"规范"的结构体。具体到本代码则为标准输入流。buf_str
则是用于储存从 reader 中获取的数据(每次以行为单位读取数据)。buf_iter
用于解析和迭代遍历行内的数据。
一个读入的过程为: 从 Reader
中根据换行符为分隔读取数据,保存到 buf_str
中,然后根据空白符做切分保存到 buf_iter
中,然后对每一块做解析转换成需要的格式。
在之后是一个宏,用于优化使用体验。然而一定会使用到全局变量,rust 中对于全局变量的使用较为繁琐,这里采用两种方法:
- 使用全局裸指针,使用
box::leak
将读取器内存泄漏,然后将裸指针指向这块内存进行调用,不过这个需要用到 unsafe ,由于 codeforces 算法都为单线程,所以不用考虑并发问题,所以这个 unsafe 是可控的。
- 使用
RefCell
获取内部可变性。
通常来说,应该是使用全局裸指针性能更好,因为 RefCell
会在运行期进行借用检测。但经测试两者的性能差距不明显,所以读者自行选择。
全局读取器使用裸指针和 refcell 性能测试 · GitHub
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| thread_local! { pub static STDIN: std::cell::RefCell<Scanner<StdinLock<'static>>> = std::cell::RefCell::new(Scanner::new(io::stdin().lock())); } #[allow(unused_macros)] macro_rules! safe_i { () => {{ safe_i!(i32) }}; ($t:ty) => { STDIN.with(|r| { let mut r = r.borrow_mut(); r.sc::<$t>() }) }; }
|
除了上面的解决方式之外,讨论中还出现了一种解决方式,具体实现原理基本上和 c++ 的快读方式相似,然后通过把快读封装成 trait,给基本类型使用宏实现快读的 trait 实现数据的读入。理论上这种读取方式应该更快,请读者自行取用。
处理输出
对于输出也需要特殊处理一下,使用普通的 println!
宏可能会导致输出时间过长导致超时。原因是每次调用 println!
的时候都会给标准输出上锁。遇到需要许多输出的题目可能会因为频繁上锁解锁导致输出时间过长。
println!
的注释中也提到可以使用 writeln!
宏进行输出。对于输出加锁导致变慢的问题可以使用缓冲区解决。将输出保存到缓冲区中,最后调用 flush!
将缓冲区的内容写入输出流,这样就减少了加锁解锁的时间,实现了快速输出。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41
| static mut OUT: *mut std::io::BufWriter<std::io::StdoutLock<'_>> = std::ptr::null_mut(); #[allow(unused_macros)] macro_rules! w { ($fmt:expr) => { unsafe{ write!(*OUT, "{}", $fmt);} }; ($fmt:expr, $($args:tt)*) => { unsafe{ write!(*OUT, $fmt, $($args)*);} }; } #[allow(unused_macros)] macro_rules! wln { () => { unsafe{ writeln!(*OUT);} }; ($fmt:expr) => { unsafe{ writeln!(*OUT, "{}", $fmt);} }; ($fmt:expr, $($args:tt)*) => { unsafe{ writeln!(*OUT, $fmt, $($args)*);} }; } #[allow(unused_macros)] macro_rules! flush { () => { unsafe { (*OUT).flush(); } }; }
fn main(){ OUT = Box::leak(Box::new(io::BufWriter::new(io::stdout().lock()))) as *mut std::io::BufWriter<std::io::StdoutLock<'_>>; let a: i32 = 1; wln!(); wln!(a); w!(a); wln!("test: {}",a); flush!(); }
|
同样的,对应输出器的全局变量也有两种写法,一种 static mut
的全局变量,一种是 Refcell
,这里作为思考题请读者自行实现。
处理随机数
同样的,rust 的 std 中没有随机数库,需要自己手动造一下轮子。代码来源不可考了,贴一下原始的论文吧:
vigna.di.unimi.it/ftp/papers/ScrambledLinear.pdf
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42
| #[derive(Debug, Clone, PartialEq, Eq)] struct Rand { s: [u64; 4], }
impl Rand { pub fn new(mut state: u64) -> Self { const PHI: u64 = 0x9e3779b97f4a7c15; let mut seed = <[u64; 4]>::default(); for chunk in &mut seed { state = state.wrapping_add(PHI); let mut z = state; z = (z ^ (z >> 30)).wrapping_mul(0xbf58476d1ce4e5b9); z = (z ^ (z >> 27)).wrapping_mul(0x94d049bb133111eb); z = z ^ (z >> 31); *chunk = z; } Self { s: seed } }
#[inline] pub fn next_u32(&mut self) -> u32 { (self.next_u64() >> 32) as u32 }
#[inline] pub fn next_u64(&mut self) -> u64 { let result_plusplus = self.s[0] .wrapping_add(self.s[3]) .rotate_left(23) .wrapping_add(self.s[0]);
let t = self.s[1] << 17; self.s[2] ^= self.s[0]; self.s[3] ^= self.s[1]; self.s[1] ^= self.s[2]; self.s[0] ^= self.s[3]; self.s[2] ^= t; self.s[3] = self.s[3].rotate_left(45); result_plusplus } }
|
使用随机化算法的题目不是特别多,具体使用方法请看如下示例:
Submission #252682933 - Codeforces
图模板
图的表示一般为邻接表和邻接矩阵。对于邻接矩阵注意如果能用定长数组就用定长数组。而对于邻接表的数据存储可以有两种方式,一种是放在 struct 中,每次创建的时候都分配一次内存,或者将数据存储再全局变量中,动态地扩展长度。从性能测试的结果也能看出后面一种方式是更快的,由于算法题中不涉及到并发问题,所以操作全局变量是安全的,所以请尽情地使用 unsafe 吧(
图结构性能测试 · GitHub
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69
| struct PathType { from: usize, to: usize, v: i64, }
impl fmt::Display for PathType { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { write!(f, "({}, {}, {})", self.from, self.to, self.v) } } static mut PATHS: Vec<PathType> = vec![]; static mut POINT: Vec<Vec<usize>> = vec![];
struct Graph { pub start_from: usize, }
#[allow(dead_code)] impl Graph { pub fn new(p_size: usize, start_from: usize) -> Graph { unsafe { PATHS.clear(); POINT.reserve((p_size + start_from).saturating_sub(POINT.capacity())); POINT.clear(); POINT.resize_with(p_size + start_from, || vec![]); } Graph { start_from } }
pub fn add_path(&mut self, from: usize, to: usize, v: i64) { unsafe { POINT[from].push(PATHS.len()); PATHS.push(PathType { from, to, v }); } }
pub fn add_bi_path(&mut self, from: usize, to: usize, v: i64) { unsafe { POINT[from].push(PATHS.len()); PATHS.push(PathType { from, to, v }); POINT[to].push(PATHS.len()); PATHS.push(PathType { from: to, to: from, v, }); } }
pub fn get(&self, now_p: usize) -> impl Iterator<Item = &'_ PathType> { unsafe { POINT[now_p].iter().map(move |x| PATHS.get_unchecked(*x)) } } }
impl fmt::Display for Graph { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { unsafe { for now_p in self.start_from..POINT.len() { write!(f, "{} -- ", now_p); for path in self.get(now_p) { write!(f, "{} ", path)? } writeln!(f)? } } Ok(()) } }
|
使用方法参见:
Submission #252942104 - Codeforces
柯里化
柯里化在算法题中使用不多,大部分作用都是为了保存局部变量,用于传递给函数(由于 rust 不鼓励使用全局变量,所以容易导致传递较多的变量)。当然这种功能也能使用闭包实现,就请读者自行取用吧。使用方法参见:
Submission #252806831 - Codeforces
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| #[allow(unused_macros)] macro_rules! curry2 ( ($f:expr) => { |a| move |b| $f(a, b) }; );
#[allow(unused_macros)] macro_rules! curry3 ( ($f:expr) => { |a| move |b| move |c| $f(a, b, c) }; );
|
至此,请愉快地使用 rust 受苦玩耍吧(