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

企业网站需求文档公司做网站合同

企业网站需求文档,公司做网站合同,无锡自助建站软件,上海seo公司排名#x1f3e0;关于专栏#xff1a;Linux的浅学到熟知专栏用于记录Linux系统编程、网络编程等内容。 #x1f3af;每天努力一点点#xff0c;技术变化看得见 文章目录 进程程序替换什么是程序替换及其原理替换函数execlexeclpexecleexecvexecvpexecvpeexecve 替换函数总结实现… 关于专栏Linux的浅学到熟知专栏用于记录Linux系统编程、网络编程等内容。 每天努力一点点技术变化看得见 文章目录 进程程序替换什么是程序替换及其原理替换函数execlexeclpexecleexecvexecvpexecvpeexecve 替换函数总结实现简易Shell 进程程序替换 什么是程序替换及其原理 父进程创建子进程的目的只有一个让子进程帮助父进程完成某些任务。如果要让子进程执行与父进程不同的代码有两种方式↓↓↓ 通过if分支判断语句决定父子进程各自的执行代码 #include stdio.h #include unistd.h #include sys/types.hint main() {pid_t id fork();if(id 0)//进程创建错误{perror(fork);exit(1);}else if(id 0)//子进程{int cnt 5;while(cnt){printf(child process %d is doing something different from parent process!\n);cnt--;}exit(0);}else//父进程{int status 0;pid_t ret waitpid(id, status, 0);if(ret id){printf(parent wait child process success! exitcode %d\n, WEXITSTATUS(status));}}return 0; }通过进程程序替换让子进程执行与父进程完全不同代码 下面仅是演示代码关于进程程序替换的详细内容将在下文介绍↓↓↓ #include stdio.h #include stdlib.h #include unistd.h #include sys/types.h #include sys/wait.hint main() {pid_t id fork();if(id 0){perror(fork);exit(1);}else if(id 0){execlp(top, top, NULL);exit(2);}else{int status 0;pid_t ret waitpid(id, status, 0);if(WIFEXITED(status)){printf(wait success! exitcode %d\n, WEXITSTATUS(status));}}return 0; }从上面可知用fork创建子进程后可以执行的程序和父进程相同的程序但可能执行不同的代码分支也可以通过调用exec系列函数接口来执行另一个程序。 当程序调用exec系列函数中的一个时该进程的用户空间的代码和数据完全被新程序替换从新程序的启动例程开始执行。调用exec并没有创建新进程所以调用exec前后该子进程的id并没有改变。调用了exec函数后会对子进程的数据和代码做写入此时会发生写时拷贝即子进程不再与父进程共享代码和数据而是在物理空间中拥有自己独立的代码和数据。 ★psCPU如何得知替换后程序的入口Linux中形成的可执行文件是有格式的即ELF可执行文件的表头包含可执行程序的入口地址、页表、mm_struct程序地址空间等。 替换函数 下面了解一下exec系列函数-替换换函数共有7个先看一下它们对它们有一个大致印象↓↓↓除了execve在2号man手册其余均位于3号man手册 下面给出每个exec系列函数的使用方法↓↓↓ execl int execl(const char* path, const char* arg, ...) execl的第一个参数需要传入可执行文件的绝对路径例如可以传入/usr/bin/ls。而余下参数为可变参数传入形式就和我们使用命令行命令一样先给出命令名称再给出命令行参数最终以NULL结尾例如“ls”、“-a”、“-l”、NULL。 ★psexecl的l表示list列出的意思即需要列出每个命令行参数。 下面给出接口使用示例↓↓↓ #include stdio.h #include stdlib.h #include unistd.h #include sys/types.h #include sys/wait.hint main() {pid_t id fork();if(id 0){perror(fork);exit(1);}else if(id 0){execl(/usr/bin/ls, ls, -a, -l, NULL);exit(2);}else{int status 0;pid_t ret waitpid(id, status, 0);if(WIFEXITED(status)){printf(Wait %d success! exitcode is %d\n, ret, WEXITSTATUS(status));}}return 0; }execlp int execlp(const char* file, const char* arg, ...); execlp第一个参数如果是存在于PATH环境变量中的可执行文件如命令等可以直接写出可执行文件名称即可不用写绝对路径因为execlp在执行时会在PATH环境变量中的各个目录下查找对应的可执行文件但如果是不存放于PATH环境变量中的各个目录下的可执行文件则需要使用绝对路径。而余下参数为可变参数传入形式就和我们使用命令行命令一样先给出命令名称再给出命令行参数最终以NULL结尾这与execl相同。 ★psexec系列函数中主要带有p的如果可执行文件存在于PATH环境变量中均不需要使用绝对路径只需要给出可执行文件名即可。 下面给出接口使用示例↓↓↓ #include stdio.h #include stdlib.h #include unistd.h #include sys/types.h #include sys/wait.hint main() {pid_t id fork();if(id 0){perror(fork);exit(1);}else if(id 0){execlp(ls, ls, -a, -l, NULL);exit(2);}else{int status 0;pid_t ret waitpid(id, status, 0);if(WIFEXITED(status)){printf(Wait %d success! exitcode is %d\n, ret, WEXITSTATUS(status));}}return 0; }★ps如果我们的exec系列函数能够执行系统命令那如何执行我们自己编写的可执行程序呢下面演示C语言程序调用C程序↓↓↓ excute.cpp↓↓↓ #include iostream using namespace std;int main() {for(int i 0; i 5; i){cout Jammingpro is coding... endl;}return 0; }execlp2.c #include stdio.h #include stdlib.h #include unistd.h #include sys/types.h #include sys/wait.hint main() {pid_t id fork();if(id 0){perror(fork);exit(1);}else if(id 0){execlp(./excute, excute, NULL);exit(2);}else{int status 0;pid_t ret waitpid(id, status, 0);if(WIFEXITED(status)){printf(Wait %d success! exitcode is %d\n, ret, WEXITSTATUS(status));}}return 0; }execle int execle(const char *path, const char *arg,..., char * const envp[]); 该接口函数的第一个参数path需要使用传入可执行文件的绝对路径紧接其后的arg与execl用法一致传入形式就和我们使用命令行命令一样先给出命令名称再给出命令行参数最终以NULL结尾。最后一个参数envp表示环境变量。 子进程默认继承父进程的环境变量那环境变量是什么时候传给子进程的呢环境变量也是数据创建子进程的时候环境变量就被子进程继承下去了。即使对代码和数据修改时发生写时拷贝也不会影响父子进程共享同一片环境变量空间。所以发生程序替换时环境变量信息不会被替换。如果需要对子进程的环境变量做修改可以选择execle、execvpe、execve这3个exec系列接口。这里先通过execle接口介绍execle、execvpe、execve的两个应用场景↓↓↓ 希望子进程增加新增或覆盖某些环境变量 下面代码中父进程fork创建子进程后子进程在进行程序替换前给自己增加了一个Jammingpro666的环境变量。子进程能获取该环境变量而父进程无法获取该环境变量。 execle_test.c↓↓↓ #include stdio.h #include stdlib.hint main() {char* s getenv(Jammingpro);if(s ! NULL) printf(%s\n, s);return 0; }execle1.c↓↓↓ #include stdio.h #include stdlib.h #include unistd.h #include sys/types.h #include sys/wait.hint main() {extern char **environ;pid_t id fork();if(id 0){perror(fork);exit(1);}else if(id 0)//子进程{extern char **environ;putenv(Jammingpro666);//给子进程新增环境变量该语句子进程会执行execle(./execle_test, execle_test, NULL, environ);exit(2);}else//父进程{int status 0;pid_t ret waitpid(id, status, 0);char* s getenv(Jammingpro);if(s ! NULL) printf(%s, s);if(WIFEXITED(status)){printf(Wait %d success! exitcode is %d\n, ret, WEXITSTATUS(status));}}return 0; }安全考虑及定制化场景 出于安全考虑不希望子进程获取父进程的环境变量或者因为子进程需要定制与父进程完全不一样的环境变量。可以定义一个字符指针数组在该数组中存储子进程专属的环境变量当使用execle、execvpe、execve将该环境变量传入时会直接覆盖从父进程继承下来的环境变量。 下面代码中给子进程创建专门的环境变量子进程此时可以获取专门的环境变量Jammingpro666但无法获取父进程的环境变量PWD因为使用exec带e的接口时直接覆盖了子进程从父进程那里继承的环境变量。而父进程能获取从bash继承下来的环境变量而无法获取子进程专属的环境变量。 execle_test2.c↓↓↓ #include stdio.h #include stdlib.hint main() {char* s getenv(Jammingpro);if(s ! NULL) printf(%s\n, s);else printf(Dont have Jammingpro\n);s getenv(PWD);if(s ! NULL) printf(%s\n, s);else printf(Dont have PWD\n);return 0; }execle2.c↓↓↓ #include stdio.h #include stdlib.h #include unistd.h #include sys/types.h #include sys/wait.hint main() {extern char **environ;pid_t id fork();if(id 0){perror(fork);exit(1);}else if(id 0)//子进程{char* env[] {Jammingpro666, xiaoming888};execle(./execle_test2, execle_test2, NULL, env);exit(2);}else//父进程{int status 0;pid_t ret waitpid(id, status, 0);char* s getenv(Jammingpro);if(s ! NULL) printf(%s\n, s);else printf(Dont have Jammingpro\n);s getenv(PWD);if(s ! NULL) printf(%s\n, s);else printf(Dont have PWD\n);if(WIFEXITED(status)){printf(Wait %d success! exitcode is %d\n, ret, WEXITSTATUS(status));}}return 0; }★ps关于putenv与exec带e系列接口原理探索 子进程和父进程的PCB内都有一个环境变量表指针当子进程刚创建时子进程的环境变量表指针与父进程指向同一个位置一旦子进程调用putenv尝试对环境表做修改此时则会发生写时拷贝。 环境变量表是是一个字符指针数组也就是说环境变量表中并不会直接存储对应的环境变量而是存储各个环境变量的存储地址。当我们使用putenv时本质是将我们定义环境变量字符串或char类型数组的首地址存储到该环境变量中。如果定义一个char env[100] Jammingpro666再执行putenv(env)则会将env的首地址存储到环境变量表中。此时不可以修改env数组中的内容一旦修改则对应的环境变量会跟着发生变化。 putenv只是修改环境变量表中某个表项的指向。但如果我们使用的是exec带e系列函数则会直接修改进程PCB中环境变量表指针的指向。 execv int execv(const char *path, char *const argv[]); 该接口第一个参数需要传入可执行程序的绝对路径第二参数需要传入命令行参数不同的是这里并不是使用可变参数列表的方式而是使用字符指针数组的方式。例如我们需要执行ls命令则第一个参数需要传入/usr/bin/ls第二个参数需要先定义一个字符指针数组char* opts[] {ls, -a, -l, NULL}再将该数组作为第二参数传入。 ★psexec系列函数中带v的则第二个参数需要以字符指针数组的形式传入命令行参数。 下面给出接口使用示例↓↓↓ #include stdio.h #include stdlib.h #include unistd.h #include sys/types.h #include sys/wait.hint main() {extern char **environ;pid_t id fork();if(id 0){perror(fork);exit(1);}else if(id 0)//子进程{char* opts[] {ls, -a, -l, NULL};execv(/usr/bin/ls, opts);exit(2);}else//父进程{int status 0;pid_t ret waitpid(id, status, 0);if(WIFEXITED(status)){printf(Wait %d success! exitcode is %d\n, ret, WEXITSTATUS(status));}}return 0; }execvp int execvp(const char *file, char *const argv[]); 该接口第一个参数传入可执行文件如果该可执行文件可以在PATH环境变量中找到则不需要使用绝对路径否则需要使用绝对路径第二个参数需要以字符指针数组的形式传入命令行参数。 下面给出接口使用示例↓↓↓ #include stdio.h #include stdlib.h #include unistd.h #include sys/types.h #include sys/wait.hint main() {extern char **environ;pid_t id fork();if(id 0){perror(fork);exit(1);}else if(id 0)//子进程{char* opts[] {top, NULL};execvp(top, opts);exit(2);}else//父进程{int status 0;pid_t ret waitpid(id, status, 0);if(WIFEXITED(status)){printf(Wait %d success! exitcode is %d\n, ret, WEXITSTATUS(status));}}return 0; }execvpe int execvpe(const char *file, char *const argv[],char *const envp[]); 该接口第一个参数传入可执行文件如果该可执行文件可以在PATH环境变量中找到则不需要使用绝对路径否则需要使用绝对路径第二个参数需要以字符指针数组的形式传入命令行参数第三个参数需要传入环境变量。 下面给出接口使用示例给子进程传入自定义环境变量↓↓↓ #include stdio.h #include stdlib.h #include unistd.h #include sys/types.h #include sys/wait.hint main() {extern char **environ;pid_t id fork();if(id 0){perror(fork);exit(1);}else if(id 0)//子进程{char* opts[] {top, NULL};char* env[] {Jammingpro666};execvpe(./execle_test, opts, env);exit(2);}else//父进程{int status 0;pid_t ret waitpid(id, status, 0);if(WIFEXITED(status)){printf(Wait %d success! exitcode is %d\n, ret, WEXITSTATUS(status));}}return 0; }execve int execve(const char *filename, char *const argv[],char *const envp[]); 该接口第一个参数需要传入可执行程序的绝对路径第二个参数需要以指针数组的方式传入命令行参数第三个参数需要传入环境变量。 下面给出该接口的使用方式↓↓↓ #include stdio.h #include stdlib.h #include unistd.h #include sys/types.h #include sys/wait.hint main() {extern char **environ;pid_t id fork();if(id 0){perror(fork);exit(1);}else if(id 0)//子进程{char* opts[] {top, NULL};char* env[] {Jammingpro666};execve(./execle_test, opts, env);exit(2);}else//父进程{int status 0;pid_t ret waitpid(id, status, 0);if(WIFEXITED(status)){printf(Wait %d success! exitcode is %d\n, ret, WEXITSTATUS(status));}}return 0; }替换函数总结 上面总结上面介绍的各个替换函数↓↓↓ 函数名参数格式PATH中可执行程序是否需要带绝对路径是否使用当前环境变量execl列表是是execlp列表不是是execle列表是不是需自己组装环境变量execv数组是是execvp数组不是是execvpe数组不是不是需自己组装环境变量execve数组是不是需自己组装环境变量 上面的exec系列函数中如果带有p的则第一个参数的可执行文件若存在于PATH环境变量中则只需要填写可执行文件名如果带有v的则需要以字符指针数组的形式传入命令行参数如果是带e的则需要自己组装环境变量。 上面的各个接口统称为加载器它们为即将替换进来的可执行程序加载入参数列表、环境变量等信息。下面我们使用execvpe接口给自定义可执行程序传入命令行参数及环境变量该可执行程序将会把命令行参数及环境变量打印至显示器↓↓↓ printInfo.c #include stdio.hint main(int argc, char* argv[]; char* env[]) {printf(传入%d个命令行参数分别是:\n, argc);int i 0;for(; argv[i]; i){printf([%d]-%s\n, i, argv[i]);}printf(--------------------------------------\n);printf(环境变量分别是:\n);i 0;for(; env[i]; i){printf([%d]-%s\n, i, env[i]);}return 0; }execvpe2.c #include stdio.h #include stdlib.h #include unistd.h #include sys/types.h #include sys/wait.hint main() {extern char **environ;pid_t id fork();if(id 0){perror(fork);exit(1);}else if(id 0)//子进程{char* opts[] {printInfo, NULL};char* env[] {Jammingpro666, xiaoming888};execvpe(./printInfo, opts, env);exit(2);}else//父进程{int status 0;pid_t ret waitpid(id, status, 0);if(WIFEXITED(status)){printf(Wait %d success! exitcode is %d\n, ret, WEXITSTATUS(status));}}return 0; }上述各个接口中只有execve是系统调用其他均是对该系统调用接口的封装。这也就是为什么execve位于2号手册而其他接口函数位于3号手册的原因。 实现简易Shell 我们来模拟实现一个Shell这个Shell具有一些常用的简易功能。下面我们用一张图了解一下Linux中bashShell的一种的执行过程↓↓↓ 一个shell程序需要循环做的事如下 1.获取命令行 2.解析命令行 3.创建子程序 4.替换子程序 5.父进程等待子进程退出 下面我们先来完成第一步获取命令行↓↓↓ #include stdio.h #include stdlib.h #include string.h #include assert.h #include unistd.h #include sys/types.h #include sys/wait.h#define LEFT [ #define RIGHT ] #define LABLE ##define COM_LEN 1024 //输入命令行长度 #define PWD_LEN 128 //当前路径长度char command[COM_LEN]; //命令行缓冲区 char pwd[PWD_LEN]; //当前路径缓冲区//获取用户名 const char* getUser() {return getenv(USER); }//获取当前路径 const char* getPWD() {getcwd(pwd, sizeof(pwd));return pwd; }//获取主机名 const char* getHostName() {return getenv(HOSTNAME); }//输出提示信息获取用户输入 void getCommand() {printf(LEFT%s%s %sRIGHTLABLE , getHostName(), getUser(), getPWD());char* s fgets(command, sizeof(command), stdin);assert(s ! NULL);(void)s;s[strlen(command) - 1] \0;printf(%s\n, command); }int main() {while(1){getCommand();}return 0; }下面我们需要对获取的字符串进行切割↓↓↓ ★ps下面spliteString中使用了条件编译当编译时带上-DDEBUG就可以输出分隔的结果 #include stdio.h #include stdlib.h #include string.h #include assert.h #include unistd.h #include sys/types.h #include sys/wait.h#define LEFT [ #define RIGHT ] #define LABLE # #define SEP \t#define COM_LEN 1024 #define ARG_LEN 64 #define PWD_LEN 128char command[COM_LEN]; char *argv[ARG_LEN]; char pwd[PWD_LEN]; int argc 0;const char* getUser() {return getenv(USER); }const char* getPWD() {getcwd(pwd, sizeof(pwd));return pwd; }const char* getHostName() {return getenv(HOSTNAME); }//输出提示信息获取用户输入 void getCommand() {printf(LEFT%s%s %sRIGHTLABLE , getHostName(), getUser(), getPWD());char* s fgets(command, sizeof(command), stdin);assert(s ! NULL);(void)s;s[strlen(command) - 1] \0; }//分隔字符串 void spliteString() {argc 0;argv[argc] strtok(command, SEP);while(argv[argc] strtok(NULL, SEP)); #ifdef DEBUG int j 0;for(;argv[j]; j){printf([%d]-%s\n, j, argv[j]);} #endif }int main() {while(1){getCommand();spliteString();}return 0; }使用normalExcute创建子进程使用程序替换的方式让子进程执行指定程序如果子进程替换失败则返回EXIT_CODE。父进程等待子进程并将子进程退出码保存在exitcode中。 #include stdio.h #include stdlib.h #include string.h #include assert.h #include unistd.h #include sys/types.h #include sys/wait.h#define LEFT [ #define RIGHT ] #define LABLE # #define SEP \t #define EXIT_CODE 66#define COM_LEN 1024 #define ARG_LEN 64 #define ENV_LEN 32 #define PWD_LEN 128char command[COM_LEN]; char *argv[ARG_LEN]; char pwd[PWD_LEN]; int argc 0; int exitcode 0;const char* getUser() {return getenv(USER); }const char* getPWD() {getcwd(pwd, sizeof(pwd));return pwd; }const char* getHostName() {return getenv(HOSTNAME); }//输出提示信息获取用户输入 void getCommand() {printf(LEFT%s%s %sRIGHTLABLE , getHostName(), getUser(), getPWD());char* s fgets(command, sizeof(command), stdin);assert(s ! NULL);(void)s;s[strlen(command) - 1] \0; }//分隔字符串 void spliteString() {argc 0;argv[argc] strtok(command, SEP);while(argv[argc] strtok(NULL, SEP)); #ifdef DEBUG int j 0;for(;argv[j]; j){printf([%d]-%s\n, j, argv[j]);} #endif }void normalExcute() {pid_t id fork();assert(id ! -1);if(id 0){exitcode 0;execvp(argv[0], argv);exit(EXIT_CODE);}else {int status 0;pid_t id waitpid(id, status, 0);exitcode WEXITSTATUS(status);} }int main() {while(1){getCommand();spliteString();normalExcute();}return 0; } 截至到这里我们已经实现了能够执行大部分命令的Shell程序。但如果我们执行cd命令当前路径却没有改变。这是为什么呢 当子进程被创建后由子进程执行cd命令则修改环境变量PWD时子进程会发生写时拷贝保证父子进程的独立性。因而子进程的PWD改变不会影响父进程。像这样的命令需要使用内建命令的方式解决即遇到这类命令时不让子进程执行而是由父进程自己执行对应的函数。下面通过设置buildExcute函数不能交由子进程执行的命令进行了特殊处理↓↓↓ #include stdio.h #include stdlib.h #include string.h #include assert.h #include unistd.h #include sys/types.h #include sys/wait.h#define LEFT [ #define RIGHT ] #define LABLE # #define SEP \t #define EXIT_CODE 66#define COM_LEN 1024 #define ARG_LEN 64 #define ENV_LEN 32 #define PWD_LEN 128char *env[ENV_LEN]; char command[COM_LEN]; char *argv[ARG_LEN]; char pwd[PWD_LEN]; int argc 0; int envNum 0; int exitcode 0;const char* getUser() {return getenv(USER); }const char* getPWD() {getcwd(pwd, sizeof(pwd));return pwd; }const char* getHostName() {return getenv(HOSTNAME); }//输出提示信息获取用户输入 void getCommand() {printf(LEFT%s%s %sRIGHTLABLE , getHostName(), getUser(), getPWD());char* s fgets(command, sizeof(command), stdin);assert(s ! NULL);(void)s;s[strlen(command) - 1] \0; }//分隔字符串 void spliteString() {argc 0;argv[argc] strtok(command, SEP);while(argv[argc] strtok(NULL, SEP)); #ifdef DEBUG int j 0;for(;argv[j]; j){printf([%d]-%s\n, j, argv[j]);} #endif }void normalExcute() {pid_t id fork();assert(id ! -1);if(id 0){exitcode 0;execvp(argv[0], argv);exit(EXIT_CODE);}else {int status 0;pid_t id waitpid(id, status, 0);exitcode WEXITSTATUS(status);} }int buildExcute() {if(argc 3 strcmp(argv[0], cd) 0){int ret chdir(argv[1]);if(ret ! -1) exitcode 0;else exitcode EXIT_CODE;return 1;}else if(argc 3 strcmp(argv[0], export) 0){env[envNum] (char*)malloc(sizeof(argv[1]) 1);strcpy(env[envNum], argv[1]);int ret putenv(env[envNum]);if(ret 0) exitcode 0;else exitcode EXIT_CODE;return 1;}else if(argc 3 strcmp(argv[0], echo) 0){if(strcmp(argv[1], $?) 0) printf(%d\n, exitcode);else printf(%s\n, argv[1]);return 1;}else if(strcmp(argv[0], ls) 0){argv[argc - 1] (char*)--colorauto;argv[argc] NULL;}return 0; }int main() {while(1){getCommand();spliteString();int ret buildExcute();if(!ret) normalExcute();}return 0; } 至此能执行大部分命令的Shell程序就大功告成了。但该程序并没有所有内建命令做处理。 ★ps当我们登录Linux的时候就要启动一个shell进程该进程会从用户home目录的.bash_profile中获取环境变量从而读入环境变量。从上面的自定义实现shell程序中可知export导入环境变量仅在当前shell进程有效。如果需要设置永久性环境变量需要修改系统的环境变量配置文件。 ★ps现象程序替换成功后exec系列函数后的代码不会被执行若替换失败则继续向下执行。这也解释了exec系列函数为什么只有失败返回值而没有正确返回值。一旦正确后序代码不会执行该正确返回值也无法使用。 欢迎进入从浅学到熟知Linux专栏查看更多文章。 如果上述内容有任何问题欢迎在下方留言区指正b(▽)d
http://www.zqtcl.cn/news/380016/

