更换 CentOS 7 的下载源为阿里云

1、备份

mv /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/CentOS-Base.repo.backup

2、下载新的CentOS-Base.repo 到/etc/yum.repos.d/

CentOS 5

wget -O /etc/yum.repos.d/CentOS-Base.repo http://mirrors.aliyun.com/repo/Centos-5.repo

CentOS 6

wget -O /etc/yum.repos.d/CentOS-Base.repo http://mirrors.aliyun.com/repo/Centos-6.repo

CentOS 7

wget -O /etc/yum.repos.d/CentOS-Base.repo http://mirrors.aliyun.com/repo/Centos-7.repo

3、生成缓存

yum makecache

http://mirrors.aliyun.com/help/centos

NGINX+fastcgi(cgi)运作原理

1.NGINX服务器原理
nginx在启动后,会有一个master进程和多个worker进程。master进程主要用来管理worker进程。
master主要做以下几件事:

(1).接收来自外界的信号
(2).向各worker进程发送信号
(3).监控worker进程的运行状态,当worker进程退出后(异常情况下),
会自动重新启动新的worker进程。
而基本的网络事件,则是放在worker进程中来处理了。多个worker进程之间是对等,他们同等竞争来自客户端的请求,各进程互相之间是独立 的。一个请求,只可能在一个worker进程中处理,一个worker进程,不可能处理其它进程的请求。worker进程的个数是可以设置的,一般我们会 设置与机器cpu核数一致,这里面的原因与nginx的进程模型以及事件处理模型是分不开的。nginx的进程模型,可以由下图来表示: 

chapter-2-1.png

因为master来管理worker进程,所以我们要操作nginx只需要与master进程通信就行了。master进程会接收来自外界发来的信号,再根据信号做不同的事情。所以我们要控制nginx,只需要通过kill向master进程发送信号就行了。
worker进程又是如何处理请求的呢?我们前面有提到,worker进程之间是平等的,每个进程,处理请求的机会也是一样的。当我们提供80端口的 http服务时,一个连接请求过来,每个进程都有可能处理这个连接,怎么做到的呢?首先,每个worker进程都是从master进程fork过来,在 master进程里面,先建立好需要listen的socket(listenfd)之后,然后再fork出多worker进程。所有worker进程 的listenfd会在新连接到来时变得可读,为保证只有一个进程处理该连接,所有worker进程在注listenfd读事件前抢 accept_mutex,抢到互斥锁的那个进程注册listenfd读事件,在读事件里调用accept接受该连接。当一个worker进程在 accept这个连接之后,就开始读取请求,解析请求,处理请求,产生数据后,再返回给客户端,最后才断开连接,这样一个完整的请求就是这样的了。我们可 以看到,一个请求,完全由worker进程来处理,而且只在一个worker进程中处理。

2.FAST_CGI模型
FAST_CGI实在CGI上改进而来的,在说FAST_CGI之前必须先说说CGI。CGI叫是公共网关接口,它实际上是一段程序。是外部应用程序与Web服务器之间的接口标准,是在CGI程序和Web服务器之间传递信息的规程。上面讲到的nginx程序是不能直接调用外部程序的,要调用外部程序必须使用cig接口。nginx调用CGI的过程是这样的:

    1.浏览器通过HTML表单或超链接请求指向一个CGI应用程序的URL。

    2.nginx服务器收到请求后,nginx服务器的worker进程需要执行执行指定的CGI应用程序(此时CGI初始化,启动CGI应用程序。)

    3.CGI应用程序执行所需要的操作(通常是基于浏览者输入的内容)。

    4.CGI应用程序把结果格式化为nginx或浏览器能够理解的文档(通常是HTML网页),然后终止此CGI应用程序。

    5.nginx把结果返回到浏览器中。
