控制终端(/dev/tty)

如果当前进程有控制终端(Controlling Terminal)的话,那么 /dev/tty 就是当前进程的控制终端的设备特殊文件。可以使用命令”ps –ax”来查看进程与哪个控制终端相连。对于你登录的shell,/dev/tty 就是你使用的终端,设备号是(5,0)。使用命令”tty”可以查看它具体对应哪个实际终端设备。/dev/tty 有些类似于到实际所使用终端设备的一个连接。

控制台终端(/dev/ttyn, /dev/console)

在UNIX系统中,计算机显示器通常被称为控制台终端(console)。它仿真了类型为Linux的一种终端(TERM=Linux),并且有一些设备特殊文件与之相关联:tty0、tty1、tty2等。当你在控制台上登录时,使用的是tty1。使用Alt+[F1—F6]组合键时,我们就可以切换到tty2、tty3等上面去。tty1 –tty6等称为虚拟终端,而tty0则是当前所使用虚拟终端的一个别名,系统所产生的信息会发送到该终端上。因此不管当前正在使用哪个虚拟终端,系统信息都会发送到控制台终端上。
你可以登录到不同的虚拟终端上去,因而可以让系统同时有几个不同的会话期存在。只有系统或超级用户root可以向/dev/tty0进行写操作。

struct termios 结构

用来持有所有的当前线路设置, 给终端 设备一个特定端口. 这些线路设置控制当前波特率, 数据大小, 数据流控设置, 以及许多其他值。

struct termios {

unsigned short c_iflag;              /* 输入模式标志*/
    unsigned short c_oflag;             /* 输出模式标志*/
    unsigned short c_cflag;             /* 控制模式标志*/
    unsigned short c_lflag;              /*区域模式标志或本地模式标志或局部模式*/
    unsigned char c_line;                /*行控制line discipline */
    unsigned char c_cc[NCC];      /* 控制字符特性*/

};

成员的值

(一)c_iflag 标志常量

Input mode ( 输入模式)可以在输入值传给程序之前控制其处理的方式。其中输入值可能是由序列埠或键盘的终端驱动程序所接收到的字元。可以利用termios结构的c_iflag的标志来加以控制,其定义的方式皆以‘or’来加以组合。

  • IGNBRK   :忽略输入中的 BREAK 状态。 (忽略命 令行中的中 断)
  • BRKINT   :(命令行出 现中断时,可产生一插断)如果设置了IGNBRK,将忽略 BREAK。如果没有设置,但是设置了 BRKINT,那么 BREAK 将使得输入和输出队列被刷新,如果终端是一个前台进程组的控制终端,这个进程组中所有进程将收到 SIGINT 信号。如果既未设置 IGNBRK 也未设置 BRKINT,BREAK 将视为与NUL 字符同义,除非设置了 PARMRK,这种情况下它被视为序列377 。
  • IGNPAR   :忽略桢错误和奇偶校验错。
  • PARMRK :如果没有设置 IGNPAR,在有奇偶校验错或桢错误的字符前插入 377 。如果既没有设置 IGNPAR 也没有设置PARMRK,将有奇偶校验错或桢错误的字符视为 。
  • INPCK     :启用输入奇偶检测。
  • ISTRIP     :去掉第八位。
  • INLCR     :将输入中的 NL 翻译为 CR。(将收到 的换行符号转换为Return)
  • IGNCR     :忽略输入中的回车。
  • ICRNL     :将输入中的回车翻译为新行 (除非设置了 IGNCR)(否则当输入信号有 CR 时不会终止输入)。
  • IUCLC     :(不属于 POSIX) 将输入中的大写字母映射为小写字母。
  • IXON        :启用输出的 XON/XOFF 流控制。
  • IXANY      :(不属于 POSIX.1;XSI) 允许任何字符来重新开始输出。(?)
  • IXOFF      :启用输入的 XON/XOFF 流控制。
  • IMAXBEL :(不属于 POSIX) 当输入队列满时响零。Linux 没有实现这一位,总是将它视为已设置。

(二) c_oflag 标志常量

