使用supervisor运行kafka

安装 supervisor 参考 supervisor安装

创建并编辑 /etc/supervisord.d/kafka.ini 文件,内容如下:

[program:kafka]
command=/opt/kafka/bin/kafka-server-start.sh /opt/kafka/config/server.properties
environment=JMX_PORT=9999
process_name=%(program_name)s
numprocs=1
directory=/opt/kafka/
autostart=true
autorestart=true
startsecs=10
startretries=5
user=root
redirect_stderr=false
stdout_logfile=/var/log/supervisor/kafka.log
stdout_logfile_maxbytes=100MB
stdout_logfile_backups=10
stdout_capture_maxbytes=10MB
stdout_events_enabled=false
stderr_logfile=/var/log/supervisor/kafka_error.log
stderr_logfile_maxbytes=1MB
stderr_logfile_backups=5
stderr_capture_maxbytes=1MB

载入新加的 kafka.ini配置文件

supervisorctl update

centos7 安装手动安装java环境

  1. 在这儿 “https://www.oracle.com/java/technologies/javase-jdk13-downloads.html” 下载相应的java版本
  2. 讲下载好的java文件上传到你自己的服务器上面,我一般是放在/opt目录下面
  3. cd /opt && tar zxvf jdk-13.0.2_linux-x64_bin.tar.gz && mv jdk-13.0.2 jdk && rm jdk-13.0.2_linux-x64_bin.tar.gz
  4. 将以下内容追加到”/etc/profile”中,再 “source /etc/profile”使文件生效
# jdk setting
export JAVA_HOME=/opt/jdk
export JRE_HOME=$JAVA_HOME/jre
export CLASSPATH=$JAVA_HOME/lib/
export PATH=$PATH:$JAVA_HOME/bin
  1. 最后使用”java -version”检查一下是否安装成功

zookeeper集群搭建

  1. 使用的测试环境如下所示:
ip 系统版本 主机名 zookeeper版本
172.16.50.61 centos7 kafka_zk_01 3.6.0
172.16.50.62 centos7 kafka_zk_02 3.6.0
172.16.50.63 centos7 kafka_zk_03 3.6.0
  1. 安装java环境,参考centos7 安装手动安装java环境
  2. 使用以下命令在三台服务器上面安装zookeeper
cd /opt && wget https://downloads.apache.org/zookeeper/zookeeper-3.6.0/apache-zookeeper-3.6.0-bin.tar.gz
tar zxvf apache-zookeeper-3.6.0-bin.tar.gz && mv apache-zookeeper-3.6.0-bin zookeeper
mkdir -p /opt/zookeeper/data ## 创建这个目录是下面步骤的配置文件使用
mkdir -p /opt/zookeeper/logs ## 创建这个目录是下面步骤的配置文件使用
  1. 在三台服务器上面都创建并编辑这个文件”/opt/zookeeper/conf/zoo.cfg”,写入以下内容:
clientPort=2181
tickTime=2000
initLimit=10
syncLimit=5
dataDir=/opt/zookeeper/data
dataLogDir=/opt/zookeeper/logs
maxClientCnxns=600
autopurge.snapRetainCount=30
autopurge.purgeInterval=1
server.1=172.16.50.61:2888:3888
server.2=172.16.50.62:2888:3888
server.3=172.16.50.63:2888:3888
  1. 在”kafka_zk_01″这台服务器上面运行以下命令 :
echo "1" > /opt/zookeeper/data/myid
  1. 在”kafka_zk_02″这台服务器上面运行以下命令 :
echo "2" > /opt/zookeeper/data/myid
  1. 在”kafka_zk_03″这台服务器上面运行以下命令 :
echo "3" > /opt/zookeeper/data/myid
  1. 创建 “/etc/systemd/system/zookeeper.service” 文件 ,写入以下内容,使用systemd控制zookerper的启动
[Unit]
Description=zookeeper.service
After=network.target