FAST_CGI的过程和CGI差不多,不同在于FAST_CGI在服务器启动时载入FAST_CGI管理程序,这个管理程序自身初始化并启动多个子程序(这些子程序都是CGI程序),然后等待连接到来,当请求到达服务器时,FAST_CGI进程管理器选择并连接到一个CGI进程,同时服务器将CGI环境变量和标准输入发送到CGI程序,此时CGI程序就开始处理请求。处理完成后关闭此请求的连接(CGI程序不关闭继续运行),然后等待下次的请求连接。

nginx指令中的优化(配置文件)

nginx指令中的优化(配置文件)

worker_processes 8;

nginx进程数,建议按照cpu数目来指定,一般为它的倍数。

worker_cpu_affinity 00000001 00000010 00000100 00001000 00010000 00100000 01000000 10000000;

为每个进程分配cpu,上例中将8个进程分配到8个cpu,当然可以写多个,或者将一个进程分配到多个cpu。

worker_rlimit_nofile 102400;

这个指令是指当一个nginx进程打开的最多文件描述符数目,理论值应该是最多打开文件数(ulimit -n)与nginx进程数相除,但是nginx分配请求并不是那么均匀,所以最好与ulimit -n的值保持一致。

use epoll;

使用epoll的I/O模型,这个不用说了吧。

worker_connections 102400;

每个进程允许的最多连接数,理论上每台nginx服务器的最大连接数为worker_processes*worker_connections。

keepalive_timeout 60;

keepalive超时时间。

client_header_buffer_size 4k;

客户端请求头部的缓冲区大小,这个可以根据你的系统分页大小来设置,一般一个请求的头部大小不会超过1k,不过由于一般系统分页都要大于1k,所以这里设置为分页大小。分页大小可以用命令getconf PAGESIZE取得。

open_file_cache max=102400 inactive=20s;

这个将为打开文件指定缓存,默认是没有启用的,max指定缓存数量,建议和打开文件数一致,inactive是指经过多长时间文件没被请求后删除缓存。

open_file_cache_valid 30s;

这个是指多长时间检查一次缓存的有效信息。

open_file_cache_min_uses 1;

open_file_cache指令中的inactive参数时间内文件的最少使用次数,如果超过这个数字,文件描述符一直是在缓存中打开的,如上例,如果有一个文件在inactive时间内一次没被使用,它将被移除。

内核参数的优化

net.ipv4.tcp_max_tw_buckets = 6000

timewait的数量,默认是180000。

net.ipv4.ip_local_port_range = 1024    65000

允许系统打开的端口范围。

net.ipv4.tcp_tw_recycle = 1

启用timewait快速回收。

net.ipv4.tcp_tw_reuse = 1

开启重用。允许将TIME-WAIT sockets重新用于新的TCP连接。

net.ipv4.tcp_syncookies = 1

开启SYN Cookies,当出现SYN等待队列溢出时,启用cookies来处理。

net.core.somaxconn = 262144

web应用中listen函数的backlog默认会给我们内核参数的net.core.somaxconn限制到128,而nginx定义的NGX_LISTEN_BACKLOG默认为511,所以有必要调整这个值。

net.core.netdev_max_backlog = 262144

每个网络接口接收数据包的速率比内核处理这些包的速率快时,允许送到队列的数据包的最大数目。

net.ipv4.tcp_max_orphans = 262144

系统中最多有多少个TCP套接字不被关联到任何一个用户文件句柄上。如果超过这个数字,孤儿连接将即刻被复位并打印出警告信息。这个限制仅仅是为了防止简单的DoS攻击,不能过分依靠它或者人为地减小这个值,更应该增加这个值(如果增加了内存之后)。

net.ipv4.tcp_max_syn_backlog = 262144

记录的那些尚未收到客户端确认信息的连接请求的最大值。对于有128M内存的系统而言,缺省值是1024,小内存的系统则是128。

net.ipv4.tcp_timestamps = 0

时间戳可以避免序列号的卷绕。一个1Gbps的链路肯定会遇到以前用过的序列号。时间戳能够让内核接受这种“异常”的数据包。这里需要将其关掉。