Output mode ( 输 出模式)主要负责控制输出字元的处理方式。输出字元在传送到序列埠或显示器之前是如何被程序来处理。输出模式是利用termios结构的c_oflag的标志来加以控制,其定义的方式皆以‘or’来加以组合。

  • OPOST   :启用具体实现自行定义的输出处理。
  • OLCUC  :(不属于 POSIX) 将输出中的小写字母映射为大写字母。
  • ONLCR  :(XSI) 将输出中的新行符映射为回车- 换行。
  • OCRNL  :将输出中的回车映射为新行符。
  • ONOCR :不在第 0 列输出回车。
  • ONLRET:不输出回车。
  • OFILL     :发送填充字符作为延时,而不是使用定时来延时。
  • OFDEL   :(不属于 POSIX) 填充字符是 ASCII DEL (0177)。如果不设置,填充字符则是 ASCII NUL。
  • NLDLY   :新行延时掩码。取值为 NL0 和 NL1。
  • CRDLY   :回车延时掩码。取值为 CR0, CR1, CR2, 或 CR3。
  • TABDLY  :水平跳格延时掩码。取值为 TAB0, TAB1, TAB2, TAB3(或 XTABS)。取值为 TAB3,即 XTABS,将扩展跳格为空格 (每个跳格符填充 8 个空格)。(?)
  • BSDLY   :回退延时掩码。取值为 BS0 或 BS1。(从来没有被实现过)
  • VTDLY    :竖直跳格延时掩码。取值为 VT0 或 VT1。
  • FFDLY    :进表延时掩码。取值为 FF0 或 FF1。

(三)c_cflag 标志常量

Control mode ( 控制模式)主要用于控制终端设备的硬件设置。利用termios结构的c_cflag的标志来加以控制。控制模式用在序列线连接到数据设备,也可以用在与终端设备的交谈。一般来说,改变终端设备的组态要比使用termios的控制模式来改变行(lines)的行为来得容易。

  • CBAUD      :(不属于 POSIX) 波特率掩码 (4+1 位)。
  • CBAUDEX :(不属于 POSIX) 扩展的波特率掩码 (1 位),包含在CBAUD 中。
  • (POSIX 规定波特率存储在 termios 结构中,并未精确指定它的位置,而是提供了函数 cfgetispeed() 和 cfsetispeed() 来存取它。一些系统使用 c_cflag 中 CBAUD 选择的位,其他系统使用单独的变量,例如 sg_ispeed 和 sg_ospeed 。)
  • CSIZE        :字符长度掩码(传送或接收字元时用的位数)。 取值为CS5(传送或接收字元时用5bits), CS6, CS7, 或 CS8。
  • CSTOPB   :设置两个停止位,而不是一个。
  • CREAD     :打开接受者。
  • PARENB   :允许输出产生奇偶信息以及输入的奇偶校验(启用同位产生与侦测)。
  • PARODD   :输入和输出是奇校验(使用奇同位而非偶同位)。
  • HUPCL      :在最后一个进程关闭设备后,降低 modem 控制线 (挂 断)。(?)
  • CLOCAL    :忽略 modem 控制线。
  • LOBLK       :(不属于 POSIX) 从非当前 shell 层阻塞输出(用于shl )。(?)
  • CIBAUD     :(不属于 POSIX) 输入速度的掩码。CIBAUD 各位的值与CBAUD 各位相同,左移了 IBSHIFT 位。
  • CRTSCTS :(不属于 POSIX) 启用 RTS/CTS (硬件) 流控制。

(四)c_lflag 标志常量

Local mode ( 局部模式)主要用来控制终端设备不同的特色。利用termios结构里的c_lflag的标志来设定局部模式。在巨集中有两个比较重要的标志:

1. ECHO:它可以让你阻止键入字元的回应。

