算符优先分析器的构造

news2025/1/9 16:37:14

一、实验目的

(1)理解自底向上的语法分析的基本思想。
(2)理解算符优先文法的概念。
(4)掌握算符优先分析器的工作原理和工作流程。
(3)掌握算符分析表和优先函数的构造。

二、实验原理与内容

2.1实验原理

  算符优先分析法是一种简单、直观、广为使用的语法分析方法,这种方法特别适用于程序设计语言中的表达式的分析。算符优先分析法就是仿照算术表达式的运算过程而提出的一种自底向上的语法分析方法,它可以分析算符优先文法,分析的基本思想是:根据文法终结符之间的优先关系,通过比较相邻算符的优先次序来确定句型中的最左素短语,并进行归约。
  所谓的算符优先文法是指:对算符文法中任意两个终结符对a、b之间至多有一种优先关系成立的文法,而算符文法G中的任何一个产生式中都不包含两个非终结符相邻的情况。对于满足这样条件的文法,我们就可以用算符优先分析法对其进行分析。
  确定了符合要求的文法之后,自底向上的分析方法的关键就是如何在当前句型中寻找可归约的子串,算符优先分析法在归约过程中,通过终结符之间的优先关系确定当前的句型中的最左素短语,与非终结符无关,只需知道把当前句型中的最左素短语归约为一非终结符,不必知道该非终结符的名字是什么,这样也就去掉了单非终结符的归约,一旦找到最左素短语,就将它归约成一个非终结符。
  在算符优先分析法分析过程中,可以设置一个栈S,用来存放归约或者待形成最左素短语的符号串,用一个工作单元a存放当前读入的输入字符,归约成功的标志是当读入的输入字符是句子的结束符号#时,栈S中只剩下#N。

2.2实验内容

  给定一个上下文无关文法G:
