当前位置: 首页 > news >正文

Rail开发日志_5

设定一个目标。在原有的框架上实现Rail类型。这是最最最基本的数据载体,基本得以至于拿它命名这个语言。

测试项

代码先这么写,想好期望的目标。

fn main() {test(String::from(r"
a1 = b1
a2 = b2
+ c1a3 = b3a4 = b4a5 = b5+ c2a6 = b6+ c3a7 = b7+ c4
"));
}

构建思路

into_tokens()之前不需要改动,因为词法分析和语言无关。
其实很简单,只要追踪Comment就知道代码往哪加了。这是在Rail之前构建Comment得唯一原因。
无脑乱写,最终的测试代码如下。

use std::mem;const OMMISION: &str = "/";
const CATEGORY: &str = "+";
const RAIL: &str = "=";#[derive(Debug, Clone)]
enum Line {PlaceHolder,Comment { content: String },Rail { name: String, content: String },Category { name: String },
}#[derive(Debug)]
enum Node {PlaceHolder,Comment { content: String },Rail { name: String, content: String },Domain { category: Line, rails: Vec<Node> },
}struct Root {name: String,rails: Vec<Node>,
}fn into_tokens(lines_input: &str) -> (Vec<Vec<String>>, Vec<usize>) {let mut word_current = String::new();let mut line_current = Vec::new();let mut lines_list_splited = Vec::new();let mut indent_current = 0;let mut indent_list = Vec::new();let mut indent_state_in = true;for char in lines_input.chars() {match char {'\n' => {if !word_current.is_empty() {line_current.push(word_current.clone());word_current.clear();}if !line_current.is_empty() {lines_list_splited.push(line_current.clone());indent_list.push(indent_current);}line_current.clear();indent_current = 0;indent_state_in = true;}' ' => {if indent_state_in {indent_current += 1;} else if !word_current.is_empty() {line_current.push(word_current.clone());word_current.clear();}}other => {indent_state_in = false;word_current.push(other);}}}if !word_current.is_empty() {line_current.push(word_current);}if !line_current.is_empty() {lines_list_splited.push(line_current);indent_list.push(indent_current);}let indent_list = indent_list.into_iter().map(|x| x / 2).collect();(lines_list_splited, indent_list)
}fn trim_lines(lines: &mut Vec<Vec<String>>) {let trim = |s: &str| -> String {let mut result = String::new();let mut chars = s.chars().peekable();while let Some(c) = chars.next() {if c == '\\' {match chars.peek() {Some('_') => {chars.next();result.push(' ');}Some('\\') => {chars.next();result.push('\\');}_ => result.push(c),}} else {result.push(c);}}result};for line in lines {for word in line.iter_mut() {*word = trim(word);}}
}fn into_lines(token_list: &Vec<Vec<String>>, range: usize) -> Vec<Line> {let mut local = Vec::new();for i in 0..range {let token = &token_list[i];let lenth = token.len();let data = match lenth {1 => match token[0].as_str() {OMMISION => Line::PlaceHolder,_ => continue,},2 => match token[0].as_str() {OMMISION => Line::Comment { content: token[1].clone() },CATEGORY => Line::Category { name: token[1].clone() },_ => continue,},3 => match token[1].as_str() {RAIL => Line::Rail { name: token[0].clone(), content: token[2].clone() },_ => continue}_ => continue,};local.push(data);}local
}fn into_nodes(domain_name: &str, indent_list: Vec<usize>, line_list: Vec<Line>, range: usize) -> Root {let mut root = Root { name: domain_name.to_string(), rails: Vec::new() };let mut stack: Vec<Vec<Node>> = vec![vec![]];let mut context = 0;for i in 0..range {let indent = indent_list[i];let indent_move = indent as isize - context as isize;let line = &line_list[i];if indent_move < 0 {for _ in 0..-indent_move {if let Some(child_layer) = stack.pop() {if let Some(parent_layer) = stack.last_mut() {if let Some(Node::Domain { rails, .. }) = parent_layer.last_mut() {*rails = child_layer;}}}}}let node = match line {Line::PlaceHolder => Node::PlaceHolder,Line::Comment { content } => Node::Comment { content: content.clone() },Line::Rail { name, content } => Node::Rail { name: name.clone(), content: content.clone()},Line::Category { name } => Node::Domain {category: Line::Category { name: name.clone() },rails: Vec::new(),},};if indent_move > 0 {if let Some(current_layer) = stack.last_mut() {if let Some(Node::Domain { rails, .. }) = current_layer.last_mut() {let new_layer = mem::take(rails);stack.push(new_layer);}}}if let Some(current_layer) = stack.last_mut() {current_layer.push(node);}context = indent;}while let Some(child_layer) = stack.pop() {if let Some(parent_layer) = stack.last_mut() {if let Some(Node::Domain { rails, .. }) = parent_layer.last_mut() {*rails = child_layer;}} else {root.rails = child_layer;}}root
}fn print_node(node: &Node, indent: usize) {let indent_str = "_".repeat(indent);match node {Node::PlaceHolder => println!("{indent_str}/"),Node::Comment { content } => println!("{indent_str}/ {}", content),Node::Rail { name, content } => println!("{indent_str}{} = {}", name, content),Node::Domain { category, rails } => {match category {Line::Category { name } => println!("{indent_str}+ {}", name),_ => {}}for child in rails {print_node(child, indent + 2);}}}
}fn check_nodes(root: &Root) {println!("[{}]", root.name);for node in &root.rails {print_node(node, 0);}
}fn main() {test(String::from(r"
a1 = b1
a2 = b2
+ c1a3 = b3a4 = b4a5 = b5+ c2a6 = b6+ c3a7 = b7+ c4
"));
}fn test(input: String) {let (mut line_list, indent_list) = into_tokens(&input);trim_lines(&mut line_list);let range = line_list.len();let lines = into_lines(&line_list, range);let root = into_nodes("local", indent_list, lines, range);check_nodes(&root);
}

测试结果与预期一致。

[local]
a1 = b1
a2 = b2
+ c1
__a3 = b3
__a4 = b4
__a5 = b5
__+ c2
____a6 = b6
____+ c3
__a7 = b7
__+ c4
http://www.dtcms.com/a/273976.html

相关文章:

  • 物联网与互联网融合生态
  • 鸿蒙 Secure Boot 全流程解析:从 BootROM 到内核签名验证的实战指南
  • 使用Selenium自动化获取抖音创作者平台视频数据
  • 深入解析:UPF/PGW-U如何通过PPP/L2TP隧道实现终端PAP/CHAP接入
  • [python]在drf中使用drf_spectacular
  • FPGA通信设计十问
  • 液冷智算数据中心崛起,AI算力联动PC Farm与云智算开拓新蓝海(二)
  • MyBatis02-mybatis-config.xml配置文件讲解
  • Django--02模型和管理站点
  • 链表算法之【判断链表中是否有环】
  • 从零实现一个GPT 【React + Express】--- 【3】解析markdown,处理模型记忆
  • RapidFuzz-CPP:高效字符串相似度计算的C++利器
  • ICLR 2025 | InterpGN:时间序列分类的透明革命,Shapelet+DNN双引擎驱动!
  • 【TCP/IP】18. 因特网服务质量
  • 输入流挂起
  • Promise :then 与 catch 同时触发
  • AIStarter新版重磅来袭!永久订阅限时福利抢先看
  • 精准安装追踪:openinstall 如何让邀请码绑定更智能?
  • 瑞士四种官方语言探秘:多元文化的和谐交融
  • 用Netplan配置网桥bridge笔记250711
  • 飞算 JavaAI:开启 Java 开发新时代
  • 单链表,咕咕咕
  • 使用 Python 对本地图片进行图像分类
  • 镜像(Mirror/Image)
  • 飞算JavaAI:革新Java开发的智能助手
  • 100G系列光模块产品与应用场景介绍
  • 7.12 note
  • 【实时Linux实战系列】硬实时与软实时设计模式
  • Vue3 Pinia
  • 深入MyBatis:CRUD操作与高级查询实战