net.ipv4.tcp_synack_retries = 1

为了打开对端的连接,内核需要发送一个SYN并附带一个回应前面一个SYN的ACK。也就是所谓三次握手中的第二次握手。这个设置决定了内核放弃连接之前发送SYN+ACK包的数量。

net.ipv4.tcp_syn_retries = 1

在内核放弃建立连接之前发送SYN包的数量。

net.ipv4.tcp_fin_timeout = 1

如果套接字由本端要求关闭,这个参数决定了它保持在FIN-WAIT-2状态的时间。对端可以出错并永远不关闭连接,甚至意外当机。缺省值是60秒。2.2 内核的通常值是180秒,你可以按这个设置,但要记住的是,即使你的机器是一个轻载的WEB服务器,也有因为大量的死套接字而内存溢出的风险,FIN- WAIT-2的危险性比FIN-WAIT-1要小,因为它最多只能吃掉1.5K内存,但是它们的生存期长些。

net.ipv4.tcp_keepalive_time = 30

当keepalive起用的时候,TCP发送keepalive消息的频度。缺省是2小时。

一个完整的内核优化配置

net.ipv4.ip_forward = 0
net.ipv4.conf.default.rp_filter = 1
net.ipv4.conf.default.accept_source_route = 0
kernel.sysrq = 0
kernel.core_uses_pid = 1
net.ipv4.tcp_syncookies = 1
kernel.msgmnb = 65536
kernel.msgmax = 65536
kernel.shmmax = 68719476736
kernel.shmall = 4294967296
net.ipv4.tcp_max_tw_buckets = 6000
net.ipv4.tcp_sack = 1
net.ipv4.tcp_window_scaling = 1
net.ipv4.tcp_rmem = 4096        87380   4194304
net.ipv4.tcp_wmem = 4096        16384   4194304
net.core.wmem_default = 8388608
net.core.rmem_default = 8388608
net.core.rmem_max = 16777216
net.core.wmem_max = 16777216
net.core.netdev_max_backlog = 262144
net.core.somaxconn = 262144
net.ipv4.tcp_max_orphans = 3276800
net.ipv4.tcp_max_syn_backlog = 262144
net.ipv4.tcp_timestamps = 0
net.ipv4.tcp_synack_retries = 1
net.ipv4.tcp_syn_retries = 1
net.ipv4.tcp_tw_recycle = 1
net.ipv4.tcp_tw_reuse = 1
net.ipv4.tcp_mem = 94500000 915000000 927000000
net.ipv4.tcp_fin_timeout = 1
net.ipv4.tcp_keepalive_time = 30
net.ipv4.ip_local_port_range = 1024    65000

一个简单的nginx优化配置文件

user  www www;
worker_processes 8;
worker_cpu_affinity 00000001 00000010 00000100 00001000 00010000 00100000 01000000;
error_log  /www/log/nginx_error.log  crit;
pid        /usr/local/nginx/nginx.pid;
worker_rlimit_nofile 204800;

events
{
  use epoll;
  worker_connections 204800;
}