[Service]
Type=forking
WorkingDirectory=/opt/zookeeper/
Environment=ZOO_LOG_DIR=/opt/zookeeper/logs
Environment=ZOOPIDFILE=/opt/zookeeper/logs/zookeeper_server.pid
PIDFile=/opt/zookeeper/logs/zookeeper_server.pid
Environment=PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin:/opt/jdk/bin
ExecStart=/opt/zookeeper/bin/zkServer.sh start
ExecStop=/opt/zookeeper/bin/zkServer.sh stop
ExecReload=/opt/zookeeper/bin/zkServer.sh restart
User=root

[Install]
WantedBy=multi-user.target
  1. 最后使用以下命令启动zookeeper,并开启开机启动功能
systemctl daemon-reload
systemctl start zookeeper
systemctl enable zookeeper

Kafka 概述:深入理解架构[转载]

原地址: https://juejin.im/post/5e217c3fe51d450200787f23

  本文主要讲解 Kafka 是什么、Kafka 的架构包括工作流程和存储机制,以及生产者和消费者,最终大家会掌握 Kafka 中最重要的概念,分别是 broker、producer、consumer、consumer group、topic、partition、replica、leader、follower,这是学会和理解 Kafka 的基础和必备内容。

定义

  Kafka 是一个分布式的基于发布/订阅模式的消息队列(Message Queue),主要应用与大数据实时处理领域。

消息队列

  Kafka 本质上是一个 MQ(Message Queue),使用消息队列的好处? (面试会问)

  1. 解耦:允许我们独立的扩展或修改队列两边的处理过程。
  2. 可恢复性:即使一个处理消息的进程挂掉,加入队列中的消息仍然可以在系统恢复后被处理。
  3. 缓冲:有助于解决生产消息和消费消息的处理速度不一致的情况。
  4. 灵活性&峰值处理能力:不会因为突发的超负荷的请求而完全崩溃,消息队列能够使关键组件顶住突发的访问压力。
  5. 异步通信:消息队列允许用户把消息放入队列但不立即处理它。
发布/订阅模式


  一对多,生产者将消息发布到 topic 中,有多个消费者订阅该主题,发布到 topic 的消息会被所有订阅者消费,被消费的数据不会立即从 topic 清除。

架构


  Kafka 存储的消息来自任意多被称为 Producer 生产者的进程。数据从而可以被发布到不同的 Topic 主题下的不同 Partition 分区。在一个分区内,这些消息被索引并连同时间戳存储在一起。其它被称为 Consumer 消费者的进程可以从分区订阅消息。Kafka 运行在一个由一台或多台服务器组成的集群上,并且分区可以跨集群结点分布。下面给出 Kafka 一些重要概念,让大家对 Kafka 有个整体的认识和感知,后面还会详细的解析每一个概念的作用以及更深入的原理。

  • Producer: 消息生产者,向 Kafka Broker 发消息的客户端。
  • Consumer: 消息消费者,从 Kafka Broker 取消息的客户端。
  • Consumer Group: 消费者组(CG),消费者组内每个消费者负责消费不同分区的数据,提高消费能力。一个分区只能由组内一个消费者消费,消费者组之间互不影响。所有的消费者都属于某个消费者组,即消费者组是逻辑上的一个订阅者。
  • Broker: 一台 Kafka 机器就是一个 broker。一个集群由多个 broker 组成。一个 broker 可以容纳多个 topic。
  • Topic: 可以理解为一个队列,topic 将消息分类,生产者和消费者面向的是同一个 topic。
  • Partition: 为了实现扩展性,提高并发能力,一个非常大的 topic 可以分布到多个 broker (即服务器)上,一个 topic 可以分为多个 partition,每个 partition 是一个 有序的队列。
  • Replica: 副本,为实现备份的功能,保证集群中的某个节点发生故障时,该节点上的 partition 数据不丢失,且 Kafka 仍然能够继续工作,Kafka 提供了副本机制,一个 topic 的每个分区都有若干个副本,一个 leader 和若干个 follower。
  • Leader: 每个分区多个副本的“主”副本,生产者发送数据的对象,以及消费者消费数据的对象,都是 leader。
  • Follower: 每个分区多个副本的“从”副本,实时从 leader 中同步数据,保持和 leader 数据的同步。leader 发生故障时,某个 follower 还会成为新的 leader。
  • offset: 消费者消费的位置信息,监控数据消费到什么位置,当消费者挂掉再重新恢复的时候,可以从消费位置继续消费。
  • Zookeeper: Kafka 集群能够正常工作,需要依赖于 zookeeper,zookeeper 帮助 Kafka 存储和管理集群信息。