2. ICANON(正规模式)标志:它可以对所接收的字元在两种不同的终端设备模式之间来回切换。

  • ISIG           :当接受到字符 INTR, QUIT, SUSP, 或 DSUSP 时,产生相应的信号。
  • ICANON    :启用标准模式 (canonical mode)。允许使用特殊字符EOF, EOL, EOL2, ERASE, KILL, LNEXT, REPRINT, STATUS, 和WERASE,以及按行的缓冲。
  • XCASE      :(不属于 POSIX; Linux 下不被支持) 如果同时设置了ICANON,终端只有大写。输入被转换为小写,除了有前缀的字符。输出时,大写字符被前缀(某些系统指定的特定字符 ,小写字符被转换成大写。
  • ECHO       :回显输入字符。
  • ECHOE     :如果同时设置了 ICANON,字符 ERASE 擦除前一个输入字符,WERASE 擦除前一个词。
  • ECHOK     :如果同时设置了 ICANON,字符 KILL 删除当前行。
  • ECHONL  :如果同时设置了 ICANON,回显字符 NL,即使没有设置 ECHO。
  • ECHOCTL:(不属于 POSIX) 如果同时设置了 ECHO,除了 TAB,NL, START, 和 STOP 之外的 ASCII 控制信号被回显为 ^X, 这里 X 是比控制信号大 0x40 的 ASCII 码。例如,字符 0x08(BS) 被回显为 ^H。
  • ECHOPRT:(不属于 POSIX) 如果同时设置了 ICANON 和IECHO,字符在删除的同时被打印。
  • ECHOKE  :(不属于 POSIX) 如果同时设置了 ICANON,回显 KILL时将删除一行中的每个字符,如同指定了 ECHOE 和 ECHOPRT 一样。
  • DEFECHO:(不属于 POSIX) 只在一个进程读的时候回显。
  • FLUSHO   :(不属于 POSIX; Linux 下不被支持) 输出被刷新。这个标志可以通过键入字符 DISCARD 来开关。
  • NOFLSH   :禁止在产生 SIGINT, SIGQUIT 和 SIGSUSP 信号时刷新输入和输出队列,即关闭queue中的flush。
  • TOSTOP   :向试图写控制终端的后台进程组发送 SIGTTOU 信号(传送欲写入的信息到后台 处理)。
  • PENDIN    :(不属于 POSIX; Linux 下不被支持) 在读入下一个字符时,输入队列中所有字符被重新输出。(bash 用它来处理typeahead)
  • IEXTEN     :启用实现自定义的输入处理。这个标志必须与 ICANON同时使用,才能解释特殊字符 EOL2,LNEXT,REPRINT 和WERASE,IUCLC 标志才有效。

(五)c_cc 数组

特殊控制字元可提供使用者设定一些特殊的功能,如Ctrl+C的字元组合。特殊控制字元主要是利用termios结构里c_cc的阵列成员来做设定。c_cc阵列主要用于正规与非正规两种环境,但要注意的是正规与非正规不可混为一谈。其定义了特殊的控制字符。符号下标 (初始值) 和意义为:

  • VINTR       :(003, ETX, Ctrl-C, or also 0177, DEL, rubout) 中断字符。发出 SIGINT 信号。当设置 ISIG 时可被识别,不再作为输入传递。
  • VQUIT       :(034, FS, Ctrl-) 退出字符。发出 SIGQUIT 信号。当设置 ISIG 时可被识别,不再作为输入传递。
  • VERASE   :(0177, DEL, rubout, or 010, BS, Ctrl-H, or also#) 删除字符。删除上一个还没有删掉的字符,但不删除上一个EOF 或行首。当设置 ICANON 时可被识别,不再作为输入传递。
  • VKILL         :(025, NAK, Ctrl-U, or Ctrl-X, or also @) 终止字符。删除自上一个 EOF 或行首以来的输入。当设置 ICANON 时可被识别,不再作为输入传递。
  • VEOF         :(004, EOT, Ctrl-D) 文件尾字符。更精确地说,这个字符使得 tty 缓冲中的内容被送到等待输入的用户程序中,而不必等到 EOL。如果它是一行的第一个字符,那么用户程序的read() 将返回 0,指示读到了 EOF。当设置 ICANON 时可被识别,不再作为输入传递。VMIN :非 canonical 模式读的最小字符数(MIN 主要是表示能满足read的最小字元数)。
  • VEOL         :(0, NUL) 附加的行尾字符。当设置 ICANON 时可被识别。  VTIME :非 canonical 模式读时的延时,以十分之一秒为单位、VEOL2 :(not in POSIX; 0, NUL) 另一个行尾字符。当设置ICANON 时可被识别。VSWTCH :(not in POSIX; not supported under Linux; 0,NUL) 开关字符。(只为 shl 所用。)  VSTART :(021, DC1, Ctrl-Q) 开始字符。重新开始被 Stop 字符中止的输出。当设置 IXON 时可被识别,不再作为输入传递。
  • VSTOP       :(023, DC3, Ctrl-S) 停止字符。停止输出,直到键入Start 字符。当设置 IXON 时可被识别,不再作为输入传递。
  • VSUSP      :(032, SUB, Ctrl-Z) 挂起字符。发送 SIGTSTP 信号。当设置 ISIG 时可被识别,不再作为输入传递。
  • VDSUSP    :(not in POSIX; not supported under Linux; 031,EM, Ctrl-Y) 延时挂起信号。当用户程序读到这个字符时,发送SIGTSTP 信号。当设置 IEXTEN 和 ISIG,并且系统支持作业管理时可被识别,不再作为输入传递。
  • VLNEXT      :(not in POSIX; 026, SYN, Ctrl-V) 字面上的下一个。引用下一个输入字符,取消它的任何特殊含义。当设置IEXTEN 时可被识别,不再作为输入传递。
  • VWERASE  :(not in POSIX; 027, ETB, Ctrl-W) 删除词。当设置 ICANON 和 IEXTEN 时可被识别,不再作为输入传递。
  • VREPRINT :(not in POSIX; 022, DC2, Ctrl-R) 重新输出未读的字符。当设置 ICANON 和 IEXTEN 时可被识别,不再作为输入传递。
  • VDISCARD :(not in POSIX; not supported under Linux;017, SI, Ctrl-O) 开关:开始/结束丢弃未完成的输出。当设置IEXTEN 时可被识别,不再作为输入传递。
  • VSTATUS    :(not in POSIX; not supported under Linux;status request: 024, DC4, Ctrl-T).
  • 这些符号下标值是互不相同的,除了 VTIME,VMIN 的值可能分别与 VEOL,VEOF 相同。 (在 non-canonical 模式下,特殊字符的含义更改为延时含义。MIN 表示应当被读入的最小字符数。TIME 是以十分之一秒为单位的计时器。如果同时设置了它们,read 将等待直到至少读入一个字符,一旦读入 MIN 个字符或者从上次读入字符开始经过了 TIME 时间就立即返回。如果只设置了 MIN,read 在读入 MIN 个字符之前不会返回。如果只设置了TIME,read 将在至少读入一个字符,或者计时器超时的时候立即返回。如果都没有设置,read 将立即返回,只给出当前准备好的字符。)
    MIN与 TIME组合有以下四种:
           1、 MIN = 0 , TIME =0有READ立即回传否则传回 0 ,不读取任何字元
           2、 MIN = 0 , TIME >0READ 传回读到的字元,或在十分之一秒后传回TIME若来不及读到任何字元,则传回0
           3、 MIN > 0 , TIME =0READ 会等待,直到MIN字元可读
           4、 MIN > 0 , TIME > 0每一格字元之间计时器即会被启动READ 会在读到MIN字元,传回值或TIME的字元计时(1/10秒)超过时将值 传回

相关的函数

(一)tcgetattr()

  1. 原型
    int tcgetattr(int fd, struct termois &termios_p);
  2. 功能
    取得终端介质(fd)初始值,并把其值 赋给temios_p;函数可以从后台进程中调用;但是,终端属性可能被后来的前台进程所改变。

(二)tcsetattr()

  1. 原型
    int tcsetattr(int fd, int optional_actions, const struct termios *termios_p);
  2. 功能
    设置与终端相关的参数 (除非需要底层支持却无法满足),使用termios_p 引用的 termios 结构。optional_actions(tcsetattr函数的第二个参数)指定了什么时候改变会起作用:
            * TCSANOW    :改变立即发生 
            * TCSADRAIN :改变在所有写入 fd 的输出都被传输后生效。这个函数应当用于修改影响输出的参数时使用。(当前输出完成时将值改变) 
            * TCSAFLUSH:改变在所有写入 fd 引用的对象的输出都被传输后生效,所有已接受但未读入的输入都在改变发生前丢弃(同TCSADRAIN,但会舍弃当前所有值)。

(三)tcsendbreak()

传送连续的 0 值比特流,持续一段时间,如果终端使用异步串行数据传输的话。如果 duration 是 0,它至少传输 0.25 秒,不会超过 0.5 秒。如果duration 非零,它发送的时间长度由实现定义。 如果终端并非使用异步串行数据传输,tcsendbreak() 什么都不做。

(四)tcdrain()

等待直到所有写入 fd 引用的对象的输出都被传输。

(五)tcflush()

丢弃要写入 引用的对象,但是尚未传输的数据,或者收到但是尚未读取的数据,取决于 queue_selector 的值:
               * TCIFLUSH    :刷新收到的数据但是不读。
               * TCOFLUSH  :刷新写入的数据但是不传送。
               * TCIOFLUSH :同时刷新收到的数据但是不读,并且刷新写入的数据但是不传送。

(六)tcflow()

挂起 fd 引用的对象上的数据传输或接收,取决于 action 的值:
                * TCOOFF :挂起输出 
                * TCOON  :重新开始被挂起的输出 
                * TCIOFF  :发送一个 STOP 字符,停止终端设备向系统传送数据 
                * TCION    :发送一个 START 字符,使终端设备向系统传输数据 打开一个终端设备时的默认设置是输入和输出都没有挂起。

(七)波特率函数

被用来获取和设置 termios 结构中,输入和输出波特率的值。新值不会马上生效,直到成功调用了 tcsetattr() 函数。设置速度为 B0 使得 modem ”挂机“。与 B38400 相应的实际比特率可以用setserial(8) 调整。输入和输出波特率被保存于 termios 结构中。
       cfmakeraw 设置终端属性如下:
                termios_p->c_iflag &= ~(IGNBRK|BRKINT|PARMRK|ISTRIP|INLCR|IGNCR|ICRNL|IXON);
                termios_p->c_oflag &= ~OPOST;
                termios_p->c_lflag &= ~(ECHO|ECHONL|ICANON|ISIG|IEXTEN);
                termios_p->c_cflag &= ~(CSIZE|PARENB);
                termios_p->c_cflag |= CS8;

  1. cfgetospeed() 返回 termios_p 指向的 termios 结构中存储的输出波特率
  2. cfsetospeed() 设置 termios_p 指向的 termios 结构中存储的输出波特率为speed。取值必须是以下常量之一:
      B0             B50             B75            B110         B134        B150
      B200        B300           B600          B1200      B1800
      B2400      B4800        B9600        B19200    B38400
      B57600    B115200   B230400
    其中:零值 B0 用来中断连接。如果指定了 B0,不应当再假定存在连接。通常,这样将断开连接。CBAUDEX 是一个掩码,指示高于 POSIX.1 定义的速度的那一些(57600 及以上)。因此,B57600 & CBAUDEX 为非零。
  3. cfgetispeed() 返回 termios 结构中存储的输入波特率。
  4. cfsetispeed() 设置 termios 结构中存储的输入波特率为 speed。如果输入波特率被设为0,实际输入波特率将等于输出波特率。
  5. RETURN VALUE 返回值
    1、cfgetispeed() 返回 termios 结构中存储的输入波特率。
    2、cfgetospeed() 返回 termios 结构中存储的输出波特率。
    3、其他函数返回:
          (1)0:成 功
          (2)-1:失 败,
    并且为 errno 置值来指示错误。

注意 tcsetattr() 返回成功,如果任何所要求的修改可以实现的话。因此,当进行多重修改时,应当在这个函数之后再次调用 tcgetattr() 来检测是否所有修改都成功实现。

示例程序:密码回显‘*’函数

/*密码回显*的函数*/
int inputpsword(char *pass)
{char inletter;struct termios term;  //专门的结构体termiosint i, fd = -1;fd = open("/dev/tty", O_RDWR | O_NOCTTY);    //open一个tty句柄if(fd < 0)return -1;tcgetattr(fd, &term);               //将tty句柄属性get成结构体termiosterm.c_lflag &= ~(ICANON | ECHO);   //设置term为非标准模式的不回显输入方式          tcsetattr(fd, TCSANOW, &term);      //向tty句柄输出任何字符前,必须先执行tcsetattr()for(i = 0; i < 31; i++){read(fd, &inletter, 1);     //读入口令if(inletter != '\n'){pass[i] = inletter;write(fd,"*",1);    //反显口令,显示*}else{pass[i]='\0';write(fd,"\n",1);break;}}if(i == 31){ //到长度限制,手工对字符串变量结尾(’\0’),同时反显”\n”pass[i]='\0';write(fd,"\n",1);}term.c_lflag |= ICANON | ECHO;      //恢复控制终端标准模式的回显功能的输入方式tcsetattr(fd, TCSANOW, &term);close(fd);return 0;
}

struct termios相关推荐

  1. struct termios结构体

    一.数据成员 termios函数簇提供了一个常规的终端接口,用于控制非同步通信端口.这个结构包含了至少下列成员: tcflag_t  c_iflag;//输入模式 tcflag_t  c_oflag; ...

  2. struct termios结构体【转】

    本文转载自:http://blog.csdn.net/vevenlcf/article/details/51096122 一.数据成员 termios 函数族提供了一个常规的终端接口,用于控制非同步通 ...

  3. Linux 串口编程三 使用termios与API进行串口程序开发

    在 termios 结构体以及内部终端控制标志中,并非所有的参数对于实际的物理串口都是有效的,在使用过程中也不需要对于所有标志的作用都有所理解.事实上,快速掌握一项技术的核心点也是一种学习能力.对于使 ...

  4. Linux串口通信之termios结构体说明

    termios结构体中,该结构体一般包括如下的成员: tcflag_t c_iflag; tcflag_t c_oflag; tcflag_t c_cflag; tcflag_t c_lflag; c ...

  5. 串口termios结构体的详细设置

    termios结构体内容: <span style="font-size:18px;">成员 描述 ---------------------------------- ...

  6. linux-011之termios.h头文件

    #ifndef _TERMIOS_H #define _TERMIOS_H#define TTY_BUF_SIZE 1024 //tty缓冲区长度 /* *0x54只是一个魔数,目的是位来使这些常数唯 ...

  7. linux termios结构

    termios 结构是在POSIX规范中定义的标准接口,它类似于系统V中的termio接口,通过设置termios类型的数据结构中的值和使用一小 组函数调用,你就可以对终端接口进行控制. 可以被调整来 ...

  8. linux termios 详解

    一.数据成员                                 termios 函数族提供了一个常规的终端接口,用于控制非同步通信端口. 这个结                 构包含了 ...

  9. Termios 百科

    Termios TERMIOS 解说: Linux中的控制台界面,要大量用到Termios库的内容.但是,总的来看,这些东西已经过时了,应该用更简单的方案来代替它了,尽管它是POSIX标准的一部分. ...

  10. 串口、终端应用程序 API termios

    UART简介 串口全称为串行接口,也称为COM接口,串行接口指的是比特一位位顺序传输,通信线路简单.使用两根线就可以实现双向通信,一条为TX,一个为RX.串口通信距离远,但速度相对慢,是一种常用的工业 ...

最新文章

  1. 如何使用消息队列解决分布式事物?
  2. 高分辨率下IE浏览器缩放导致出现右侧滚动条问题的解决
  3. python中的散点图代码-绘制python中的线和散点图
  4. Android LayoutInflater详解(转)
  5. Mybatis问题:There is no getter for property named 'unitId' in 'class java.lang.String'
  6. libpcap抓取数据包
  7. Linux学习:shell 命令(软件安装和卸载)
  8. json转modbus协议模块_华杰智控Profinet 分布式IO系列模块-华杰智控-新闻中心
  9. windows故障转移群集和mysql_Windows 2016 无域故障转移群集部署方法 超详细图文教程...
  10. 从资深遥控器在家工作的5个技巧
  11. easyui输入框模糊查询
  12. Enterprise Library 2.0 Hands On Lab 翻译(2):数据访问程序块(二)
  13. Axure RP9 安装
  14. 内含干货PPT下载|一站式数据管理 DMS 关键技术解读
  15. kali虚拟机无法发现外置网卡问题
  16. S3C2440之液晶屏LCD驱动
  17. 浏览器怎么打开本地jsp文件
  18. AVD Android虚拟设备root教程
  19. Ubuntu14.04 使用本地摄像头跑ORB SLAM2
  20. javax.servlet.ServletException: Could not resolve view with name ‘***‘ in servlet

热门文章

  1. 《国家网络空间安全战略》发布
  2. echars 柱状图靠左_Echarts柱状图
  3. oppo刷机工具_黔隆科技刷机教程OPPOA1忘记密码免刷机保资料解屏幕锁教程
  4. 记一次渗透之旅 ,网络安全学习至上
  5. spark快速入门java_Spark 快速入门
  6. 生产排程系统_【聚焦】纸箱世界智能制造纸板纸箱厂的高级计划与排程系统应用...
  7. java串口rxtx的若干坑
  8. 在css表格怎么居中对齐,css居中和对齐方法集锦
  9. 轻松入门微信小程序云开发(详细)
  10. python 驱动工控机板卡,研华工控机主板驱动下载