首页 语法分析器实验报告

语法分析器实验报告

举报
开通vip

语法分析器实验报告本文档如对你有帮助,请帮忙下载支持!语法分析器的设计实验报告一、实验内容语法分析程序用LL(1)语法分析方法。首先输入定义好的文法书写文件(所用的文法可以用LL(1)分析),先求出所输入的文法的每个非终结符是否能推出空,再分别计算非终结符号的FIRST集合,每个非终结符号的FOLLOW集合,以及每个规则的SELECT集合,并判断任意一个非终结符号的任意两个规则的SELECT集的交集是不是都为空,如果是,则输入文法符合LL(1)文法,可以进行分析。对于文法:G[E]:E->E+T|TT->T*F|FF->i|(E)分...

语法分析器实验报告
本文档如对你有帮助,请帮忙下载支持!语法分析器的设计 实验报告 化学实验报告单总流体力学实验报告观察种子结构实验报告观察种子结构实验报告单观察种子的结构实验报告单 一、实验内容语法分析程序用LL(1)语法分析方法。首先输入定义好的文法书写文件(所用的文法可以用LL(1)分析),先求出所输入的文法的每个非终结符是否能推出空,再分别计算非终结符号的FIRST集合,每个非终结符号的FOLLOW集合,以及每个规则的SELECT集合,并判断任意一个非终结符号的任意两个规则的SELECT集的交集是不是都为空,如果是,则输入文法符合LL(1)文法,可以进行分析。对于文法:G[E]:E->E+T|TT->T*F|FF->i|(E)分析句子i+i*i是否符合文法。二、基本 思想 教师资格思想品德鉴定表下载浅论红楼梦的主题思想员工思想动态调查问卷论语教育思想学生思想教育讲话稿 1、语法分析器实现语法分析是编译过程的核心部分,它的主要任务是按照程序的语法规则,从由词法分析输出的源程序符号串中识别出各类语法成分,同时进行词法检查,为语义分析和代码生成作准备。这里采用自顶向下的LL(1)分析方法。语法分析程序的 流程 快递问题件怎么处理流程河南自建厂房流程下载关于规范招聘需求审批流程制作流程表下载邮件下载流程设计 图如图5-4所示。开始读入文法有效?是LL(1)文法?判断句型结束报错语法分析程序流程图该程序可分为如下几步:(1)读入文法(2)判断正误(3)若无误,判断是否为LL(1)文法(4)若是,构造分析表;(5)由句型判别算法判断输入符号串是为该文法的句型。三、核心思想该分析程序有15部分组成:(1)首先定义各种需要用到的常量和变量;(2)判断一个字符是否在指定字符串中;本文档如对你有帮助,请帮忙下载支持!(3)读入一个文法;(4)将单个符号或符号串并入另一符号串;(5)求所有能直接推出&的符号;(6)求某一符号能否推出‘&’;(7)判断读入的文法是否正确;(8)求单个符号的FIRST;(9)求各产生式右部的FIRST;(10)求各产生式左部的FOLLOW;(11)判断读入文法是否为一个LL(1)文法;(12)构造分析表M;(13)句型判别算法;(14)一个用户调用函数;(15)主函数;下面是其中几部分程序段的算法思想:1、求能推出空的非终结符集Ⅰ、实例中求直接推出空的empty集的算法描述如下:voidemp(charc){参数c为空符号chartemp[10];定义临时数组inti;for(i=0;i<=count-1;i++)从文法的第一个产生式开始查找{if产生式右部第一个符号是空符号并且右部长度为then将该条产生式左部符号保存在临时数组将临时数组中的元素合并到记录可推出1,temp中&符号的数组empty中。}Ⅱ、求某一符号能否推出'&'int_emp(charc){//若能推出inti,j,k,result=1,mark=0;chartemp[20];temp[0]=c;temp[1]='\0';&,返回1;否则,返回0存放到一个临时数组empt里,标识此字符已查找其是否可推出空字如果c在可直接推出空字的empty[]中,返回1for(i=0;;i++){if(i==count)return(0);找一个左部为j=strlen(right[i]);c的产生式//j为c所在产生式右部的长度ifif右部长度为右部长度为1且右部第一个字符在1但第一个字符为终结符empty[]中.then返回1(A->B,B可推出空,then返回0(A->a,a为终结符))else{本文档如对你有帮助,请帮忙下载支持!for(k=0;k<=j-1;k++){查找临时数组empt[].并标记mark-=1(A->AB)if找到的字符与当前字符相同(A->AB)结束本次循环else(mark等于0)查找右部符号是否可推出空字把当前符号加入到临时数组,把返回值赋给empt[]里.result}if当前字符不能推出空字且还没搜索完全部的产生式then跳出本次循环继续搜索下一条产生式elseif//当前字符可推出空字,返回1}}}2、计算每个符号的first集:实例中求单个符号的FIRST集的算法描述如下:voidfirst2(inti){参数i为符号在所有输入符号中的序号c等于指示器i所指向的符号在保存终结符元素的termin[]数组查找ifc为终结符(c∈VT),thencFIRST(c)={c}在保存终结符元素的non_ter[]数组查找cifc是非终结符(c∈VN)在所有产生式中查找c所在的产生式if产生式右部第一个字符为终结符或空(即c→a(a∈VT)或c→&)把a或&加进FIRST(c)if产生式右部第一个字符为非终结符thenif产生式右部的第一个符号等于当前字符then跳到下一条产生式进行查找求当前非终结符在所有字符集中的位置if当前非终结符还没求其FIRST集then查找它的FIRST集并标识此符号已求其FIRST集求得结果并入到c的FIRST集.if当前产生式右部符号可推出空字且当前字符不是右部的最后一个字符thenthen获取右部符号下一个字符在所有字符集中的位置if此字符的FIRST集还未查找then找其FIRST集,并标其查找状态为1把求得的FIRST集并入到c的FIRST集.if当前右部符号串可推出空且是右部符号串的最后一个字符Y1Y2Yk,若对一切1<=i<=k,均有&∈FIRST(Yi),则将(即产生式为c→&∈符号加进FIRST(c))then把空字加入到当前字符c的FIRST集.本文档如对你有帮助,请帮忙下载支持!else不能推出空字则结束循环标识当前字符c已查找其FIRST集.}3.计算FOLLOW集FOLLOW集的构造可用如下方法来求:对于文法中的符号XVN,其FOLLOW(A)集合可反复应用下列规则计算,直到FOLLOW(A)集合不再增大为止。(1)对于文法开始符号S,因为SS,故#FOLLOW(S);(2)若A→B,其中BVN,(VTVN)*、(VTVN)+,则FIRST()-{}FOLLOW(B);(3)若A→B或A→B(),则FOLLOW(A)FOLLOW(B)。FOLLOW集的算法描述如下:voidFOLLOW(inti)为待求的非终结符把当前字符放到一临时数组foll[]中,标识求已求其FOLLOW集.避免循环递归ifX为开始符号then#∈FOLLOW(X)对全部的产生式找一个右部含有当前字符注:比如求FOLLOW(B)则找A→αXX的产生式或A→X(ε)的产生式ifX在产生式右部的最后(形如产生式查找非终结符A是否已经求过其A→X)FOLLOWthen集.避免循环递归if非终结符A已求过其FOLLOWFOLLOW(A)∈FOLLOW(X)集then继续查下一条产生式是否含有Xelse求A之FOLLOW集,并标识为A已求其FOLLOW集elseififX不在产生式右部的最后(形如A→B)then右部X后面的符号串能推出空字then查找是否已经求过其FOLLOW集.避免循环递归if已求过的FOLLOW集thenFOLLOW(A)∈FOLLOW(B)结束本次循环elseif不能推出空字then求FIRST()把FIRST()中所有非空元素加入到FOLLOW(B)中标识当前要求的非终结符X的FOLLOW集已求过4.计算SELECT集SELECT集的构造算法如下:对所有的规则产生式A→x:(1)若x不能推出空字(2)若x可推出空字,则SELECT(A→x)=FIRST(x);,则SELECT(A→x)=FIRST(x)–{}FOLLOW(A)。算法描述如下:for(i=0;i<=产生式总数-1;i++)先把当前产生式右部的FIRST集(一切非空元素,不包括ε)放入到当前产生式的本文档如对你有帮助,请帮忙下载支持!SELECT(i);if产生式右部符号串可推出空字then把i指向的当前产生式左部的非终结符号的FOLLOW集并入到SELECT(i)中判断是否LL(1)文法要判断是否为LL(1)文法,需要输入的文法G有如下要求:具有相同左部的规则的SELECT集两两不相交,即:SELECT(A→)∩SELECT(A→)=如果输入的文法都符合以上的要求,则该文法可以用算法描述如下:LL(1)方法分析。把第一条产生式的SELECT(0)for(i=1;i<=产生式总数-1;i++)集放到一个临时数组temp[]中求temp的长度lengthifi指向的当前产生式的左部等于上一条产生式的左部then把SELECT(i)并入到temp数组中Iftemp的长度小于length加上SELECT(i)的长度返回0else把temp清空把SELECT(i)存放到temp中结果返回1;四、算法#include#include#include/*******************************************/intcount=0;//产生式的个数intnumber;//所有终结符和非终结符的总数charstart;//开始符号chartermin[50];//终结符号charnon_ter[50];//非终结符号charv[50];//所有符号charleft[50];//左部charright[50][50];//右部charfirst[50][50],follow[50][50];//各产生式右部的FIRST和左部的FOLLOW集合charfirst1[50][50];//所有单个符号的FIRST集合charselect[50][50];//各个产生式的SELECT集合charfirstflag[50],followflag[50];//记录各符号的FIRST和FOLLOW是否已求过charempty[20];//记录可推出&的符号charnonempty[20];//记录不可推出&的符号charempt[20];//求_emp()时使用charTEMP[50];//求FOLLOW时存放某一符号串的FIRST集合intvalidity=1;//表示输入文法是否有效intll=1;//表示输入文法是否为LL(1)文法intM[20][20];//分析表本文档如对你有帮助,请帮忙下载支持!charchoose;charfoll[20];//用户输入时使用//求FOLLOW集合时使用/*******************************************判断一个字符c是否在指定字符串p中********************************************/intin(charc,char*p){inti;if(strlen(p)==0)return(0);for(i=0;;i++){if(p[i]==c)return(1);//若在,返回1if(i==(int)strlen(p))return(0);//若不在,返回0}}/*******************************************将单个符号或符号串并入另一符号串********************************************/voidmerge(char*d,char*s,inttype){//是目标符号串,s是源串,type=1,源串中的'&'一并并入目串;//type=2,源串中的'&'不并入目串inti,j;for(i=0;i<=(int)strlen(s)-1;i++){if(type==2&&s[i]=='&');else{for(j=0;;j++){if(j<(int)strlen(d)&&s[i]==d[j])break;//若已存在,则退出,继续看下一个源串字符if(j==(int)strlen(d))//若不存在,则并入{d[j]=s[i];d[j+1]='\0';break;}}}}}本文档如对你有帮助,请帮忙下载支持!/*******************************************读入一个文法********************************************/chargrammer(char*t,char*n,char*left,charright[50][50]){charvn[50],vt[50];chars;charp[50][50];inti,j;printf("请输入文法的非终结符号串:");scanf("%s",vn);getchar();i=strlen(vn);memcpy(n,vn,i);n[i]='\0';printf("请输入文法的终结符号串:");scanf("%s",vt);getchar();i=strlen(vt);memcpy(t,vt,i);t[i]='\0';printf("请输入文法的开始符号:");scanf("%c",&s);getchar();printf("请输入文法产生式的条数:");scanf("%d",&i);getchar();count=i;for(j=1;j<=i;j++){printf("请输入文法的第%d条(共%d条)产生式:",j,i);scanf("%s",p[j-1]);getchar();}for(j=0;j<=i-1;j++){if(p[j][1]!='-'||p[j][2]!='>')//检测输入错误{printf("\n输入错误!");validity=0;return('\0');}}return(s);本文档如对你有帮助,请帮忙下载支持!}/*******************************************判断读入的文法是否正确********************************************/intjudge(){inti,j;for(i=0;i<=count-1;i++){if(in(left[i],non_ter)==0){//若左部不在非终结符中,报错printf("\n文法左部出错!");validity=0;return(0);}for(j=0;j<=(int)strlen(right[i])-1;j++){if(in(right[i][j],non_ter)==0&&in(right[i][j],termin)==0&&right[i][j]!='&'){//若右部某一符号不在非终结符、终结符中且不为'&',报错printf("\n文法右部出错!");validity=0;return(0);}}}return(1);}/*******************************************求所有能直接推出&的符号********************************************/voidemp(charc){chartemp[10];inti;for(i=0;i<=count-1;i++){if(right[i][0]==c&&strlen(right[i])==1){temp[0]=left[i];temp[1]='\0';merge(empty,temp,1);//求所有能直接推出'&"的符号,结果保存到empty[]中emp(left[i]);}}本文档如对你有帮助,请帮忙下载支持!}/*******************************************求某一符号能否推出'&'********************************************/int_emp(charc){//若能推出&,返回1;否则,返回0inti,j,k,result=1,mark=0;chartemp[20];temp[0]=c;temp[1]='\0';merge(empt,temp,1);//存放到一个临时数组empt里,标识此字符已查找其是否可推出空字if(in(c,empty)==1)//如果c在可直接推出空字的empty[]中,返回1return(1);for(i=0;;i++){if(i==count)return(0);if(left[i]==c)//找一个左部为c的产生式{j=strlen(right[i]);//j为c所在产生式右部的长度if(j==1&&in(right[i][0],empty)==1)//右部长度为1且右部第一个字符在empty[]中.返回1(A->B,B可推出空)return(1);elseif(j==1&&in(right[i][0],termin)==1)//右部长度为1但第一个字符为终结符,返回0(A->a,a为终结符)continue;else{for(k=0;k<=j-1;k++){if(in(right[i][k],empt)==1)//查找临时数组empt[].(A->AB)mark=1;}if(mark==1)//找到的字符与当前字符相同(A->AB)continue;//结束本次循环else//(mark等于0){for(k=0;k<=j-1;k++){result*=_emp(right[i][k]);//递归调用,查找右部符号是否可推出空字,把返回值赋给resulttemp[0]=right[i][k];temp[1]='\0';merge(empt,temp,1);//把当前符号加入到临时数组empt[]里,标记本文档如对你有帮助,请帮忙下载支持!已查找}}}if(result==0&&i=0)//i不为-1时是产生式的序号{first[i][0]='&';//把"&"加入到当前符号串的FIRST集first[i][1]='\0';}else//i为-1时,表示求FOLLOW时用到的产生式右部的FIRST集,保存在TEMP[]中{TEMP[0]='&';TEMP[1]='\0';}}else//右部符号串字符不为"&"空字{for(j=0;;j++){if(v[j]==p[0])//求右部符号的第一个字符p[0]在所有字符集中的位置jbreak;}if(i>=0){本文档如对你有帮助,请帮忙下载支持!memcpy(first[i],first1[j],strlen(first1[j]));//把j所指向的单个符号的FIRST集拷贝到该右部符号串的FIRST集first[i][strlen(first1[j])]='\0';}else{memcpy(TEMP,first1[j],strlen(first1[j]));TEMP[strlen(first1[j])]='\0';}}}else//如果右部为符号串{for(j=0;;j++){if(v[j]==p[0])//求右部符号的第一个字符p[0]在所有字符集中的位置jbreak;}if(i>=0)merge(first[i],first1[j],2);elsemerge(TEMP,first1[j],2);for(k=0;k<=length-1;k++){empt[0]='\0';if(_emp(p[k])==1&&k=0)merge(first[i],first1[m],2);elsemerge(TEMP,first1[m],2);}elseif(_emp(p[k])==1&&k==length-1){//当前右部符号串可推出空且是右部符号串的最后一个字符temp[0]='&';temp[1]='\0';if(i>=0)merge(first[i],temp,1);else本文档如对你有帮助,请帮忙下载支持!merge(TEMP,temp,1);}elseif(_emp(p[k])==0)break;}}}/*******************************************求各产生式左部的FOLLOW********************************************/voidFOLLOW(inti){//参数i为该符号在非终结符中的位置intj,k,m,n,result=1;charc,temp[20];c=non_ter[i];//c为待求的非终结符temp[0]=c;temp[1]='\0';merge(foll,temp,1);//把当前字符放到一临时数组foll[]中,标识求已求其FOLLOW集.避免循环递归if(c==start){//若为开始符号-----开始符号S,则#∈FOLLOW(S)temp[0]='#';temp[1]='\0';merge(follow[i],temp,1);}for(j=0;j<=count-1;j++){if(in(c,right[j])==1){//比如求FOLLOW(B)//找一个右部含有当前字符c的产生式则找A→αB或A→αBβ(β=>*&)的产生式for(k=0;;k++){if(right[j][k]==c)break;//k为c在该产生式右部的序号,如B在产生式A→αB中的位置}for(m=0;;m++){if(v[m]==left[j])break;//m为产生式左部非终结符在所有符号中的序号}//如果c在产生式右部的最后,形如产生式A→αB,则FOLLOW(A)∈FOLLOW(B)if(k==(int)strlen(right[j])-1){if(in(v[m],foll)==1)//查找该非终结符是否已经求过其FOLLOW集.避免循本文档如对你有帮助,请帮忙下载支持!环递归{//是则FOLLOW(A)∈FOLLOW(B)merge(follow[i],follow[m],1);//把c所在产生式的左部非终结符的FOLLOW集加入到FOLLOW(c)中continue;//结束本次循环,进入j++循环}if(followflag[m]=='0'){//如果该非终结符的FOLLOW未求过FOLLOW(m);//求之FOLLOW集followflag[m]='1';//标识为1}merge(follow[i],follow[m],1);//FOLLOW(A)∈FOLLOW(B)}else{//如果c不在产生式右部的最后for(n=k+1;n<=(int)strlen(right[j])-1;n++){,形如A→αBβempt[0]='\0';//把empt[]置空,因为求此字符是否可推出空字_emp(c)时用到result*=_emp(right[j][n]);}if(result==1){//如果右部c后面的符号串能推出空,A→αBβ(β=>*&)则FOLLOW(A)∈FOLLOW(B)if(in(v[m],foll)==1){//查找该非终结符是否已经求过其FOLLOW集.避免循环递归merge(follow[i],follow[m],1);//FOLLOW(A)∈FOLLOW(B)continue;}if(followflag[m]=='0'){FOLLOW(m);followflag[m]='1';}merge(follow[i],follow[m],1);}//若A→αBβ,其中B∈VN,α∈(VTUVN)*、β∈(VTUVN)+,则FIRST(β)-{ε}∈FOLLOW(B);for(n=k+1;n<=(int)strlen(right[j])-1;n++){temp[n-k-1]=right[j][n];}temp[strlen(right[j])-k-1]='\0';本文档如对你有帮助,请帮忙下载支持!FIRST(-1,temp);//求FIRST(β)merge(follow[i],TEMP,2);//把FIRST(β)中所有非空元素加入到FOLLOW(B)中}}}followflag[i]='1';//标识当前要求的非终结符的FOLLOW集已求过}/*******************************************判断读入文法是否为一个LL(1)文法********************************************/intLL1(){inti,j,length,result=1;chartemp[50];for(j=0;j<=49;j++){//初始化first[j][0]='\0';follow[j][0]='\0';first1[j][0]='\0';select[j][0]='\0';TEMP[j]='\0';temp[j]='\0';firstflag[j]='0';//用来记录该字符的FIRST集是否已求过.1表示已求,0表示未求followflag[j]='0';//用来记录该字符的FOLLOW集是否已求过.1表示已求,0表示未求}for(j=0;j<=(int)strlen(v)-1;j++){first2(j);//求单个符号的FIRST集合,结果保存在first1[]里}printf("\n各非终结符推出的first集:\n");for(j=0;j<=(int)strlen(v)-1;j++){printf("%c:%s",v[j],first1[j]);}printf("\n能导空的非终结符集合:%s",empty);printf("\n_emp:");for(j=0;j<=(int)strlen(v)-1;j++)printf("%d",_emp(v[j]));for(i=0;i<=count-1;i++)FIRST(i,right[i]);//求FIRSTfor(j=0;j<=(int)strlen(non_ter)-1;j++){//求FOLLOWif(foll[j]==0)本文档如对你有帮助,请帮忙下载支持!{foll[0]='\0';FOLLOW(j);}}printf("\nfirst集:");for(i=0;i<=count-1;i++)printf("%s",first[i]);printf("\nfollow集合:");for(i=0;i<=(int)strlen(non_ter)-1;i++)printf("%s",follow[i]);for(i=0;i<=count-1;i++){//求每一产生式的SELECT集合memcpy(select[i],first[i],strlen(first[i]));//first[]存放的是各产生式右部的select[i][strlen(first[i])]='\0';for(j=0;j<=(int)strlen(right[i])-1;j++)result*=_emp(right[i][j]);FIRST集if(strlen(right[i])==1&&right[i][0]=='&')//形如产生式A->&result=1;if(result==1){for(j=0;;j++)if(v[j]==left[i])//j为左部符号在所有字符集中的位置break;merge(select[i],follow[j],1);}}printf("\nselect集合顺序是:");for(i=0;i<=count-1;i++)printf("%s",select[i]);memcpy(temp,select[0],strlen(select[0]));temp[strlen(select[0])]='\0';for(i=1;i<=count-1;i++){/*判断输入文法是否为length=strlen(temp);if(left[i]==left[i-1]){LL(1)文法*/merge(temp,select[i],1);if(strlen(temp)=0;n--)S[p++]=right[m][n];S[q+strlen(right[m])]='\0';}}}printf("S:%sstr:",S);for(p=j;p<=(int)strlen(str)-1;p++)printf("%c",str[p]);printf("\n");}}/*******************************************一个用户调用函数********************************************/voidmenu(){syntax();printf("\n是否继续?(yorn):");scanf("%c",&choose);本文档如对你有帮助,请帮忙下载支持!getchar();while(choose=='y'){menu();}}/*******************************************主函数********************************************/voidmain(){inti,j;start=grammer(termin,non_ter,left,right);//读入一个文法printf("count=%d",count);printf("\n开始符号为:%c",start);strcpy(v,non_ter);strcat(v,termin);printf("\n所有符号集为:%s",v);printf("\n非终结符集合:{%s",non_ter);printf("}");printf("\n终结符集合:{%s",termin);printf("}");printf("\n文法所有右边表达式依次是:");for(i=0;i<=count-1;i++){printf("%s",right[i]);}printf("\n文法所有左边开始符依次是:");for(i=0;i<=count-1;i++){printf("%c",left[i]);}if(validity==1)validity=judge();printf("\nvalidity=%d",validity);if(validity==1){ll=LL1();printf("\nll=%d",ll);if(ll==0)printf("\n该文法不是一个LL1文法!");else{printf("\n该文法是一个LL(1)文法!");本文档如对你有帮助,请帮忙下载支持!MM();printf("\n");for(i=0;i<=19;i++)for(j=0;j<=19;j++)if(M[i][j]>=0)printf("M[%d][%d]=%d",i,j,M[i][j]);menu();}}}由于算法仍有很多错误,最终结果没能实现,这点很失望!五、实验心得通过本次实验,我收获了很多东西。首先对编译原理这门课有了进一步的深刻理解,同时对LL(1)文法分析的原理和过程有了进一步的巩固,也锻炼了我编程的能力,巩固了平时所学的知识,真正做到了学以致用。在做实验的过程中,发现自己在编写程序过程中,总是会忽略各种细节,从而导致经常修改一些很小的低级错误才能使程序正常运行,不仅浪费时间,还影响对其他地方的修改,并且在很多步骤处理上,方法不正确。使结果不能符合要求,深刻体会到了自己在编程方面与别人的差距,在今后的学习中,我会注意改正自己在这方面的缺点,促使自己的编程水平不断进步。编译原理是一门专业学科,对于现阶段的我来说,只能掌握它的一些基本原理和概念,对于一些更深层的知识还是有很多难以理解的地方。但在这次实验过程中,锻炼了自己的思考能力,也锻炼了自己的动手编程能力,对于将知识的转化有了很大的帮助。
本文档为【语法分析器实验报告】,请使用软件OFFICE或WPS软件打开。作品中的文字与图均可以修改和编辑, 图片更改请在作品中右键图片并更换,文字修改请直接点击文字进行修改,也可以新增和删除文档中的内容。
该文档来自用户分享,如有侵权行为请发邮件ishare@vip.sina.com联系网站客服,我们会及时删除。
[版权声明] 本站所有资料为用户分享产生,若发现您的权利被侵害,请联系客服邮件isharekefu@iask.cn,我们尽快处理。
本作品所展示的图片、画像、字体、音乐的版权可能需版权方额外授权,请谨慎使用。
网站提供的党政主题相关内容(国旗、国徽、党徽..)目的在于配合国家政策宣传,仅限个人学习分享使用,禁止用于任何广告和商用目的。
下载需要: 免费 已有0 人下载
最新资料
资料动态
专题动态
个人认证用户
is_090281
常年从事设计、施工及相关教育工作,一线工作经验丰富。
格式:doc
大小:236KB
软件:Word
页数:42
分类:
上传时间:2021-11-14
浏览量:1