工作流程

  Kafka集群将 Record 流存储在称为 topic 的类别中,每个记录由一个键、一个值和一个时间戳组成。Kafka 是一个分布式流平台,这到底是什么意思?

  • 发布和订阅记录流,类似于消息队列或企业消息传递系统。
  • 以容错的持久方式存储记录流。
  • 处理记录流。

      Kafka 中消息是以 topic 进行分类的,生产者生产消息,消费者消费消息,面向的都是同一个 topic。topic 是逻辑上的概念,而 partition 是物理上的概念,每个 partition 对应于一个 log 文件,该 log 文件中存储的就是 Producer 生产的数据。Producer 生产的数据会不断追加到该 log 文件末端,且每条数据都有自己的 offset。消费者组中的每个消费者,都会实时记录自己消费到了哪个 offset,以便出错恢复时,从上次的位置继续消费。
存储机制


  由于生产者生产的消息会不断追加到 log 文件末尾,为防止 log 文件过大导致数据定位效率低下,Kafka 采取了分片和索引机制,将每个 partition 分为多个 segment,每个 segment 对应两个文件:“.index” 索引文件和 “.log” 数据文件。这些文件位于同一文件下,该文件夹的命名规则为:topic 名-分区号。例如,first 这个 topic 有三分分区,则其对应的文件夹为 first-0,first-1,first-2。index 和 log 文件以当前 segment 的第一条消息的 offset 命名。下图为 index 文件 和 log 文件的结构示意图。

  “.index” 文件存储大量的索引信息,“.log” 文件存储大量的数据,索引文件中的元数据指向对应数据文件中 message 的物理偏移量。

生产者

分区策略
分区原因
  • 方便在集群中扩展,每个 partition 可以通过调整以适应它所在的机器,而一个 topic 又可以有多个 partition 组成,因此可以以 partition 为单位读写了。
  • 可以提高并发,因此可以以 partition 为单位读写了。
分区原则

  我们需要将 Producer 发送的数据封装成一个 ProducerRecord 对象。该对象需要指定一些参数:

  • topic:string 类型,NotNull
  • partition:int 类型,可选
  • timestamp:long 类型,可选
  • key:string类型,可选
  • value:string 类型,可选
  • headers:array 类型,Nullable
  1. 指明 partition 的情况下,直接将给定的 value 作为 partition 的值。
  2. 没有指明 partition 但有 key 的情况下,将 key 的 hash 值与分区数取余得到 partition 值。
  3. 既没有 partition 有没有 key 的情况下,第一次调用时随机生成一个整数(后面每次调用都在这个整数上自增),将这个值与可用的分区数取余,得到 partition 值,也就是常说的 round-robin 轮询算法。
数据可靠性保证

  为保证 producer 发送的数据,能可靠地发送到指定的 topic,topic 的每个 partition 收到 producer 发送的数据后,都需要向 producer 发送 ack(acknowledge 确认收到),如果 producer 收到 ack,就会进行下一轮的发送,否则重新发送数据。

副本数据同步策略
  1. 何时发送 ack?确保有 follower 与 leader 同步完成,leader 再发送 ack,这样才能保证 leader 挂掉之后,能在 follower 中选举出新的 leader 而不丢数据。
  2. 多少个 follower 同步完成后发送 ack?全部 follower 同步完成,再发送 ack。