http
{
  include       mime.types;
  default_type  application/octet-stream;

  charset  utf-8;

  server_names_hash_bucket_size 128;
  client_header_buffer_size 2k;
  large_client_header_buffers 4 4k;
  client_max_body_size 8m;

  sendfile on;
  tcp_nopush     on;

  keepalive_timeout 60;

  fastcgi_cache_path /usr/local/nginx/fastcgi_cache levels=1:2
                keys_zone=TEST:10m
                inactive=5m;
  fastcgi_connect_timeout 300;
  fastcgi_send_timeout 300;
  fastcgi_read_timeout 300;
  fastcgi_buffer_size 16k;
  fastcgi_buffers 16 16k;
  fastcgi_busy_buffers_size 16k;
  fastcgi_temp_file_write_size 16k;
  fastcgi_cache TEST;
  fastcgi_cache_valid 200 302 1h;
  fastcgi_cache_valid 301 1d;
  fastcgi_cache_valid any 1m;
  fastcgi_cache_min_uses 1;
  fastcgi_cache_use_stale error timeout invalid_header http_500;
  
  open_file_cache max=204800 inactive=20s;
  open_file_cache_min_uses 1;
  open_file_cache_valid 30s;
  


  tcp_nodelay on;
  
  gzip on;
  gzip_min_length  1k;
  gzip_buffers     4 16k;
  gzip_http_version 1.0;
  gzip_comp_level 2;
  gzip_types       text/plain application/x-javascript text/css application/xml;
  gzip_vary on;


  server
  {
    listen       8080;
    server_name  ad.test.com;
    index index.php index.htm;
    root  /www/html/;

    location /status
    {
        stub_status on;
    }

    location ~ .*\.(php|php5)?$
    {
        fastcgi_pass 127.0.0.1:9000;
        fastcgi_index index.php;
        include fcgi.conf;
    }

    location ~ .*\.(gif|jpg|jpeg|png|bmp|swf|js|css)$
    {
      expires      30d;
    }

    log_format  access  '$remote_addr - $remote_user [$time_local] "$request" '
              '$status $body_bytes_sent "$http_referer" '
              '"$http_user_agent" $http_x_forwarded_for';
    access_log  /www/log/access.log  access;
      }
}

关于FastCGI的几个指令

fastcgi_cache_path /usr/local/nginx/fastcgi_cache levels=1:2 keys_zone=TEST:10m inactive=5m;

这个指令为FastCGI缓存指定一个路径,目录结构等级,关键字区域存储时间和非活动删除时间。

fastcgi_connect_timeout 300;

指定连接到后端FastCGI的超时时间。

fastcgi_send_timeout 300;

向FastCGI传送请求的超时时间,这个值是指已经完成两次握手后向FastCGI传送请求的超时时间。

fastcgi_read_timeout 300;

接收FastCGI应答的超时时间,这个值是指已经完成两次握手后接收FastCGI应答的超时时间。

fastcgi_buffer_size 16k;

指定读取FastCGI应答第一部分需要用多大的缓冲区,这里可以设置为fastcgi_buffers指令指定的缓冲区大小,上面的指令指定它将使用1个16k的缓冲区去读取应答的第一部分,即应答头,其实这个应答头一般情况下都很小(不会超过1k),但是你如果在fastcgi_buffers指令中指定了缓冲区的大小,那么它也会分配一个fastcgi_buffers指定的缓冲区大小去缓存。

fastcgi_buffers 16 16k;

指定本地需要用多少和多大的缓冲区来缓冲FastCGI的应答,如上所示,如果一个PHP脚本所产生的页面大小为256k,则会为其分配16个16k的缓冲区来缓存,如果大于256k,增大于256k的部分会缓存到fastcgi_temp指定的路径中,当然这对服务器负载来说是不明智的方案,因为内存中处理数据速度要快于硬盘,通常这个值的设置应该选择一个你的站点中的php脚本所产生的页面大小的中间值,比如你的站点大部分脚本所产生的页面大小为256k就可以把这个值设置为16 16k,或者4 64k 或者64 4k,但很显然,后两种并不是好的设置方法,因为如果产生的页面只有32k,如果用4 64k它会分配1个64k的缓冲区去缓存,而如果使用64 4k它会分配8个4k的缓冲区去缓存,而如果使用16 16k则它会分配2个16k去缓存页面,这样看起来似乎更加合理。

fastcgi_busy_buffers_size 32k;

这个指令我也不知道是做什么用,只知道默认值是fastcgi_buffers的两倍。

fastcgi_temp_file_write_size 32k;

在写入fastcgi_temp_path时将用多大的数据块,默认值是fastcgi_buffers的两倍。

fastcgi_cache TEST