相关文章:

  • 做网站哪些公司专业做app软件开发公司
  • 蒙特网站建设湖北省建设厅网站上岗证查询
  • 宁波网站建设 联系哪家电子商务网站建设过程范文
  • 南宁商城网站建设网站建设的需求文档
  • dedeampz 部署wordpress 网站访问慢如何评价网站是否做的好处
  • 怎样建设个人影视网站设计学专业
  • 没有公司 接单做网站网站建设加盟合作
  • 如何将域名和网站绑定做网站找投资人
  • 网站开发 平台WordPress首页可见
  • 沧州做网站费用打开上海发布
  • 重庆潼南网站建设公司电话网站能调用一些字体
  • 摄影网站设计素材做彩票网站电话多少
  • 开网站公司企业管理网课
  • 相城高端网站建设施工建设集团网站
  • .电子商务网站的开发原则包括网络服务示范区创建情况
  • 网站如何做权重php做网站登陆验证
  • 昆山制造网站的地方网站建设 有聊天工具的吗
  • 自己做网站制作需要多少钱如何免费注册网站域名
  • 如何做网站美化怎样写网站文案
  • 做网站排名的wordpress 调整 行距
  • 三亚文明城市建设服务中心报名网站房地产活动策划网站
  • 休闲食品网站建设规划书常德做网站专业公司
  • 做美工好的网站网页设计排版布局
  • 网站建设公司合同模板下载wordpress微信公众平台开发教程
  • 快速wordpress 建网站免费代理游戏
  • 网站模板 寻模板大气宽屏网站模板企业源码带后台
  • 做图片推广的网站威海高端网站建设
  • 台州网站公司建站网站首页模板图片
  • 网站建设本科毕业设计论文网址
  • 泰州企业建站程序乐清网站建设公司