ISR

  采用第二种方案,所有 follower 完成同步,producer 才能继续发送数据,设想有一个 follower 因为某种原因出现故障,那 leader 就要一直等到它完成同步。这个问题怎么解决?leader维护了一个动态的 in-sync replica set(ISR):和 leader 保持同步的 follower 集合。当 ISR 集合中的 follower 完成数据的同步之后,leader 就会给 follower 发送 ack。如果 follower 长时间未向 leader 同步数据,则该 follower 将被踢出 ISR 集合,该时间阈值由 replica.lag.time.max.ms 参数设定。leader 发生故障后,就会从 ISR 中选举出新的 leader。

ack 应答机制

  对于某些不太重要的数据,对数据的可靠性要求不是很高,能够容忍数据的少量丢失,所以没必要等 ISR 中的 follower 全部接受成功。所以 Kafka 为用户提供了三种可靠性级别,用户根据可靠性和延迟的要求进行权衡,选择以下的配置。

  • 0:producer 不等待 broker 的 ack,这提供了最低延迟,broker 一收到数据还没有写入磁盘就已经返回,当 broker 故障时有可能丢失数据。
  • 1:producer 等待 broker 的 ack,partition 的 leader 落盘成功后返回 ack,如果在 follower 同步成功之前 leader 故障,那么将会丢失数据。
  • -1(all):producer 等待 broker 的 ack,partition 的 leader 和 follower 全部落盘成功后才返回 ack。但是在 broker 发送 ack 时,leader 发生故障,则会造成数据重复。
故障处理细节


  HW,HighWatermark,高水位,表示 Consumer 可以消费到的最高 Partition 偏移量。HW 保证了 Kafka 集群中消息的一致性。确切地说,是保证了 Partition 的 Follower 与 Leader 间数 据的一致性。LEO,Log End Offset,日志最后消息的偏移量。消息是被写入到 Kafka 的日志文件中的, 这是当前最后一个写入的消息在 Partition 中的偏移量。对于 Leader 新写入的消息,Consumer 是不能立刻消费的。Leader 会等待该消息被所有 ISR 中的 Partition Follower 同步后才会更新 HW,此时消息才能被 Consumer 消费。ISR 队列中最小的 LEO。
  Coordinator:一般指的是运行在每个 Broker 上的 Group Coordinator 进程,用于管理 Consumer Group 中的各个成员,主要用于 Offset 位移管理和 Rebalance。一个 Coordinator 可以同时管理多个消费者组。Rebalance:当消费者组中的数量发生变化,或者 Topic 中的 Partition 数量发生了变化时,Partition 的所有权会在消费者间转移,即 Partition 会重新分配,这个过程称为再均衡 Rebalance。再均衡能够给消费者组及 Broker 带来高性能、高可用性和伸缩,但在再均衡期间消费者是无法读取消息的,即整个 Broker 集群有小一段时间是不可用的。因此要避免不必要的再均衡。Offset Commit:Consumer 从 Broker 中取一批消息写入 Buffer 进行消费,在规定的时间内消费完消息后,会自动将其消费消息的 Offset 提交给 Broker,以记录下哪些消息是消费过的。当然,若在时限内没有消费完毕,其是不会提交 Offset 的。

  1. Follower 故障;follower 发生故障后会被临时踢出 ISR 集合,待该 follower 恢复后,follower 会 读取本地磁盘记录的上次的 HW,并将 log 文件高于 HW 的部分截取掉,从 HW 开始向 leader 进行同步数据操作。等该 follower 的 LEO 大于等于该 partition 的 HW,即 follower 追上 leader 后,就可以重新加入 ISR 了。
  2. Leader 故障;leader 发生故障后,会从 ISR 中选出一个新的 leader,之后,为保证多个副本之间的数据一致性,其余的 follower 会先将各自的 log 文件高于 HW 的部分截掉,然后从新的 leader 同步数据。注意:这只能保证副本之间的数据一致性,并不能保证数据不丢失或者不重复。