开启FastCGI缓存并且为其制定一个名称。个人感觉开启缓存非常有用,可以有效降低CPU负载,并且防止502错误。但是这个缓存会引起很多问题,因为它缓存的是动态页面。具体使用还需根据自己的需求。

fastcgi_cache_valid 200 302 1h;
fastcgi_cache_valid 301 1d;
fastcgi_cache_valid any 1m;

为指定的应答代码指定缓存时间,如上例中将200,302应答缓存一小时,301应答缓存1天,其他为1分钟。

fastcgi_cache_min_uses 1;

缓存在fastcgi_cache_path指令inactive参数值时间内的最少使用次数,如上例,如果在5分钟内某文件1次也没有被使用,那么这个文件将被移除。

fastcgi_cache_use_stale error timeout invalid_header http_500;

不知道这个参数的作用,猜想应该是让nginx知道哪些类型的缓存是没用的。 以上为nginx中FastCGI相关参数,另外,FastCGI自身也有一些配置需要进行优化,如果你使用php-fpm来管理FastCGI,可以修改配置文件中的以下值:

<value name="max_children">60</value>

同时处理的并发请求数,即它将开启最多60个子线程来处理并发连接。

<value name="rlimit_files">102400</value>

最多打开文件数。

<value name="max_requests">204800</value>

每个进程在重置之前能够执行的最多请求数。

Nginx 反向代理&负载均衡

当客户端发来http请求时,不会立刻转发到上游服务器,而是把用户请求(包括http包体)完整地接收到Nginx所在服务器的硬盘或内存中,再向上游服务器发起连接,把缓存的客户端请求转发到上游服务器。优缺点。优点:公网和内网,降低上游服务器的并发压力。(P59)

Nginx的这种反向代理方案是为了降低上游服务器的并发能力


Nginx根据客户端IP地址计算(ip_hash)出一个key,将key按照upstream集群里的上游服务器数量进行取模,然后以取模后的结果把请求转发到相应的上游服务器中。

这就确保了同一个客户端的请求只会转发到指定的上游服务器中。–服务器中有这个客户访问的缓存


upstream leokim{
    server 192.168.31.153:80 weight=1;
    server 192.168.31.153:80 weight=1;
}