<常量说明部分>一const<常量定义>{,<常量定义>)
<常量定义>一<标识符>=<无符号整数>
<变量说明部分>→{var}01<标识符>{,<标识符>}:integer;|{var}01<标识
<无符号整数>—<数字>{<数字>}
符>{,<标识符>}:real;|{var}01<标识符>{,<标识符>}:boolean;
<语句部分>—<语句>|<复合语句>
<标识符><字母>{<字母>|<数字>}
<复合语句>-begin<语句>{;begin<语句>end;}end.
<语句>一<赋值语句>|<条件语句>|<循环语句>
<赋值语句>-<标识符>:=<表达式>
<表达式>→[+1-k项>|<表达式>+<项>|<表达式>-<项>
<项>一<因子>|<项>*<因子>|<项>/<因子>
<因子>—<标识符>|<常量>|(<表达式>)
<常量>一<无符号整数>
<条件语句>→if<条件>then<语句>else<语句>
<条件>一→<表达式><关系运算符><表达式>{(and|or)<表达式>×关系运算
符×表达式}
<循环语句>→while(<条件>)((and| or)(<条件>)}do<语句>
<关系运算符>=|<|>|≤|≥
<字母>→a|b|c|…|z |A|B|C|…|Z
<数字>→0|1|2|…|9

三、实验结果

在这里插入图片描述

图3.1 文法分析演示

在这里插入图片描述

图3.2 具体分析过程

在这里插入图片描述

图3.3 LR分析表

在这里插入图片描述

图3.4 项目集和GOTO函数

四、代码实现

#include<iostream>
#include<fstream>
#include<vector>
#include<string>
#include<map>
#include<set>
#include<cstring>
#include<unordered_map>

using namespace std;

vector< vector<char> > G; //文法G[S]产生式 ,~为空字
unordered_map<char, set<char> > ts; //终结符(char)terminal symbol,及它的first集合(set<char>)
unordered_map<char, set<char> > nts; //非终结符(char)non-terminal symbol,及它的first集合(set<char>)
map< map<string, char> , string> table; //LR分析表

struct C { //闭包CLOSURE
	vector< vector<char> > project; //项目集
	vector< set<char> > outlook; //展望串
	unordered_map<char, int> go; //GO函数
};
vector<C> c; 

void show_G() {
	cout << "原文法拓广为文法G[M]:" << endl;
	for (unsigned int i = 0; i < G.size(); i++) { //输出G'[S'] 
		cout << i << ")";
		for (unsigned int j = 0; j < G[i].size(); j++) {
			if (j == 1)cout << "->";
			cout << G[i][j];
		}
		cout << endl;
	}
	cout << endl << endl;
}

void show_Symbol() {
	for (unordered_map<char, set<char> >::iterator it = nts.begin(); it != nts.end(); it++) { //输出非终结符 
		cout << it->first;
	}
	cout << endl;
	for (unordered_map<char, set<char> >::iterator it = ts.begin(); it != ts.end(); it++) { //输出终结符 
		cout << it->first;
	}
	cout << endl << endl;
}

void show_First() {
	for (auto it1 : ts) {
		cout << it1.first << ": ";
		for (auto it2 : it1.second) {
			cout << it2 << ",";
		}
		cout << endl;
	}
	cout << endl << endl;
	for (auto it1 : nts) {
		cout << it1.first << ": ";
		for (auto it2 : it1.second) {
			cout << it2 << ",";
		}
		cout << endl;
	}
	cout << endl << endl;
}

void show_Closure() {  //项目集和GO函数输出到Closure.txt文件
	fstream f("Closure.txt", ios::out);
	if (!f) {
		cout << "Closure.txt文件打开出错!" << endl;
		return;
	}
	f << "该文法的项目集和GO函数:" << endl;
	for (unsigned int i = 0; i < c.size(); i++) {
		f << "I" << i << ":" << endl;
		for (unsigned int j = 0; j < c[i].project.size(); j++) {
			for (unsigned int k = 0; k < c[i].project[j].size(); k++) {
				if (k == 1) f << "->";
				if (c[i].project[j][k] == ' ') f << "·";
				else f << c[i].project[j][k];
			}
			f << ",";
			for (auto it : c[i].outlook[j]) {
				if (it == *(c[i].outlook[j].begin())) f << it;
				else f << "/" << it;
			}
			f << endl;
		}
		for (auto it : c[i].go) {
			f <<"GO(I"<<i<<","<< it.first << ") = I" << it.second << endl;
		}
		f << endl;
	}
	cout << "已将项目集和GO函数生成到Closure.txt文件中。" << endl << endl;
}

void show_Table() {  //LR分析表输出到LR_Table.txt文件
	fstream f("LR_Table.txt",ios::out);
	if (!f) {
		cout << "LR_Table.txt文件打开出错!" << endl;
		return;
	}
	for (int i = -1; i < (int)c.size(); i++) {
		if(i==-1) f << " " << '\t';
		else f << i << '\t';
		for (auto it : ts) {
			if (i == -1) f << it.first << '\t';
			else {
				map<string, char> m;
				m[to_string(i)] = it.first;
				f<<table[m]<<'\t';
			}
		}
		if (i == -1) f  <<"#"<< '\t';
		else {
			map<string, char> m;
			m[to_string(i)] = '#';
			f << table[m] << '\t';
		}
		for (auto it : nts) {
			if (it.first == 'M') continue;
			if (i == -1)f << it.first << '\t';
			else {
				map<string, char> m;
				m[to_string(i)] = it.first;
				f << table[m] << '\t';
			}
		}
		f << endl;
	}
	f.close();
	cout << "已将LR分析表生成到LR_Table.txt文件中。" << endl << endl;
	/*for (auto it1 : table) {
		for (auto it2 : it1.first) {
			cout << it2.first << it2.second << ":"<<it1.second<<endl;
		}
	}*/
}

void read_G() { //读取文法G[S]->G'[M],并区分终结符和非终结符 
	char ch; //当前读入的字符
	int i = 0; //当前行读取的第i个字符 
	vector<char> v; //存放输入的一行产生式 
	char X; //若遇到形如X->α|β|……,用于保存X以便消除|
	set<char> m;
	nts['M'] = m;
	while (ch = getchar()) { 
		if (ch == '#') break;
		if (ch == '\n') { //换行 
			if (!v.empty())G.push_back(v);
			v.clear();
			i = 0;
			continue;
		}
		if (ch != ' ' || ch != '\t') { //去掉空格等多余字符 
			if (ch == '|') { //消除元语言符号'或|'
				G.push_back(v);
				v.clear();
				i = 3;
				v.push_back(X);
				continue;
			}
			i++;
			if (i == 1) {
				X = ch;
				nts[ch] = m; //产生式左边(第一个字符)的为非终结符 
			}
			else if (i != 2 && i != 3&&ch!='~') ts[ch] = m; //此时ts里既有非终结符又有终结符
			if (i != 2 && i != 3)v.push_back(ch); //去掉产生式的-> 
		}
	}
	if (G.empty()) exit(0);

	//加入新树根M
	v.clear();
	v.push_back('M');
	v.push_back(G[0][0]);
	G.insert(G.begin(), v);

	//去掉ts中的非终结符
	for (unordered_map<char, set<char> >::iterator it = nts.begin(); it != nts.end(); it++) {
		unordered_map<char, set<char> >::iterator iter;
		iter = ts.find(it->first);
		if (iter != ts.end())ts.erase(iter);
	}
}

void get_First() { //得到First集合
	for (auto &it : ts) it.second.insert(it.first);//终结符的first集合是它自己
	
	//求非终结符的First集合
	int r = 0;
	int change = 1;
	while (change) {
		if (r == 20)break;
		r++;
		change = 0;
		for (auto &it : nts) { //对每个非终结符
			for (unsigned int i = 0; i < G.size(); i++) { //遍历产生式
				if (G[i][0] == it.first) {
					unsigned int size = it.second.size(); //操作前First(X)的大小
					unordered_map<char, set<char> >::iterator iter=ts.find(G[i][1]);
					if (ts.find(G[i][1]) != ts.end()||G[i][1]=='~') { //形如X->a……或X->空字,把a加入First(X)中
						it.second.insert(G[i][1]);
						if (it.second.size() > size) change = 1;
					}
					else {  //形如X->Y……,把First(Y)加入First(X)
						unsigned int col = 1;
						while(1){ //若X->Y1Y2……,循环把First(Y)加入First(X)
							int flag = 0; //标记当前First(Y)中是否有空字
							unordered_map<char, set<char> >::iterator itt= nts.find(G[i][col]);
							for(auto &iter:itt->second){ //遍历First(Y)
								if (iter == '~') flag = 1;
								else it.second.insert(iter);
							}
							if (flag) {
								col++;
								if (G[i].size() <= col) {
									it.second.insert('~'); //形如X->Y,将空字加入First(X)
									break;
								}
								else if (ts.find(G[i][col]) != ts.end()) { //形如X->Ya……,将a加入First(X)
									it.second.insert(G[i][col]);
									break;
								}
								else { //形如X->YZ……,将First(Z)加入First(X)
								}
							}
							else break;
						}
						if (it.second.size() > size) change = 1;
					}
				}
			}
		}
	}
}

void get_Closure() {  //计算CLOSURE,包括GO
	int i = 0; //闭包编号
	C clo; //生成第一个闭包(I0)
	c.push_back(clo);
	while(1) {
		if (i == c.size()) break; //没有新闭包,跳出循环(即已获得全部闭包及项目)
		if (i == 0) { //确定项目集I0的第一个项目
			vector<char> v(G[0]);
			v.insert(v.begin() + 1, ' ');
			c[i].project.push_back(v);
			set<char> m;
			m.insert('#');
			c[i].outlook.push_back(m);
		}
		for (unsigned int j = 0; j < c[i].project.size(); j++){  //遍历已有项目,生成该闭包所有项目
			for (unsigned int k = 0; k < c[i].project[j].size(); k++) {  //扫描单个项目,找到当前位置·(这里用空格代替)
				if (c[i].project[j][k] == ' ') {
					if (k == c[i].project[j].size() - 1) break;  //形如X->β·,不会生成新的项目
					for (unsigned int x = 0; x < G.size(); x++) { //形如X->α·Yβ,  遍历G'[M],查找所有对应的产生式,以求出新的项目并加入项目集
						if (G[x][0] == c[i].project[j][k + 1]) { //对应的产生式
							vector<char> v(G[x]); //用于保存新项目
							v.insert(v.begin() + 1, ' '); //计算新项目
							int exist = 0; //标记该新项目是否已存在
							for (unsigned int y = 0; y < c[i].project.size(); y++) { //遍历已有项目,判断是新项目还是已有项目
								if (c[i].project[y] == v) { //已有项目,只需保存项目下标(用于添加新的展望串)
									exist = y; 
									break;
								}
							}
							if(exist==0) c[i].project.push_back(v); //新项目,加入项目集
							set<char> m; //用于保存新展望串
							//形如【形如X->α·Yβ,a】,计算展望串,即计算First(βa)
							//情况一:β为空字,First(βa)=a
							//情况二:β中第一个字符为结束符b,First(βa)=b
							//情况三:β中第一个字符为非结束符B,若First(B)中没有空字,First(βa)=First(B);
							//                                   若First(B)中包含空字,First(βa)=First(B)+First(γa),其中γ为β除去第一个字符后形成的符号串
							bool kong = true; //标记情况三B中是否有空字
							int t = 0; //表示当前符号串γ为β除去第t个字符
							while (kong) { //若为情况三且B含空字,计算First(γa)
								kong = false;
								if (k + t + 1 == c[i].project[j].size() - 1) {  //情况一
									for (auto it : c[i].outlook[j]) m.insert(it);
								}
								else if (ts.find(c[i].project[j][k + t + 2]) != ts.end()) {  //情况二
									m.insert(c[i].project[j][k + 2 + t]);
								}
								else {  //情况三
									set<char> m1((nts.find(c[i].project[j][k + 2 + t]))->second);
									for (auto it : m1) {
										if (it== '~') {  //含空字
											kong = true;
											t++;
										}
										else { //不含空字
											m.insert(it);
										}
									}
								}
							}
							if (exist) //已存在项目,将新展望串加入原展望串中
							{
								for (auto it : m) {
									c[i].outlook[exist].insert(it);
								}
							}
							else c[i].outlook.push_back(m); //新项目,新展望串加入展望串集
						}
					}
					break;
				}
			}
		}

		for (unsigned int j = 0; j < c[i].project.size(); j++) {  //遍历本闭包的所有项目,计算GO函数,并生成新的闭包
			for (unsigned int k = 0; k < c[i].project[j].size(); k++) {  //扫描单个项目,找到当前位置·(这里用空格代替)
				if (c[i].project[j][k] == ' ') {
					if (k == c[i].project[j].size() - 1) break; //形如【X->β·】,不会生成新闭包
					//计算GO函数,并生成新的闭包
					vector<char> new_closure_pro(c[i].project[j]); //用于保存新项目
					new_closure_pro[k] = new_closure_pro[k + 1]; //计算新项目
					new_closure_pro[k + 1] = ' ';
					set<char> new_closure_search(c[i].outlook[j]); //用于保存新展望串
					bool dif = false; //标记 生成的新项目是否属于已有的闭包(判断是否需生成新闭包)
					for (unsigned int x = 0; x < c.size(); x++) { //遍历已有闭包
						//dif = false;
						for (unsigned int y = 0; y < c[x].project.size(); y++) { //将新项目和新展望串与已有闭包的所有项目比较,若相同,则属于已有闭包,无需生成新闭包
							dif = false;
							if (new_closure_pro == c[x].project[y]) { 
								if (c[x].outlook[0].size() != new_closure_search.size()) { //比较展望串长
									dif = true;
									continue;
								}
								auto iter = c[x].outlook[0].begin();
								for (auto it : new_closure_search) {  //比较展望串每个字符
									if (it!= *iter) {
										dif = true;
										break;
									}
									iter++;
								}
								if (dif == false) { //属于已有闭包,只计算GO函数
									c[i].go[new_closure_pro[k]] = x;
									break;
								}
							}
							else dif = true;
							if (dif == false) break;
						}
						if (dif==false) break;
					}
					if (c[i].go.count(new_closure_pro[k]) != 0 && dif) { //若有多个形如【X->α·Yβ】的产生式(相互不一样),它们有同一GO函数(即属于同一闭包),但生成的新项目不一样
						c[c[i].go[new_closure_pro[k]]].project.push_back(new_closure_pro); //只需将新项目及新展望串加入GO对应的闭包
						c[c[i].go[new_closure_pro[k]]].outlook.push_back(new_closure_search);
						break;
					}
					if (dif) { //不属于已有闭包,生成新闭包,并计算GO函数
						C new_closure;
						new_closure.project.push_back(new_closure_pro);
						new_closure.outlook.push_back(new_closure_search);
						c.push_back(new_closure);
						c[i].go[new_closure_pro[k]] = c.size() - 1;
					}
				}
			}
		}
		i++; //下一闭包
	}
}

int get_Table() { //由CLOSURE计算LR(1)分析表table
	for (unsigned int i = 0; i < c.size(); i++) { //遍历所有闭包
		for (unsigned int j = 0; j < c[i].project.size(); j++) { //遍历每个闭包中的所有项目
			for (unsigned int k = 0; k < c[i].project[j].size(); k++) { //扫描该项目,找到当前位置
				if (c[i].project[j][k] == ' ') {
					//cout << 1 << endl;
					if (k== c[i].project[j].size()-1) { //形如 【X->α·,β】,归约/acc
						if (c[i].project[j][0] == 'M'){  //形如 【M->X·,#】,令table[i,#]=acc
							map<string, char> m;
							m[to_string(i)] = '#';
							if (table.find(m) != table.end() && table[m] != "acc") {
								cout << "该文法不是LR(1)文法,存在多重定义入口!" << endl;
								return 0;
							}
							else table[m] = "acc";
						}
						else { //形如 【X->α·,a】,归约,令table[i,a]=rj
							int id;
							for (unsigned int x = 0; x < G.size(); x++) { //扫描G'[M]找到对应的产生式的编号
								vector<char> v(c[i].project[j]);
								v.pop_back();
								if (G[x] == v) {
									id = x;
									break;
								}
							}
							for (auto it : c[i].outlook[j]) {
								map<string, char> m;
								m[to_string(i)] = it;
								if (table.find(m) != table.end() && table[m] != (string)"r" + to_string(id)) {
									cout << "该文法不是LR(1)文法,存在多重定义入口!" << endl;
									return 0;
								}
								else table[m] = (string)"r" + to_string(id);
							}
						}
					}
					else{ //形如 【X->α·β,γ】
						char next = c[i].project[j][k + 1];
						if (ts.find(next) != ts.end()) {  //形如 【X->α·aβ,γ】,令table[i,a]=sj
							map<string, char> m;
							m[to_string(i)] = next;
							if (table.find(m) != table.end()&&table[m]!=(string)"s"+to_string(c[i].go[next])) {
								cout << "该文法不是LR(1)文法,存在多重定义入口!" << endl;
								return 0;
							}
							else table[m] = (string)"s" + to_string(c[i].go[next]);
						}
						else { //形如 【X->α·Yβ,γ】,令table[i,Y]=j
							map<string, char> m;
							m[to_string(i)] = next;
							if (table.find(m) != table.end() && table[m] != to_string(c[i].go[next])) {
								cout << "该文法不是LR(1)文法,存在多重定义入口!" << endl;
								return 0;
							}
							else table[m] = to_string(c[i].go[next]);
						}
					}
					break;
				}
			}
		}
	}
	return 1;
}

int check(int time) { //输入并分析语句sentence并输出分析过程,成功分析返回1,否则返回0
	string sentence = "\0"; //需要分析的语句
	cin >> sentence;
	fstream f("sentence_"+to_string(time)+".txt", ios::out); //分析过程输出到该文件
	if (!f) {
		cout << "无法打开文件sentence.txt" << endl;
	}
	f << "步骤\t\t状态栈\t\t符号栈\t\t输入串\t\t动作说明" << endl;
	vector<string> status; //状态栈
	vector<char> symbol; //符号栈
	int step = 1; //步骤数
	if (sentence[0] != '#') {
		cout << "语句应以#开头,并以#结束" << endl;
		return 0;
	}
	symbol.push_back('#');
	sentence = sentence.substr(1);
	status.push_back("0");
	while (1) {
		f << step++ << "\t\t"; //输出步骤数
		for (unsigned int i = 0; i < status.size(); i++) //输出状态栈
			if (i == 0) f << status[i];
			else f << " " << status[i];
		f << "\t\t";
		for (unsigned int i = 0; i < symbol.size(); i++) //输出符号栈
			if (i == 0) f << symbol[i];
			else f << " " << symbol[i];
		f << "\t\t";
		for (unsigned int i = 0; i < sentence.length(); i++) //输出输入串
			if (i == 0) f << sentence[i];
			else f << " " << sentence[i];
		f << "\t\t";
		string cur_status = status[status.size() - 1]; //当前状态
		char cur_symbol = sentence[0]; //当前“展望”字符
		string new_status; //下一入栈的新状态
		map<string, char> m;
		m[cur_status] = cur_symbol;
		new_status = table[m];
		if (new_status == "acc") { 
			cout << "分析成功,该语句合法!(具体分析过程请查看文件sentence_" << time << ".txt)" << endl << endl;
			f << "acc:分析成功" << endl;
			return 1; 
		}
		else if (new_status[0] == 's') { //形如 table[i,b]=sj,状态sj入栈,并读入一个字符
			status.push_back(new_status.substr(1)); //入栈的状态要去掉第一个字符‘s’
			symbol.push_back(cur_symbol); //读入一个字符
			sentence = sentence.substr(1);
			f << "ACTION[" << cur_status << "," << cur_symbol << "]=" << new_status << ",即状态" << new_status << "入栈" << endl;
		}
		else if (new_status[0] == 'r') { //形如 table[i,b]=rj,用产生式G(j)归约,且table[x,y]入栈
			new_status = new_status.substr(1); //去掉‘r’
			f << "r" << new_status << ":用";
			int gid = atoi(new_status.c_str()); //计算产生式编号
			int len = G[gid].size() - 1; //被归约的字符串长度
			char reduced_symbol = G[gid][0]; //归约得到的非终结符
			for (int i = 0; i < len; i++) {
				status.pop_back(); //归约,即去掉栈顶的len个状态项
				symbol.pop_back();
			}
			map<string, char> m;
			m[status[status.size() - 1]] = reduced_symbol;
			new_status = table[m];
			status.push_back(new_status);
			symbol.push_back(reduced_symbol);
			for (unsigned int i = 0; i < G[gid].size(); i++){
				if (i == 1) f << "->";
				f << G[gid][i];
			}
			f << "归约,且GOTO[" << status[status.size() - 1] << "," << reduced_symbol << "]=" << new_status << "入栈" << endl;
		}
		else {
			cout << "该语句有语法错误!(详情请查看分析过程文件sentence_" << time << ".txt)" << endl << endl;
			return 0;
		}
	}
	f.close();
	cout << "该语句有语法错误!(详情请查看分析过程文件sentence_" << time << ".txt)" << endl << endl;
	return 0;
}

int main() {
	cout << "请输入文法(以单行#结束):" << endl;
	read_G();
	cout << endl;
	show_G();

	get_First();

	get_Closure();
	show_Closure();

	if (get_Table()) {
		show_Table();
		cout << endl;

		cout << "请输入需分析的语句(以#开头,并以#结束):" << endl;
		int time = 0;
		while (1) {
			check(++time);
		}
	}
	system("pause");
}





本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/86653.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

函数式接口

Lambda表达式的本质&#xff1a;作为函数式接口的实例 如果一个接口中&#xff0c;只声明一个抽象方法&#xff0c;则此接口就称为函数式接口 FunctionalInnterface public interface MyInterface{void method1(); }要想用Lambda表达式就一定要在函数式接口的条件下使用 相当于…

微信小程序直播状态接口如何获取

现如今&#xff0c;小程序直播非常的红火&#xff0c;越来越多的商家开通了微信小程序直播&#xff0c;但是在直播的过程中&#xff0c;偶尔会出现一些小问题&#xff0c;如禁播&#xff0c;异常状态等等&#xff0c;下面小编就来介绍一下微信小程序直播状态接口如何获取。 一、…

echarts5.4立体柱状图

资源下载&#xff1a;https://www.jsdelivr.com/package/npm/echarts 效果图&#xff1a; 借鉴资源&#xff1a;echarts 如何绘制三维 3D 立体柱状图 - 简书 代码示例&#xff1a; <!DOCTYPE html> <head><meta charset"utf-8"><title>ECh…

基于51单片机的智能小车系统设计

原理图&#xff1a; 程序运行图&#xff1a; 部分程序&#xff1a; /******************************************************************************* * 文件名称&#xff1a;main.c * 说明&#xff1a;本文件为小车控制的主函数 * 功能&…

分布式websocket探索

单体式架构 根据基于golang的gin框架开发的web项目所展开 如果一个Web项目采用单体式架构且配备了websocket通讯的功能&#xff0c;那么在单个实例中是能够正常运行的 在我的项目中&#xff0c;用户可以通过websocket来进行实时通讯和实时消息通知&#xff0c;同时如果在web业务…

AcrelEMS-IDC数据中心综合能效管理系统解决方案-Susie 周

1、概述 安科瑞电气紧跟数据中心发展形式&#xff0c;推出AcrelEMS-IDC数据中心综合能效管理解决方案&#xff0c;包含有电力监控、动环监控、消防监控、能耗统计分析、智能照明控制以及新能源监测几个子系统。集成了变配电监测、电源备自投、电气接点测温、智能照明控制、电能…

yearning搭建及使用

yearning搭建及使用 数据库审计管理&#xff0c;是数据安全规范中不可或缺的一环&#xff0c;通过审计管理我们能够把控、追溯sql执行情况。yearning作为一款开源的数据库审计软件&#xff0c;是我们开发运维工作中经常打交道的一个“伙伴”。 yearning提供的核心功能就是sql…

mysql 自增字段、属性

mysql自增属性 参考文章 https://www.php.cn/mysql-tutorials-489209.html https://blog.csdn.net/qq_41045806/article/details/108310772 在Mysql中&#xff0c;可以为某一属性设置自增属性&#xff0c;可以很好地为我们解决属性值重复的问题。 在mysql中&#xff0c;使用au…

DevExpress Universal全面的软件开发包

DevExpress Universal全面的软件开发包 DevExpress Universal帮助您使用所有DevExpress单平台控件等为Windows、Web、移动和平板电脑构建应用程序。它包括桌面控件(WinForms、WPF、UWP和桌面报告)、Web控件(ASP.NET、ASP.NET MVC和Core、Bootstrap Web Forms、JavaScript-jQuer…

第十章用Python获取sqlite、MySQL、Excel、csv、json中的数据

这里写目录标题项目背景获取sqlite3中的数据sqlite3库获取sqlite数据pandas库获取sqlite数据获取MySQL中的数据pymsql库获取MySQL数据pandas库获取mysql数据获取Excel中的数据xlrd库获取Excel数据pandas库获取Excel数据获取csv中的数据csv库读取csv数据pandas读取csv数据获取js…

Docker02(数据卷)

目录 一、宿主机与容器之间的文件拷贝 二、数据卷 三、数据卷容器 四、Dockerfile Dockerfile简介 自定义centos&#xff0c;具备vim及ifconfig作用 自定义tomcat8 一、宿主机与容器之间的文件拷贝 在生产环境中使用 Docker &#xff0c;往往需要对数据进行持久化&#…

Spring Cloud微服务治理框架深度解析

在学习一个技术之前&#xff0c;首先我们要了解它是做什么的&#xff0c;我们为什么要用它。不然看再多资料都理解不了&#xff0c;因此我们先来讲解下Spring Cloud Spring Cloud是一套微服务治理框架&#xff0c;几乎考虑到了微服务治理的方方面面。那么接下来具体说下 Spring…

WebRTC客户端主要流程分析

1.通信过程 因为WebRTC规范里没有包含信令协议&#xff0c;所以像OWT、mediasoup等支持WebRTC的开源项目&#xff0c;其通信两端建立连接的过程中的信令逻辑各不相同。但是&#xff0c;总体上来说&#xff0c;其通信过程必然会包括以下过程。 发起端创建本地的PeerConnection&…

基于微信小程序的校友录系统-计算机毕业设计

项目介绍 本系统采用微信开发者开发、结合后台java语言以及Mysql数据库等技术。系统主要分为管理员和用户、校友三部分&#xff0c;管理员服务端&#xff1a;首页、个人中心、用户管理、校友管理、校友风采管理、校友视频管理、视频分类管理、班级信息管理、留言板管理、论坛交…

虹科方案|适用于VMware vSphere®环境的Mac Pro®和微型服务器存储连接

一、前言 Thunderbolt支持ATTO的VMWare ESXi和ThunderLink产品线&#xff0c;使我们能够创建基于Mac的vSphere设置&#xff0c;从而能够为我们的macOS服务器提供虚拟化服务。将虚拟硬件、快照和Veeam备份与macOS服务器的简单性相结合&#xff0c;将改变SMB市场的游戏规则。 二…

3y开发都不的不写单元测试,然后被被批了

他是3y&#xff0c;一年CRUD经验用十年的markdown程序员&#x1f468;&#x1f3fb;‍&#x1f4bb;常年被誉为职业八股文选手 最近在看单元测试的东西&#xff0c;想跟大家聊聊我的感受。单元测试这块说实在的&#xff0c;我并不太熟悉&#xff0c;我几乎不写单元测试&#x…

基于Sovit3D可视化开发的智慧港口数字孪生系统

港口作为经济的晴雨表&#xff0c;是现代经济的血液&#xff0c;是促进贸易和地区发展的重要基础设施。全球贸易中约90%的贸易由海运业承载&#xff0c;港口是其中重要一环。 建设背景 作为“21世纪海上丝绸之路”的重要节点&#xff0c;港口在“一带一路”建设中有着举足轻重…

nginx下载与安装

文章目录一.下载二.安装1.将压缩包上传到我们的服务器 /opt 目录下2.安装编译文件需要的c3.解压压缩包&#xff08;首先需要进入 /opt 目录下 即压缩包文件所在目录&#xff09;4.将解压后的文件夹剪切到指定目录 nginx ,这里我们放在了 /usr/local 目录下5.进入nginx文件目录6…

CentOS 7.9升级内核(kernel-5.4.218)

记录&#xff1a;362 场景&#xff1a;在CentOS 7.9操作系统上&#xff0c;使用ELRepo镜像仓库中kernel包&#xff0c;升级Linux 3.10.0-1160.el7.x86_64内核到kernel-5.4.218。ELRepo项目专注于Linux和CentOS操作系统的硬件相关的软件包。 版本&#xff1a; 操作系统&#…

自动化测试框架如果都总结成这样,人人都能学好

1、自动化测试框架设计的核心&#xff0c;可以概括为解决以下这些问题&#xff1a; 1)测试数据要怎么准备&#xff0c;如何在框架中实现&#xff0c;是否可复用。 2)测试用例的批量导入。 3)用例之间是否存在逻辑关系&#xff0c;相互之间是否有影响&#xff0c;在框架中该如何…