Exactly Once 语义

  将服务器的 ACK 级别设置为-1,可以保证 producer 到 server 之间不会丢失数据,即 At Least Once 语义。相对的,将服务器 ACK 级别设置为0,可以保证生产者每条消息只会被发送一次,即At Most Once 语义。At Least Once 可以保证数据不丢失,但是不能保证数据不重复;相对的,At Most Once 可以保证数据不重复,但是不能保证数据不丢失。但是,对于一些非常重要的信息,比如交易数据,下游数据消费者要求数据既不重复也不丢失,即 Exactly Once 语义。0.11版本的 Kafka,引入了幂等性:producer 不论向 server 发送多少重复数据,server 端都只会持久化一条。即:

At Least Once + 幂等性 = Exactly Once

  要启用幂等性,只需要将 producer 的参数中 enable.idompotence 设置为 true 即可。 如果启用了幂等性,则ack默认就是-1。开启幂等性的 producer 在初始化时会被分配一个 PID,发往同一 partition 的消息会附带 Sequence Number。而 borker 端会对 <PID,Partition,SeqNumber> 做缓存,当具有相同主键的消息提交时,broker 只会持久化一条。但是 PID 重启后就会变化,同时不同的 partition 也具有不同主键,所以幂等性无法保证跨分区会话的 Exactly Once。

消费者

消费方式

  consumer 采用 pull(拉取)模式从 broker 中读取数据。consumer 采用 push(推送)模式,broker 给 consumer 推送消息的速率是由 broker 决定的,很难适应消费速率不同的消费者。它的目标是尽可能以最快速度传递消息,但是这样很容易造成 consumer 来不及处理消息,典型的表现就是拒绝服务以及网络拥塞。而 pull 模式则可以根据 consumer 的消费能力以适当的速率消费消息。pull 模式不足之处是,如果 Kafka 没有数据,消费者可能会陷入循环中,一直返回空数据。因为消费者从 broker 主动拉取数据,需要维护一个长轮询,针对这一点, Kafka 的消费者在消费数据时会传入一个时长参数 timeout,如果当前没有数据可供消费,consumer 会等待一段时间之后再返回,这段时长即为 timeout。

分区分配策略

  一个 consumer group 中有多个 consumer,一个 topic 有多个 partition,所以必然会涉及到 partition 的分配问题,即确定哪个 partition 由哪个 consumer 来消费。Kafka 有两种分配策略,一个是 RoundRobin,一个是 Range,默认为range,当消费者组内消费者发生变化时,会触发分区分配策略(方法重新分配)。

RoundRobin


  RoundRobin 轮询方式将分区所有作为一个整体进行 hash 排序,消费者组内分配分区个数最大差别为1,是按照组来分的,可以解决多个消费者消费数据不均衡的问题。但是,当消费者组内订阅不同主题时,可能造成消费混乱,如下图所示,consumer0 订阅主题A,consumer1 订阅主题B,将 A、B主题的分区排序后分配给消费者组,TopicB 分区中的数据可能分配到 consumer0 中。

Range


  range 方式是按照主题来分的,不会产生轮询方式的消费混乱问题。但是,如下图所示,consumer0、consumer1 同时订阅了主题A和B,可能造成消息分配不对等问题,当消费者组内订阅的主题越多,分区分配可能越不均衡。

offset 的维护

  由于 consumer 在消费过程中可能会出现断电宕机等故障,consumer 恢复后,需要从故障前的位置继续消费,所以 consumer 需要实时记录自己消费到了哪个 offset,以便故障恢复后继续消费。Kafka 0.9 版本之前,consumer 默认将 offset 保存在 Zookeeper 中,从 0.9 版本开始,consumer 默认将 offset 保存在 Kafka 一个内置的 topic 中,该 topic 为 __consumer_offsets。

如果生命只剩100天[转载]