server {
    listen       8081;
    server_name  localhost;

    #charset koi8-r;
    #access_log  /var/log/nginx/log/host.access.log  main;

    location / {
       proxy_pass http://leokim;
       # root   /usr/share/nginx/html;
       # index  index.html index.htm;
    }
    
...
...
...

这样就行了 nginx设置反向代理很简单


ip_hash和weight不能同时用。

centos7 升级 python 2.7.5 到 python3.6

从官网下载python3的压缩包,解压(以3.6.1版本为例)

创建安装目录(自定义)
sudo mkdir /usr/local/python3

cd 进入解压目录,然后
sudo ./configure –prefix=/usr/local/python3
sudo make
sudo make install

cd 进入/usr/bin
其中有python、python2、python2.7三个文件依次指向后者。
image.png

备份当前默认版本python,如果有需要还可还原:
sudo mv python python.bak

创建python3.6l的新链接(也可建立python3命令以区分,同mac)
sudo ln -s /usr/local/python3/bin/python3.6 /usr/bin/python
这样默认的python版本就替换为python3.6了。

image.png

由于yum使用python2,替换为python3后无法正常工作,
因此修改yum配置文件:
sudo vi /usr/bin/yum
将第一行指定的python版本改为python2.7:
*#!/usr/bin/python改为 #!/usr/bin/python2.7

修改urlgrabber配置文件(网上很多教程都漏了这一步)
sudo vi /usr/libexec/urlgrabber-ext-down
同yum,把头部的python改成python2.7

vim /usr/sbin/firewalld

vim /usr/bin/firewall-cmd

同yum,把头部的python改成python2.7

修改python之后firewall脚本里也都得改

在centos7的python3.6里按back space 显示^H

先查看服务器上的 erase 信号的映射:

image.png

可以看出这里的 erase 信号为 ^?,而发送过去的却是 ^H,这就是敲击 Backspace 时为什么不会删除字符而会显示 ^H 的原因,如果要解决这个问题,可以使用 Ctrl-Backspace 键来发送 ^? 信号来删除字符,但是这样还是太麻烦了,需要用组合键,而最简单的方法就是把 Linux 服务器中的 erase 信号设置为 ^H:

再看一下 erase 信号的映射:

image.png

变为 ^H 了,这样就可以敲击 Backspace 来删除字符,但是这样只是临时的,系统重启过后就会失效,可以把这条命令写入家目录下的 .bash_profile 文件中实现永久修改

centos7 下 vsftp搭建

妈的 filezilla因为之前设置的代理给我坑惨了 链接不到局域网

我反复在办公室的centos7电脑上安装了一下午ftp 卧槽

后来重新按这个一切最简单配置的

安装vsftpd

1、源码安装

2、Yum 源安装

这里用的第2种方式,配制原理都是一样的,安装过程很简单,这里有几点是配制需要注意的

a 、如下图安装vsftp

b、打开/etc/vsftpd如下图

c、如下配制需要修改的   vi vsftpd.conf

1、把任何人都可以访问修改为anonymous_enable=NO

2、添加一句话在里面  userlist_deny=NO   意思是只有user_list这里面的用户可以登陆ftp

3、相反的是 ftpusers这里用户都不允许登陆

4、添ftp用户并指定‘目录’  useradd -d /usr/local/apache/htdocs -s /sbin/nologin ftpuser

5、给ftpuser这个用户添加一个密码 passwdftpuser

6、在user_list文件里面添加上ftpuser这个用户

7、重启vsftpd  service vsftpd restart

8、此时vsftpd基本可以用了,下篇写一下权限问题

C语言宏应用——-#define STR(X) #X

#:会把参数转换为字符串

#define STR(x)    #x#define MAX    100

STR(MAX) 会被扩展成"MAX"

这样就有一个缺陷,如果入参为宏,并不能打印出宏的值(比如上一个例子,只打印出了MAX,并没有打印出MAX的值)

可以将宏扩展下

 

#define _STR(x)    #x#define STR(x)    _STR(x)#define MAX    100

 

STR(MAX)会按照以下顺序替换

STR(100)

_STR(100)

"100"

 

 

 

最后输出100

C语言 宏

1. 不带参数的宏定义:

    宏定义又称为宏代换、宏替换,简称“宏”。

    格式: #define 标识符 字符串

    其中的标识符就是所谓的,也称为“宏名”。

    预处理(预编译)工作也叫做宏展开:将宏名替换为字符串。

    掌握"宏"概念的关键是“换”。一切以换为前提、做任何事情之前先要换,准确理解之前就要“换”。

    即在对相关命令或语句的含义和功能作具体分析之前就要换:

    例:   #define PI 3.1415926   把程序中出现的PI全部换成3.1415926

    说明:

    (1)宏名一般用大写

    (2)使用宏可提高程序的通用性和易读性,减少不一致性,减少输入错误和便于修改。例如:数组大小常用宏定义

    (3)预处理是在编译之前的处理,而编译工作的任务之一就是语法检查,预处理不做语法检查。

    (4)宏定义末尾不加分号;

    (5)宏定义写在函数的花括号外边,作用域为其后的程序,通常在文件的最开头。

    (6)可以用#undef命令终止宏定义的作用域

    (7)宏定义可以嵌套

    (8)字符串" "中永远不包含宏

    (9)宏定义不分配内存,变量定义分配内存。
    
2.  带参数的宏定义:
    除了一般的字符串替换,还要做参数代换

    格式:   #define 宏名(参数表) 字符串

    例如:#define S(a,b) a*b

    area=S(3,2);第一步被换为area=a*b; ,第二步被换为area=3*2;

    类似于函数调用,有一个哑实结合的过程:

    (1)实参如果是表达式容易出问题

    #define S(r) r*r

    area=S(a+b);第一步换为area=r*r;,第二步被换为area=a+b*a+b;

    正确的宏定义是#define S(r) ((r)*(r))

    (2)宏名和参数的括号间不能有空格

    (3)宏替换只作替换,不做计算,不做表达式求解

    (4)函数调用在编译后程序运行时进行,并且。宏替换在编译前进行,不分配内存

    (5)宏的哑实结合不存在类型,也没有类型转换。

    (6)函数只有一个返回值,利用宏则可以设法得到多个值

    (7)宏展开使源程序变长,函数调用不会

    (8)宏展开不占运行时间,只占编译时间,函数调用占运行时间(分配内存、保留现场、值传递、返回值)
3.  宏定义其他冷门、重点知识

    #define用法

    1、 用无参宏定义一个简单的常量

    #define LEN 12

    这个是最常见的用法,但也会出错。

    比如下面几个知识点你会吗?可以看下:

    (1) #define NAME "zhangyuncong"

    程序中有"NAME"则,它会不会被替换呢?

    (2) #define 0x abcd

    可以吗?也就是说,可不可以用把标识符的字母替换成别的东西?

    (3) #define NAME "zhang

    这个可以吗?

    (4) #define NAME "zhangyuncong"

    程序中有上面的宏定义,并且,程序里有句:

    NAMELIST这样,会不会被替换成"zhangyuncong"LIST

    四个题答案都是否定的。

    第一个,""内的东西不会被宏替换。这一点应该大都知道。

    第二个,宏定义前面的那个必须是合法的

    第三个,宏定义也不是说后面东西随便写,不能把字符串的两个""拆开。

    第四个:只替换标识符,不替换别的东西。NAMELIST整体是个标识符,而没有NAME标识符,所以不替换。

    也就是说,这种情况下记住:#define 第一位置第二位置

    (1) 不替换程序中字符串里的东西。

    (2) 第一位置只能是合法的标识符(可以是关键字)

    (3) 第二位置如果有字符串,必须把""配对。

    (4) 只替换与第一位置完全相同的标识符

    还有就是老生常谈的话:记住这是简单的替换而已,不要在中间计算结果,一定要替换出表达式之后再算。

    2、 带参宏一般用法

    比如#define MAX(a,b) ((a)>(b)?(a):(b))

    则遇到MAX(1+2,value)则会把它替换成:

    ((1+2)>(value)?(1+2):(value))

    注意事项和无参宏差不多。

    但还是应注意

    #define FUN(a) "a"

    则,输入FUN(345)会被替换成什么?

    其实,如果这么写,无论宏的实参是什么,都不会影响其被替换成"a"的命运。

    也就是说,""内的字符不被当成形参,即使它和一模一样。

    那么,你会问了,我要是想让这里输入FUN(345)它就替换成"345"该怎么实现呢?

    请看下面关于#的用法

    3、 有参宏定义中#的用法

    #define STR(str) #str

    #用于把宏定义中的参数两端加上字符串的""

    比如,这里STR(my#name)会被替换成"my#name"

    一般由任意字符都可以做形参,但以下情况会出错:

    STR())这样,编译器不会把“)”当成STR()的参数。

    STR(,)同上,编译器不会把“,”当成STR的参数。

    STR(A,B)如果实参过多,则编译器会把多余的参数舍去。(VC++2008为例)

    STR((A,B))会被解读为实参为:(A,B),而不是被解读为两个实参,第一个是(A第二个是B)。       4、 有参宏定义中##的用法

    #define WIDE(str) L##str

    则会将形参str的前面加上L

    比如:WIDE("abc")就会被替换成L"abc"

    如果有#define FUN(a,b) vo##a##b()

    那么FUN(id ma,in)会被替换成void main()

    5、 多行宏定义:

    #define doit(m,n) for(int i=0;i<(n);++i)\

    {\

    m+=i;\

    }