原地址: https://mp.weixin.qq.com/s/UeVRxSjekHo8M_SQ4kL03Q

  在我寻求康复时,竟然在登山步行道上与父亲的偶遇,这个不可思议的巧合,让近年来发生在我身上的许多事件,仿佛都有了一种奇妙的因果联系。中国人讲“身体发肤,受之父母,不敢毁伤”,我不仅毁伤、糟蹋了,还让自己险些丢了性命。每次站在父亲灵前,总有历劫归来、大难不死的庆幸与忏悔。想当初,放射科医生看到我腹部有二十几个肿瘤,连他都吓得脸色发白,不敢正面看我,我就像被正式宣判死刑一样,先前还期待能够侥幸逃过厄运,一下子希望全部落空了,摆在眼前的冷冰冰的结局就是死期将至,我可能只剩100天好活。

  100天,那可是一眨眼就会过去的!无数个清晨黑夜,我睁大了眼,唯恐一闭上眼,我能看到这个世界的机会就一分一秒地减少了。伤心、绝望、懊悔、愤怒、跟老天爷讨价还价……各种情绪轮番在我的脑海里翻滚煎熬。我苦苦闷着、撑着,像一头受伤的野兽,被关在狭小的牢笼里。全世界都远去了,我过去所在意的一切、一切全都远去了,只剩下几件看似寻常的小事,在那个时刻,却活生生地跳到眼前、心上,催促着我:“你还有多少时间迟疑呢!你再不做就来不及了!”

  我脑海里一遍一遍地想到先铃,想到孩子,想到母亲和哥哥姐姐,也想到几位好朋友,我还想到了我错过的许多短暂的美好时刻。过去,我总觉得时间还很多—等我准备好这个演讲,做完那个采访,忙完这件投资案子;等我把每天发的微博内容都处理好。所以每件事都比这些“小事”重要。结果到头来,在我的生命仅存最后的100天时,我才发现,我这一生最大的错误是,我彻头彻尾地舍本逐末,把最要紧的事搁到最后,却把人生最弥足珍贵的时光,浪费在追逐那些看起来五彩斑斓的泡沫。

  过去我曾在美国教会学校就读,而且在以基督教为主的美国生活了三十多年,受到耳濡目染,一定程度上都认为人生只有一次。如果人生只有一次,那么人生当然要分秒必争,而且要无所不用其极地做到最大化影响力、最大化效率。在这样的信念之下,我不断挑选、改变人生跑道。从卡内基·梅隆大学(CMU)到苹果,因为我觉得蹲在研究室里写论文不能最大化影响力;加入微软回到中国,一方面因为这是我父亲期望我做的,另一方面是因为中国地大物博、人口众多,而当时的环境也充满机会,我如果回去,可以产生一定程度的影响力。所以我写了七封给年轻人的信,出版了五本书,发过一万多条微博,举行了五百场演讲……一切都是为了给年轻人带来正面的影响。后来我加入谷歌,是为了学会如何打造顶尖的网络产品;离开谷歌创办创新工场,则是希望用我的专长来帮助年轻人,做出可以产生实质利益的产品。

  我信心满满地到处宣扬我的理念,我建议年轻人要做最好的自己、要最大化影响力;我鼓励年轻人要积极主动、寻找兴趣、建立正确的价值观;我还用理想中的墓志铭来确认我的人生方向……但是,一帆风顺的人生履历,让我的骄傲情绪悄悄滋生;理工科培养出来的思维模式,包括因果逻辑、结果导向和一切以量化判断……让我在追求效率时变得冷漠无情。我走在一条其实颇为正确的道路上,但是,过度的名声却让我的中心轴偏了。乔布斯曾说过:“记住你即将死去。”这句话如今已成为我的座右铭, 每天提醒我看清楚什么才是生命中重要的选择;因为所有的荣耀与骄傲、难堪与恐惧,都会在死亡面前消失,只留下真正重要的东西。如果觉察到自己沉溺于担心会失去某些东西时,“记住你即将死去”会是最好的解药。

  我曾以为微软官司是我这一生最极端的炼狱,经历过那段恐怖时光,一切挑战都显得微不足道。但是经历过死亡的威胁与病痛的折磨,微软官司当时所担心的名誉受损、工作生涯等问题,已经毫无意义,人生更大的挑战是如何克服面对死亡的恐惧,以及如果生命只剩100 天了,我会怎么做?在夹杂着悲伤、愤怒、绝望和追悔的情绪里,茫然四顾,但死亡的紧迫感却提醒我,无论如何要在生命最后的时刻,好好地做几件事:

  1. 让我的亲人、朋友知道我真心爱他们,是他们让我的生命充满了温暖和光辉。
  2. 我要跟他们一起度过难忘的时光,让我们彼此的生命都记住在那个时刻里我们互放的光亮。
  3. 我要在活着的每一个时刻都是全心全意地活着,我不会再花心思去臆测、追想那些还没来到或者已经远去的事。

一生都在照顾临终病人的护士邦妮·韦尔(Bonnie Ware)也说,人在临终时最后悔的五件事是:

  1. 我希望当初有勇气过自己真正想要的生活,而不是别人希望我过的生活。
  2. 我希望当初我没有花这么多精力在工作上。
  3. 我希望当初我能有勇气表达我的感受。
  4. 我希望当初我能和朋友保持联系。
  5. 我希望当初我能让自己活得更开心一点儿。

  病中我不只一次想过,如果我的人生将要走到尽头,我心中不想亏欠任何人,真心希望能够用我的余生来弥补爱我的人对我所有的付出,希望我的亲人、朋友、帮助过我的人……他们会觉得认识我是值得的;我们之间的相处、互动,可以成为最闪亮、最美好的回忆。如果人生只剩100天,我会和先铃一起回忆我们共同度过的美好与艰辛时光;我会和先铃再回到匹兹堡学习教堂的无边草地上,带着我们自己做的波兰香肠三明治,还有附近的炸蔬菜船和她最爱喝的桃子鸡尾酒,回忆我们学生时代简单和快乐的生活。回忆我们还是穷学生时,如何在河边无照偷偷钓鱼,到电影院一天看六部影片,看到想吐;减价时大采购,结果遇到大雪拿不动,只好把一块块冻成球的肉从山坡上滚下来。我一定要让她知道,这一生因为有她相伴,我的人生是如此丰富!我会带最喜欢熊的德宁到泰迪熊博物馆(Teddy Bear Museum)的咖啡馆和她聊天,听她讲讲服装设计界又发生了什么新奇的事件,也听听她对男朋友的看法。我还要跟德亭再去一次威尼斯,大吃有名的Gelato Fantasy 冰激凌,坐在运河上的贡多拉,帮她取景拍照。我也一定要约我的室友罗斯见面,跟他去买25 公斤的奶酪,做上10 个奶酪蛋糕,吃到我们想吐为止,然后重温我们过去的每一次恶搞……

  至于母亲,我会躺在她大大的肚子上,一张一张翻看我们的老照片,再一遍又一遍地听她说起当年如何如何。然后我要告诉她我是多么爱她, 我愿生生世世做她的孩子。我还会到父亲灵前,告诉他我终于明白了,他希望我做的,嘴上虽然没说,但他都做给我看了,我也终于了解,人到无求品自高,人生应为所当为,若将名利挂心头,便如苍蝇追逐腐肉,把人生的追求浪费在满足最低层级的欲望上。我希望跟我有缘的人能够跟我一样感恩这样美好的缘分, 对未来有很多乐观正面的思考,那该有多么好!而且,只要好好体验人生、享受世界的真善美,让自己的生命不断升华成长,不必留下什么,这个世界就会因你而芬芳。若真要留下什么,那就是留下我们的孩子。如果真要衡量什么,一个善良的后代,能带给世界的正面影响,一定会超过邪恶的人。

  当大家都有相同的体会时,就很接近真理了。当我确定自己患有淋巴癌第四期,并不会立即出现生命危险时,我还有机会重拾健康、弥补过去的缺失,庆幸之余,我就忍不住想,既然对“如果生命只剩最后100天”已经有过缜密的思考,为什么不从现在开始每天都这么过呢?

(全文摘自我的新书《向死而生:我修的死亡学分》,书的链接:http://item.jd.com/11709327.html)