mirror of
https://github.com/LCTT/TranslateProject.git
synced 2025-01-13 22:30:37 +08:00
commit
b6a6cdac22
95
README.md
95
README.md
@ -5,7 +5,7 @@ LCTT 是“Linux中国”([https://linux.cn/](https://linux.cn/))的翻译
|
|||||||
|
|
||||||
LCTT 已经拥有几百名活跃成员,并欢迎更多的Linux志愿者加入我们的团队。
|
LCTT 已经拥有几百名活跃成员,并欢迎更多的Linux志愿者加入我们的团队。
|
||||||
|
|
||||||
![logo](http://img.linux.net.cn/static/image/common/lctt_logo.png)
|
![logo](https://linux.cn/static/image/common/lctt_logo.png)
|
||||||
|
|
||||||
LCTT 的组成
|
LCTT 的组成
|
||||||
-------------------------------
|
-------------------------------
|
||||||
@ -38,7 +38,7 @@ LCTT 的组成
|
|||||||
* 2013/09/16 公开发布了翻译组成立消息后,又有新的成员申请加入了。并从此建立见习成员制度。
|
* 2013/09/16 公开发布了翻译组成立消息后,又有新的成员申请加入了。并从此建立见习成员制度。
|
||||||
* 2013/09/24 鉴于大家使用 GitHub 的水平不一,容易导致主仓库的一些错误,因此换成了常规的 fork+PR 的模式来进行翻译流程。
|
* 2013/09/24 鉴于大家使用 GitHub 的水平不一,容易导致主仓库的一些错误,因此换成了常规的 fork+PR 的模式来进行翻译流程。
|
||||||
* 2013/10/11 根据对 LCTT 的贡献,划分了 Core Translators 组,最先的加入成员是 vito-L 和 tinyeyeser。
|
* 2013/10/11 根据对 LCTT 的贡献,划分了 Core Translators 组,最先的加入成员是 vito-L 和 tinyeyeser。
|
||||||
* 2013/10/12 取消对 LINUX.CN 注册用户的依赖,在 QQ 群内、文章内都采用 GitHub 的注册 ID。
|
* 2013/10/12 取消对 LINUX.CN 注册用户的关联,在 QQ 群内、文章内都采用 GitHub 的注册 ID。
|
||||||
* 2013/10/18 正式启动 man 翻译计划。
|
* 2013/10/18 正式启动 man 翻译计划。
|
||||||
* 2013/11/10 举行第一次北京线下聚会。
|
* 2013/11/10 举行第一次北京线下聚会。
|
||||||
* 2014/01/02 增加了 Core Translators 成员: geekpi。
|
* 2014/01/02 增加了 Core Translators 成员: geekpi。
|
||||||
@ -52,7 +52,11 @@ LCTT 的组成
|
|||||||
* 2015/04/19 发起 LFS-BOOK-7.7-systemd 项目。
|
* 2015/04/19 发起 LFS-BOOK-7.7-systemd 项目。
|
||||||
* 2015/06/09 提升 ictlyh 和 dongfengweixiao 为 Core Translators 成员。
|
* 2015/06/09 提升 ictlyh 和 dongfengweixiao 为 Core Translators 成员。
|
||||||
* 2015/11/10 提升 strugglingyouth、FSSlc、Vic020、alim0x 为 Core Translators 成员。
|
* 2015/11/10 提升 strugglingyouth、FSSlc、Vic020、alim0x 为 Core Translators 成员。
|
||||||
|
* 2016/02/18 由于选题 DeadFire 重病,任命 oska874 接手选题工作。
|
||||||
|
* 2016/02/29 选题 DeadFire 病逝。
|
||||||
* 2016/05/09 提升 PurlingNayuki 为校对。
|
* 2016/05/09 提升 PurlingNayuki 为校对。
|
||||||
|
* 2016/09/10 LCTT 三周年。
|
||||||
|
* 2016/12/24 拟定 LCTT [Core 规则](core.md),并增加新的 Core 成员: @ucasFL、@martin2011qi,及调整一些组。
|
||||||
|
|
||||||
活跃成员
|
活跃成员
|
||||||
-------------------------------
|
-------------------------------
|
||||||
@ -60,121 +64,108 @@ LCTT 的组成
|
|||||||
目前 TP 活跃成员有:
|
目前 TP 活跃成员有:
|
||||||
- Leader @wxy,
|
- Leader @wxy,
|
||||||
- Source @oska874,
|
- Source @oska874,
|
||||||
- Proofreader @PurlingNayuki,
|
- Proofreaders @jasminepeng,
|
||||||
- CORE @geekpi,
|
- CORE @geekpi,
|
||||||
- CORE @GOLinux,
|
- CORE @GOLinux,
|
||||||
- CORE @ictlyh,
|
- CORE @ictlyh,
|
||||||
- CORE @carolinewuyan,
|
|
||||||
- CORE @strugglingyouth,
|
- CORE @strugglingyouth,
|
||||||
- CORE @FSSlc
|
- CORE @FSSlc
|
||||||
- CORE @zpl1025,
|
- CORE @zpl1025,
|
||||||
- CORE @runningwater,
|
- CORE @runningwater,
|
||||||
- CORE @bazz2,
|
- CORE @bazz2,
|
||||||
- CORE @Vic020,
|
- CORE @Vic020,
|
||||||
- CORE @dongfengweixiao,
|
|
||||||
- CORE @alim0x,
|
- CORE @alim0x,
|
||||||
|
- CORE @tinyeyeser,
|
||||||
|
- CORE @Locez,
|
||||||
|
- CORE @ucasFL
|
||||||
|
- CORE @martin2011qi
|
||||||
- Senior @DeadFire,
|
- Senior @DeadFire,
|
||||||
- Senior @reinoir222,
|
- Senior @reinoir222,
|
||||||
- Senior @tinyeyeser,
|
|
||||||
- Senior @vito-L,
|
- Senior @vito-L,
|
||||||
- Senior @jasminepeng,
|
|
||||||
- Senior @willqian,
|
- Senior @willqian,
|
||||||
- Senior @vizv,
|
- Senior @vizv,
|
||||||
|
- Senior @dongfengweixiao,
|
||||||
|
- Senior @PurlingNayuki,
|
||||||
|
- Senior @carolinewuyan,
|
||||||
|
- cposture,
|
||||||
- ZTinoZ,
|
- ZTinoZ,
|
||||||
- martin2011qi,
|
|
||||||
- theo-l,
|
- theo-l,
|
||||||
- Luoxcat,
|
- Luoxcat,
|
||||||
|
- GHLandy,
|
||||||
- wi-cuckoo,
|
- wi-cuckoo,
|
||||||
|
- StdioA,
|
||||||
- disylee,
|
- disylee,
|
||||||
- haimingfg,
|
|
||||||
- KayGuoWhu,
|
|
||||||
- wwy-hust,
|
- wwy-hust,
|
||||||
- felixonmars,
|
- felixonmars,
|
||||||
|
- KayGuoWhu,
|
||||||
|
- mr-ping,
|
||||||
|
- wyangsun,
|
||||||
- su-kaiyao,
|
- su-kaiyao,
|
||||||
- GHLandy,
|
|
||||||
- ivo-wang,
|
- ivo-wang,
|
||||||
- cvsher,
|
- cvsher,
|
||||||
- wyangsun,
|
- OneNewLife
|
||||||
- DongShuaike,
|
- DongShuaike,
|
||||||
- flsf,
|
- flsf,
|
||||||
- SPccman,
|
- SPccman,
|
||||||
- Stevearzh
|
- Stevearzh,
|
||||||
- mr-ping,
|
- bestony,
|
||||||
- Linchenguang,
|
- Linchenguang,
|
||||||
- Linux-pdz,
|
- Linux-pdz,
|
||||||
- 2q1w2007,
|
- 2q1w2007,
|
||||||
|
- NearTan,
|
||||||
- H-mudcup,
|
- H-mudcup,
|
||||||
- cposture,
|
- GitFuture,
|
||||||
|
- MikeCoder,
|
||||||
- xiqingongzi,
|
- xiqingongzi,
|
||||||
- goreliu,
|
- goreliu,
|
||||||
- NearTan,
|
- rusking,
|
||||||
|
- jiajia9linuxer,
|
||||||
|
- name1e5s,
|
||||||
- TxmszLou,
|
- TxmszLou,
|
||||||
- ZhouJ-sh,
|
- ZhouJ-sh,
|
||||||
- wangjiezhe,
|
- wangjiezhe,
|
||||||
- icybreaker,
|
- icybreaker,
|
||||||
|
- zky001,
|
||||||
|
- vim-kakali,
|
||||||
- shipsw,
|
- shipsw,
|
||||||
|
- LinuxBars,
|
||||||
|
- Moelf,
|
||||||
|
- Chao-zhi
|
||||||
- johnhoow,
|
- johnhoow,
|
||||||
- soooogreen,
|
- soooogreen,
|
||||||
|
- kokialoves,
|
||||||
- linuhap,
|
- linuhap,
|
||||||
|
- ChrisLeeGit,
|
||||||
- blueabysm,
|
- blueabysm,
|
||||||
|
- yangmingming,
|
||||||
- boredivan,
|
- boredivan,
|
||||||
- name1e5s,
|
|
||||||
- StdioA,
|
|
||||||
- yechunxiao19,
|
- yechunxiao19,
|
||||||
- l3b2w1,
|
|
||||||
- XLCYun,
|
- XLCYun,
|
||||||
- KevinSJ,
|
- KevinSJ,
|
||||||
|
- l3b2w1,
|
||||||
- tenght,
|
- tenght,
|
||||||
|
- firstadream,
|
||||||
- coloka,
|
- coloka,
|
||||||
- luoyutiantang,
|
- luoyutiantang,
|
||||||
- sonofelice,
|
- sonofelice,
|
||||||
- jiajia9linuxer,
|
|
||||||
- scusjs,
|
- scusjs,
|
||||||
- tnuoccalanosrep,
|
|
||||||
- woodboow,
|
- woodboow,
|
||||||
- 1w2b3l,
|
- 1w2b3l,
|
||||||
- JonathanKang,
|
- JonathanKang,
|
||||||
- crowner,
|
- crowner,
|
||||||
- dingdongnigetou,
|
- dingdongnigetou,
|
||||||
- mtunique,
|
- mtunique,
|
||||||
- CNprober,
|
|
||||||
- hyaocuk,
|
- hyaocuk,
|
||||||
- szrlee,
|
- szrlee,
|
||||||
- KnightJoker,
|
|
||||||
- Xuanwo,
|
|
||||||
- nd0104,
|
- nd0104,
|
||||||
- Moelf,
|
- chenzhijun,
|
||||||
- xiaoyu33,
|
- frankatlingingdigital,
|
||||||
- guodongxiaren,
|
- willcoderwang,
|
||||||
- ynmlml,
|
|
||||||
- vim-kakali,
|
|
||||||
- ggaaooppeenngg,
|
|
||||||
- Ricky-Gong,
|
|
||||||
- zky001,
|
|
||||||
- lfzark,
|
|
||||||
- 213edu,
|
|
||||||
- bestony,
|
|
||||||
- mudongliang,
|
|
||||||
- liuaiping,
|
- liuaiping,
|
||||||
- Timeszoro,
|
|
||||||
- rogetfan,
|
- rogetfan,
|
||||||
- JeffDing,
|
- JeffDing,
|
||||||
- Yuking-net,
|
|
||||||
|
|
||||||
|
(按增加行数排名前百,更新于2016/12/24)
|
||||||
(按增加行数排名前百)
|
|
||||||
|
|
||||||
LFS 项目活跃成员有:
|
|
||||||
|
|
||||||
- @ictlyh
|
|
||||||
- @dongfengweixiao
|
|
||||||
- @wxy
|
|
||||||
- @H-mudcup
|
|
||||||
- @zpl1025
|
|
||||||
- @KevinSJ
|
|
||||||
- @Yuking-net
|
|
||||||
|
|
||||||
(更新于2016/06/20)
|
|
||||||
|
|
||||||
谢谢大家的支持!
|
谢谢大家的支持!
|
||||||
|
|
||||||
|
39
core.md
Normal file
39
core.md
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
给核心成员的一封信
|
||||||
|
===================
|
||||||
|
|
||||||
|
鉴于您的卓越贡献,我邀请您成为 LCTT 的核心(Core)成员。
|
||||||
|
|
||||||
|
作为核心成员,您拥有如下权利:
|
||||||
|
|
||||||
|
- 您能够合并您以及其它普通成员的 PR
|
||||||
|
- 您能够校对其它普通成员的文章,并提交校对 PR
|
||||||
|
- 您能够建议或反对将某位普通成员升级为核心成员
|
||||||
|
- 您能够自主添加符合 LCTT 定位的选题并翻译
|
||||||
|
- 您能够以 LCTT 核心成员名义宣称、署名
|
||||||
|
- 在得到 LCTT 同意后,以 LCTT 身份出席活动
|
||||||
|
- 在得到 LCTT 同意后,创建 LCTT 名下的其它仓库并管理(及邀请其它成员参与)
|
||||||
|
- 成为 LinuxCN 管理团队的一员
|
||||||
|
- 参与 LCTT 及 LinuxCN 决策投票
|
||||||
|
|
||||||
|
以下是不允许的行为,虽然您可能有这样的权限:
|
||||||
|
|
||||||
|
- 直接在 LCTT 主仓库上提交修改
|
||||||
|
- 在 LCTT 主仓库上创建分支
|
||||||
|
|
||||||
|
除此以外,您有以下义务:
|
||||||
|
|
||||||
|
- 为您的 GitHub 账户创建安全的密码
|
||||||
|
- 为您的 GitHub 账户创建 2FA 验证
|
||||||
|
- 不得外借您的 GitHub 账户,您需要为您的账户负责
|
||||||
|
- 保持参与度,如长期不能参与,会转换身份为荣誉成员,并放弃相应权限
|
||||||
|
- 接待新加入的成员,并做好引导个工作
|
||||||
|
- 维护 Wiki
|
||||||
|
|
||||||
|
作为核心成员,您会执行一些管理工作,以下是一些惯例:
|
||||||
|
|
||||||
|
- 合并 PR 时,要按时间顺序,从最早的(最下方)的开始合并
|
||||||
|
- 合并 PR 时,如果与仓库或前面的 PR 冲突,则关闭该 PR,并提醒 PR 发起人修改处理
|
||||||
|
- 除非必要,合并 PR 时不要 squash-merge
|
||||||
|
|
||||||
|
wxy@LCTT
|
||||||
|
2016/12/24
|
42
lctt2016.md
Normal file
42
lctt2016.md
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
LCTT 2016:LCTT 成立三周年了!
|
||||||
|
===========================
|
||||||
|
|
||||||
|
不知不觉,LCTT 已经成立三年了,对于我这样已经迈过四张的人来说,愈发的感觉时间过得真快。
|
||||||
|
|
||||||
|
这三年来,我们 LCTT 经历了很多事情,有些事情想起来仍恍如昨日。
|
||||||
|
|
||||||
|
三年前的这一天,我的一个偶发的想法促使我在 Linux 中国的 QQ 群里面发了一则消息,具体的消息内容已经不可考了,大意是鉴于英文 man 手册的中文翻译太差,想组织一些人来重新翻译。不料发出去之后很快赢得一些有热情、有理想的小伙伴们的响应。于是我匆匆建了一个群,拉了一些人进来,甚至当时连翻译组的名称都没有确定。LCTT (Linux.Cn Translation Team)这个名称都是后来逐步定下来的。
|
||||||
|
|
||||||
|
关于 LCTT 的早期发展情况,可以参考 LCTT 2014 年周年[总结](http://linux.cn/article-3784-1.html)。
|
||||||
|
|
||||||
|
虽然说“翻译 man 手册”这个最初建群的目标因为种种原因搁浅而一直未能重启,但是,这三年来,我们组织了 [213 位志愿者](https://github.com/LCTT/TranslateProject/graphs/contributors)翻译了 2155 篇文章,接受了 [4263 个 PR](https://github.com/LCTT/TranslateProject/pulls?q=is%3Apr+is%3Aclosed),得到了 476 个星。
|
||||||
|
|
||||||
|
这三年来,我们经历了 man 项目的流产、 LFS 手册的翻译发布、选题 DeadFire 的离去。得益于 Linux 中国的网站和微博,乃至微信的兴起后的传播,志愿者们的译文传播很广,切实的为国内的开源社区做出了贡献(当然,与此同时,Linux 中国社区也随之更加壮大)。
|
||||||
|
|
||||||
|
这些年间,LCTT 来了很多人,也有人慢慢淡出,这里面涌现了不少做出了卓越贡献的人,比如:
|
||||||
|
|
||||||
|
- geekpi,作为整个 LCTT 项目中翻译量最大贡献者,却鲜少在群内说话,偶尔露面,被戏称为“鸡排兄”。
|
||||||
|
- GOLinux,紧追“鸡排兄”的第二位强人,嗯,群内大部分人的昵称都是他起的,包括楼上。
|
||||||
|
- tinyeyeser,“小眼儿”以翻译风趣幽默著称,是 LCTT 早期初创成员之一。
|
||||||
|
- Vito-L,早期成员,LCTT 的多数 Wiki 出自于其手。
|
||||||
|
- DeadFire,创始成员,从最开始到其离世,一直负责 LCTT 的所有选题工作。
|
||||||
|
- oska874,在接过选题工作的重任后,全面主持 LCTT 的工作。
|
||||||
|
- carolinewuyan,承担了相当多的校对工作。
|
||||||
|
- alim0x,独立完成了 Android 编年史系列的翻译(多达 26 篇,现在还没发布完)等等。
|
||||||
|
|
||||||
|
其它还有 ictlyh、strugglingyouth、FSSlc、zpl1025、runningwater、bazz2、Vic020、dongfengweixiao、jasminepeng、willqian、vizv、ZTinoZ、martin2011qi、felixonmars、su-kaiyao、GHLandy、flsf、H-mudcup、StdioA、crowner、vim-kakali 等等,以及还有很多这里没有提到名字的人,都对 LCTT 做出不可磨灭的贡献。
|
||||||
|
|
||||||
|
具体的贡献排行榜,可以看[这里](https://github.com/LCTT/TranslateProject/graphs/contributors)。
|
||||||
|
|
||||||
|
每年写总结时,我都需要和 gource 以及 ffmpeg 搏斗半天,今年,我又用 gource 重新制作了一份 LCTT 的 GitHub 版本仓库的变迁视频,以飨众人。
|
||||||
|
|
||||||
|
本来想写很多,或许 LCTT 和 Linux 中国已经成了我的生活的一部分,竟然不知道该写点什么了,那就此搁笔罢。
|
||||||
|
|
||||||
|
另外,为 LCTT 的诸位兄弟姐妹们献上我及管理团队的祝福,也欢迎更多的志愿者加入 LCTT ,传送门在此:
|
||||||
|
|
||||||
|
- 项目网站:https://lctt.github.io/ ,请先访问此处了解情况。
|
||||||
|
- “Linux中国”开源社区:https://linux.cn/ ,所有翻译的文章都在这里以及它的同名微博、微信发布。
|
||||||
|
|
||||||
|
LCTT 组长 wxy
|
||||||
|
|
||||||
|
2016/9/10
|
@ -0,0 +1,59 @@
|
|||||||
|
浅谈 Linux 容器和镜像签名
|
||||||
|
====================
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/images/business/containers_2015-1-osdc-lead.png?itok=E1imOYe4)
|
||||||
|
|
||||||
|
从根本上说,几乎所有的主要软件,即使是开源软件,都是在基于镜像的容器技术出现之前设计的。这意味着把软件放到容器中相当于是一次平台移植。这也意味着一些程序可以很容易就迁移,[而一些就更困难][1]。
|
||||||
|
|
||||||
|
我大约在三年半前开展基于镜像的容器相关工作。到目前为止,我已经容器化了大量应用。我了解到什么是现实情况,什么是迷信。今天,我想简要介绍一下 Linux 容器是如何设计的,以及谈谈镜像签名。
|
||||||
|
|
||||||
|
### Linux 容器是如何设计的
|
||||||
|
|
||||||
|
对于基于镜像的 Linux 容器,让大多数人感到困惑的是,它把操作系统分割成两个部分:[内核空间与用户空间][2]。在传统操作系统中,内核运行在硬件上,你无法直接与其交互。用户空间才是你真正能交互的,这包括所有你可以通过文件浏览器或者运行`ls`命令能看到的文件、类库、程序。当你使用`ifconfig`命令调整 IP 地址时,你实际上正在借助用户空间的程序来使内核根据 TCP 协议栈改变。这点经常让没有研究过 [Linux/Unix 基础][3]的人大吃一惊。
|
||||||
|
|
||||||
|
过去,用户空间中的类库支持了与内核交互的程序(比如 ifconfig、sysctl、tuned-adm)以及如网络服务器和数据库之类的面向用户的程序。这些所有的东西都堆积在一个单一的文件系统结构中。用户可以在 /sbin 或者 /lib 文件夹中找到所有操作系统本身支持的程序和类库,或者可以在 /usr/sbin 或 /usr/lib 文件夹中找到所有面向用户的程序或类库(参阅[文件系统层次结构标准][4])。这个模型的问题在于操作系统程序和业务支持程序没有绝对的隔离。/usr/bin 中的程序可能依赖 /lib 中的类库。如果一个应用所有者需要改变一些东西,就很有可能破坏操作系统。相反地,如果负责安全更新的团队需要改变一个类库,就(常常)有可能破坏面向业务的应用。这真是一团糟。
|
||||||
|
|
||||||
|
借助基于镜像的容器,比如 Docker、LXD、RKT,应用程序所有者可以打包和调整所有放在 /sbin、/lib、/usr/bin 和 /usr/lib 中的依赖部分,而不用担心破坏底层操作系统。本质上讲,容器技术再次干净地将操作系统隔离为两部分:内核空间与用户空间。现在开发人员和运维人员可以分别独立地更新各自的东西。
|
||||||
|
|
||||||
|
然而还是有些令人困扰的地方。通常,每个应用所有者(或开发者)并不想负责更新这些应用依赖:像 openssl、glibc,或很底层的基础组件,比如,XML 解析器、JVM,再或者处理与性能相关的设置。过去,这些问题都委托给运维团队来处理。由于我们在容器中打包了很多依赖,对于很多组织来讲,对容器内的所有东西负责仍是个严峻的问题。
|
||||||
|
|
||||||
|
### 迁移现有应用到 Linux 容器
|
||||||
|
|
||||||
|
把应用放到容器中算得上是平台移植,我准备突出介绍究竟是什么让移植某些应用到容器当中这么困难。
|
||||||
|
|
||||||
|
(通过容器,)开发者现在对 /sbin 、/lib、 /usr/bin、 /usr/lib 中的内容有完全的控制权。但是,他们面临的挑战是,他们仍需要将数据和配置放到 /etc 或者 /var/lib 文件夹中。对于基于镜像的容器来说,这是一个糟糕的想法。我们真正需要的是代码、配置以及数据的隔离。我们希望开发者把代码放在容器当中,而数据和配置通过不同的环境(比如,开发、测试或生产环境)来获得。
|
||||||
|
|
||||||
|
这意味着我们(或者说平台)在实例化容器时,需要挂载 /etc 或 /var/lib 中的一些文件或文件夹。这会允许我们到处移动容器并仍能从环境中获得数据和配置。听起来很酷吧?这里有个问题,我们需要能够干净地隔离配置和数据。很多现代开源软件比如 Apache、MySQL、MongoDB、Nginx 默认就这么做了。[但很多自产的、历史遗留的、或专有程序并未默认这么设计][5]。对于很多组织来讲,这是主要的痛点。对于开发者来讲的最佳实践是,开始架构新的应用,移植遗留代码,以完成配置和数据的完全隔离。
|
||||||
|
|
||||||
|
### 镜像签名简介
|
||||||
|
|
||||||
|
信任机制是容器的重要议题。容器镜像签名允许用户添加数字指纹到镜像中。这个指纹随后可被加密算法测试验证。这使得容器镜像的用户可以验证其来源并信任。
|
||||||
|
|
||||||
|
容器社区经常使用“容器镜像”这个词组,但这个命名方法会让人相当困惑。Docker、LXD 和 RKT 推行获取远程文件来当作容器运行这样的概念。这些技术各自通过不同的方式处理容器镜像。LXD 用单独的一层来获取单独一个容器,而 Docker 和 RKT 使用基于开放容器镜像(OCI)格式,可由多层组成。糟糕的是,会出现不同团队和组织对容器镜像中的不同层负责的情况。容器镜像概念下隐含的是容器镜像格式的概念。拥有标准的镜像格式比如 OCI 会让容器生态系统围绕着镜像扫描、签名,和在不同云服务提供商间转移而繁荣发展。
|
||||||
|
|
||||||
|
现在谈到签名了。
|
||||||
|
|
||||||
|
容器存在一个问题,我们把一堆代码、二进制文件和类库放入其中。一旦我们打包了代码,我们就要把它和必要的文件服务器(注册服务器)共享。代码只要被共享,它基本上就是不具名的,缺少某种密文签名。更糟糕的是,容器镜像经常由不同人或团队控制的各个镜像层组成。每个团队都需要能够检查上一个团队的工作,增加他们自己的工作内容,并在上面添加他们自己的批准印记。然后他们需要继续把工作交给下个团队。
|
||||||
|
|
||||||
|
(由很多镜像组成的)容器镜像的最终用户需要检查监管链。他们需要验证每个往其中添加文件的团队的可信度。对于最终用户而言,对容器镜像中的每一层都有信心是极其重要的。
|
||||||
|
|
||||||
|
*作者 Scott McCarty 于 8 月 24 日在 ContainerCon 会议上作了题为 [Containers for Grownups: Migrating Traditional & Existing Applications][6] 的报告,更多内容请参阅报告[幻灯片][7]。*
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://opensource.com/bus/16/8/introduction-linux-containers-and-image-signing
|
||||||
|
|
||||||
|
作者:[Scott McCarty][a]
|
||||||
|
译者:[Tanete](https://github.com/Tanete)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://opensource.com/users/fatherlinux
|
||||||
|
[1]: http://rhelblog.redhat.com/2016/04/21/architecting-containers-part-4-workload-characteristics-and-candidates-for-containerization/
|
||||||
|
[2]: http://rhelblog.redhat.com/2015/07/29/architecting-containers-part-1-user-space-vs-kernel-space/
|
||||||
|
[3]: http://rhelblog.redhat.com/tag/architecting-containers/
|
||||||
|
[4]: https://linux.cn/article-6132-1.html
|
||||||
|
[5]: http://rhelblog.redhat.com/2016/04/21/architecting-containers-part-4-workload-characteristics-and-candidates-for-containerization/
|
||||||
|
[6]: https://lcccna2016.sched.org/event/7JUc/containers-for-grownups-migrating-traditional-existing-applications-scott-mccarty-red-hat
|
||||||
|
[7]: http://schd.ws/hosted_files/lcccna2016/91/Containers%20for%20Grownups_%20Migrating%20Traditional%20%26%20Existing%20Applications.pdf
|
@ -0,0 +1,276 @@
|
|||||||
|
忘记技术债务 —— 教你如何创造技术财富
|
||||||
|
===============
|
||||||
|
|
||||||
|
电视里正播放着《老屋》节目,[Andrea Goulet][58] 和她的商业合作伙伴正悠闲地坐在客厅里,商讨着他们的战略计划。那正是大家思想的火花碰撞出创新事物的时刻。他们正在寻求一种能够实现自身价值的方式 —— 为其它公司清理<ruby>遗留代码<rt>legacy code</rt></ruby>及科技债务。他们此刻的情景,像极了电视里的场景。(LCTT 译注:《老屋》电视节目提供专业的家装、家庭改建、重新装饰、创意等等信息,与软件的改造有异曲同工之处)。
|
||||||
|
|
||||||
|
“我们意识到我们现在做的工作不仅仅是清理遗留代码,实际上我们是在用重建老屋的方式来重构软件,让系统运行更持久、更稳定、更高效,”Goulet 说。“这让我开始思考公司如何花钱来改善他们的代码,以便让他们的系统运行更高效。就好比为了让屋子变得更有价值,你不得不使用一个全新的屋顶。这并不吸引人,但却是至关重要的,然而很多人都搞错了。“
|
||||||
|
|
||||||
|
如今,她是 [Corgibytes][57] 公司的 CEO —— 这是一家提高软件现代化和进行系统重构方面的咨询公司。她曾经见过各种各样糟糕的系统、遗留代码,以及严重的科技债务事件。Goulet 认为**创业公司需要转变思维模式,不是偿还债务,而是创造科技财富,不是要铲除旧代码,而是要逐步修复代码**。她解释了这种新的方法,以及如何完成这些看似不可能完成的事情 —— 实际上是聘用优秀的工程师来完成这些工作。
|
||||||
|
|
||||||
|
### 反思遗留代码
|
||||||
|
|
||||||
|
关于遗留代码最常见的定义是由 Michael Feathers 在他的著作[<ruby>《高效利用遗留代码》<rt>Working Effectively with Legacy Code</rt></ruby>][56]一书中提出:遗留代码就是没有被测试所覆盖的代码。这个定义比大多数人所认为的 —— 遗留代码仅指那些古老、陈旧的系统这个说法要妥当得多。但是 Goulet 认为这两种定义都不够明确。“遗留代码与软件的年头儿毫无关系。一个两年的应用程序,其代码可能已经进入遗留状态了,”她说。“**关键要看软件质量提高的难易程度。**”
|
||||||
|
|
||||||
|
这意味着写得不够清楚、缺少解释说明的代码,是没有包含任何关于代码构思和决策制定的流程的成果。单元测试就是这样的一种成果,但它并没有包括了写那部分代码的原因以及逻辑推理相关的所有文档。如果想要提升代码,但没办法搞清楚原开发者的意图 —— 那些代码就属于遗留代码了。
|
||||||
|
|
||||||
|
> **遗留代码不是技术问题,而是沟通上的问题。**
|
||||||
|
|
||||||
|
![](https://s3.amazonaws.com/marquee-test-akiaisur2rgicbmpehea/H4y9x4gQj61G9aK4v8Kp_Screen%20Shot%202016-08-11%20at%209.16.38%20AM.png)
|
||||||
|
|
||||||
|
如果你像 Goulet 所说的那样迷失在遗留代码里,你会发现每一次的沟通交流过程都会变得像那条[<ruby>康威定律<rt>Conway’s Law</rt></ruby>][54]所描述的一样。
|
||||||
|
|
||||||
|
Goulet 说:“这个定律认为你的代码能反映出整个公司的组织沟通结构,如果想修复公司的遗留代码,而没有一个好的组织沟通方式是不可能完成的。那是很多人都没注意到的一个重要环节。”
|
||||||
|
|
||||||
|
Goulet 和她的团队成员更像是考古学家一样来研究遗留系统项目。他们根据前开发者写的代码构件相关的线索来推断出他们的思想意图。然后再根据这些构件之间的关系来做出新的决策。
|
||||||
|
|
||||||
|
代码构件最重要的什么呢?**良好的代码结构、清晰的思想意图、整洁的代码**。例如,如果使用通用的名称如 “foo” 或 “bar” 来命名一个变量,半年后再返回来看这段代码时,根本就看不出这个变量的用途是什么。
|
||||||
|
|
||||||
|
如果代码读起来很困难,可以使用源代码控制系统,这是一个非常有用的工具,因为它可以提供代码的历史修改信息,并允许软件开发者写明他们作出本次修改的原因。
|
||||||
|
|
||||||
|
Goulet 说:“我一个朋友认为提交代码时附带的信息,每一个概要部分的内容应该有半条推文那么长(几十个字),如需要的话,代码的描述信息应该有一篇博客那么长。你得用这个方式来为你修改的代码写一个合理的说明。这不会浪费太多额外的时间,并且能给后期的项目开发者提供非常多的有用信息,但是让人惊讶的是很少有人会这么做。我们经常能看到一些开发人员在被一段代码激怒之后,要用 `git blame` 扒代码库找出这些垃圾是谁干的,结果最后发现是他们自己干的。”
|
||||||
|
|
||||||
|
使用自动化测试对于理解程序的流程非常有用。Goulet 解释道:“很多人都比较认可 Michael Feathers 提出的关于遗留代码的定义。测试套件对于理解开发者的意图来说是非常有用的工具,尤其当用来与[<ruby>行为驱动开发模式<rt>Behavior Driven Development</rt></ruby>][53]相结合时,比如编写测试场景。”
|
||||||
|
|
||||||
|
理由很简单,如果你想将遗留代码限制在一定程度下,注意到这些细节将使代码更易于理解,便于在以后也能工作。编写并运行一个代码单元,接受、认可,并且集成测试。写清楚注释的内容,方便以后你自己或是别人来理解你写的代码。
|
||||||
|
|
||||||
|
尽管如此,由于很多已知的和不可意料的原因,遗留代码仍然会出现。
|
||||||
|
|
||||||
|
在创业公司刚成立初期,公司经常会急于推出很多新的功能。开发人员在巨大的交付压力下,测试常常半途而废。Corgibytes 团队就遇到过好多公司很多年都懒得对系统做详细的测试了。
|
||||||
|
|
||||||
|
确实如此,当你急于开发出系统原型的时候,强制性地去做太多的测试也许意义不大。但是,一旦产品开发完成并投入使用后,你就需要投入时间精力来维护及完善系统了。Goulet 说:“很多人说,‘别在维护上费心思,重要的是功能!’ **如果真这样,当系统规模到一定程序的时候,就很难再扩展了。同时也就失去市场竞争力了。**”
|
||||||
|
|
||||||
|
最后才明白过来,原来热力学第二定律对代码也同样适用:**你所面临的一切将向熵增的方向发展。**你需要与混乱无序的技术债务进行一场无休无止的战斗。随着时间的推移,遗留代码也逐渐变成一种债务。
|
||||||
|
|
||||||
|
她说:“我们再次拿家来做比喻。你必须坚持每天收拾餐具、打扫卫生、倒垃圾。如果你不这么做,情况将来越来越糟糕,直到有一天你不得不向 HazMat 团队求助。”(LCTT 译注:HazMat 团队,危害物质专队)
|
||||||
|
|
||||||
|
就跟这种情况一样,Corgibytes 团队接到很多公司 CEO 的求助电话,比如 Features 公司的 CEO 在电话里抱怨道:“现在我们公司的开发团队工作效率太低了,三年前只需要两个星期就完成的工作,现在却要花费12个星期。”
|
||||||
|
|
||||||
|
> **技术债务往往反映出公司运作上的问题。**
|
||||||
|
|
||||||
|
很多公司的 CTO 明知会发生技术债务的问题,但是他们很难说服其它同事相信花钱来修复那些已经存在的问题是值得的。这看起来像是在走回头路,很乏味,也不是新的产品。有些公司直到系统已经严重影响了日常工作效率时,才着手去处理这些技术债务方面的问题,那时付出的代价就太高了。
|
||||||
|
|
||||||
|
### 忘记债务,创造技术财富
|
||||||
|
|
||||||
|
如果你想把[<ruby>重构技术债务<rt>reframe your technical debt</rt></ruby>][52] — [敏捷开发讲师 Declan Whelan 最近造出的一个术语][51] — 作为一个积累技术财富的机会,你很可能要先说服你们公司的 CEO、投资者和其它的股东接受并为之共同努力。
|
||||||
|
|
||||||
|
“我们没必要把技术债务想像得很可怕。当产品处于开发设计初期,技术债务反而变得非常有用,”Goulet 说。“当你解决一些系统遗留的技术问题时,你会充满成就感。例如,当你在自己家里安装新窗户时,你确实会花费一笔不少的钱,但是之后你每个月就可以节省 100 美元的电费。程序代码亦是如此。虽然暂时没有提高工作效率,但随时时间推移将提高生产力。”
|
||||||
|
|
||||||
|
一旦你意识到项目团队工作不再富有成效时,就需要确认下是哪些技术债务在拖后腿了。
|
||||||
|
|
||||||
|
“我跟很多不惜一切代价招募英才的初创公司交流过,他们高薪聘请一些工程师来只为了完成更多的工作。”她说。“与此相反,他们应该找出如何让原有的每个工程师能更高效率工作的方法。你需要去解决什么样的技术债务以增加额外的生产率?”
|
||||||
|
|
||||||
|
如果你改变自己的观点并且专注于创造技术财富,你将会看到产能过剩的现象,然后重新把多余的产能投入到修复更多的技术债务和遗留代码的良性循环中。你们的产品将会走得更远,发展得更好。
|
||||||
|
|
||||||
|
> **别把你们公司的软件当作一个项目来看。从现在起,把它想象成一栋自己要长久居住的房子。**
|
||||||
|
|
||||||
|
“这是一个极其重要的思想观念的转变,”Goulet 说。“这将带你走出短浅的思维模式,并让你比之前更加关注产品的维护工作。”
|
||||||
|
|
||||||
|
这就像对一栋房子,要实现其现代化及维护的方式有两种:小动作,表面上的更改(“我买了一块新的小地毯!”)和大改造,需要很多年才能偿还所有债务(“我想我们应替换掉所有的管道...”)。你必须考虑好两者,才能让你们已有的产品和整个团队顺利地运作起来。
|
||||||
|
|
||||||
|
这还需要提前预算好 —— 否则那些较大的花销将会是硬伤。定期维护是最基本的预期费用。让人震惊的是,很多公司都没把维护当成商务成本预算进来。
|
||||||
|
|
||||||
|
这就是 Goulet 提出“**<ruby>软件重构<rt>software remodeling</rt></ruby>**”这个术语的原因。当你房子里的一些东西损坏的时候,你并不是铲除整个房子,从头开始重建。同样的,当你们公司出现老的、损坏的代码时,重写代码通常不是最明智的选择。
|
||||||
|
|
||||||
|
下面是 Corgibytes 公司在重构客户代码用到的一些方法:
|
||||||
|
|
||||||
|
* 把大型的应用系统分解成轻量级的更易于维护的微服务。
|
||||||
|
* 让功能模块彼此解耦以便于扩展。
|
||||||
|
* 更新形象和提升用户前端界面体验。
|
||||||
|
* 集合自动化测试来检查代码可用性。
|
||||||
|
* 代码库可以让重构或者修改更易于操作。
|
||||||
|
|
||||||
|
系统重构也进入到 DevOps 领域。比如,Corgibytes 公司经常推荐新客户使用 [Docker][50],以便简单快速的部署新的开发环境。当你们团队有 30 个工程师的时候,把初始化配置时间从 10 小时减少到 10 分钟对完成更多的工作很有帮助。系统重构不仅仅是应用于软件开发本身,也包括如何进行系统重构。
|
||||||
|
|
||||||
|
如果你知道做些什么能让你们的代码管理起来更容易更高效,就应该把这它们写入到每年或季度的项目规划中。别指望它们会自动呈现出来。但是也别给自己太大的压力来马上实施它们。Goulets 看到很多公司从一开始就致力于 100% 测试覆盖率而陷入困境。
|
||||||
|
|
||||||
|
**具体来说,每个公司都应该把以下三种类型的重构工作规划到项目建设中来:**
|
||||||
|
|
||||||
|
* 自动测试
|
||||||
|
* 持续交付
|
||||||
|
* 文化提升
|
||||||
|
|
||||||
|
咱们来深入的了解下每一项内容。
|
||||||
|
|
||||||
|
#### 自动测试
|
||||||
|
|
||||||
|
“有一位客户即将进行第二轮融资,但是他们没办法在短期内招聘到足够的人才。我们帮助他们引进了一种自动化测试框架,这让他们的团队在 3 个月的时间内工作效率翻了一倍,”Goulets 说。“这样他们就可以在他们的投资人面前自豪的说,‘我们一个精英团队完成的任务比两个普通的团队要多。’”
|
||||||
|
|
||||||
|
自动化测试从根本上来讲就是单个测试的组合,就是可以再次检查某一行代码的单元测试。可以使用集成测试来确保系统的不同部分都正常运行。还可以使用验收性测试来检验系统的功能特性是否跟你想像的一样。当你把这些测试写成测试脚本后,你只需要简单地用鼠标点一下按钮就可以让系统自行检验了,而不用手工的去梳理并检查每一项功能。
|
||||||
|
|
||||||
|
在产品市场尚未打开之前就来制定自动化测试机制有些言之过早。但是一旦你有一款感到满意,并且客户也很依赖的产品,就应该把这件事付诸实施了。
|
||||||
|
|
||||||
|
#### 持续交付
|
||||||
|
|
||||||
|
这是与自动化交付相关的工作,过去是需要人工完成。目的是当系统部分修改完成时可以迅速进行部署,并且短期内得到反馈。这使公司在其它竞争对手面前有很大的优势,尤其是在客户服务行业。
|
||||||
|
|
||||||
|
“比如说你每次部署系统时环境都很复杂。熵值无法有效控制,”Goulets 说。“我们曾经见过花 12 个小时甚至更多的时间来部署一个很大的集群环境。在这种情况下,你不会愿意频繁部署了。因为太折腾人了,你还会推迟系统功能上线的时间。这样,你将落后于其它公司并失去竞争力。”
|
||||||
|
|
||||||
|
**在持续性改进的过程中常见的其它自动化任务包括:**
|
||||||
|
|
||||||
|
* 在提交完成之后检查构建中断部分。
|
||||||
|
* 在出现故障时进行回滚操作。
|
||||||
|
* 自动化审查代码的质量。
|
||||||
|
* 根据需求增加或减少服务器硬件资源。
|
||||||
|
* 让开发、测试及生产环境配置简单易懂。
|
||||||
|
|
||||||
|
举一个简单的例子,比如说一个客户提交了一个系统 Bug 报告。开发团队越高效解决并修复那个 Bug 越好。对于开发人员来说,修复 Bug 的挑战根本不是个事儿,这本来也是他们的强项,主要是系统设置上不够完善导致他们浪费太多的时间去处理 bug 以外的其它问题。
|
||||||
|
|
||||||
|
使用持续改进的方式时,你要严肃地决定决定哪些工作应该让机器去做,哪些交给研发去完成更好。如果机器更擅长,那就使其自动化完成。这样也能让研发愉快地去解决其它有挑战性的问题。同时客户也会很高兴地看到他们报怨的问题被快速处理了。你的待修复的未完成任务数减少了,之后你就可以把更多的时间投入到运用新的方法来提高产品的质量上了。**这是创造科技财富的一种转变。**因为开发人员可以修复 bug 后立即发布新代码,这样他们就有时间和精力做更多事。
|
||||||
|
|
||||||
|
“你必须时刻问自己,‘我应该如何为我们的客户改善产品功能?如何做得更好?如何让产品运行更高效?’不过还要不止于此。”Goulets 说。“一旦你回答完这些问题后,你就得询问下自己,如何自动去完成那些需要改善的功能。”
|
||||||
|
|
||||||
|
#### 文化提升
|
||||||
|
|
||||||
|
Corgibytes 公司每天都会看到同样的问题:一家创业公司建立了一个对开发团队毫无推动的文化环境。公司 CEO 抱着双臂思考着为什么这样的环境对员工没多少改变。然而事实却是公司的企业文化对工作并不利。为了激励工程师,你必须全面地了解他们的工作环境。
|
||||||
|
|
||||||
|
为了证明这一点,Goulet 引用了作者 Robert Henry 说过的一段话:
|
||||||
|
|
||||||
|
> **目的不是创造艺术,而是在最美妙的状态下让艺术应运而生。**
|
||||||
|
|
||||||
|
“你们也要开始这样思考一下你们的软件,”她说。“你们的企业文件就类似那个状态。你们的目标就是创造一个让艺术品应运而生的环境,这件艺术品就是你们公司的代码、一流的售后服务、充满幸福感的开发者、良好的市场预期、盈利能力等等。这些都息息相关。”
|
||||||
|
|
||||||
|
优先考虑解决公司的技术债务和遗留代码也是一种文化。那是真正为开发团队清除障碍,以制造影响的方法。同时,这也会让你将来有更多的时间精力去完成更重要的工作。如果你不从根本上改变固有的企业文化环境,你就不可能重构公司产品。改变对产品维护及现代化的投资的态度是开始实施变革的第一步,最理想情况是从公司的 CEO 开始自顶向下转变。
|
||||||
|
|
||||||
|
以下是 Goulet 关于建立那种流态文化方面提出的建议:
|
||||||
|
|
||||||
|
* 反对公司嘉奖那些加班到深夜的“英雄”。提倡高效率的工作方式。
|
||||||
|
* 了解协同开发技术,比如 Woody Zuill 提出的[<ruby>合作编程<rt>Mob Programming</rt></ruby>][44]模式。
|
||||||
|
* 遵从 4 个[现代敏捷开发][42]原则:用户至上、实践及快速学习、把安全放在首位、持续交付价值。
|
||||||
|
* 每周为研发人员提供项目外的职业发展时间。
|
||||||
|
* 把[日工作记录][43]作为一种驱动开发团队主动解决问题的方式。
|
||||||
|
* 把同情心放在第一位。Corgibytes 公司让员工参加 [Brene Brown 勇气工厂][40]的培训是非常有用的。
|
||||||
|
|
||||||
|
“如果公司高管和投资者不支持这种升级方式,你得从客户服务的角度去说服他们,”Goulet 说,“告诉他们通过这次调整后,最终产品将如何给公司的大多数客户提高更好的体验。这是你能做的一个很有力的论点。”
|
||||||
|
|
||||||
|
### 寻找最具天才的代码重构者
|
||||||
|
|
||||||
|
整个行业都认为顶尖的工程师不愿意干修复遗留代码的工作。他们只想着去开发新的东西。大家都说把他们留在维护部门真是太浪费人才了。
|
||||||
|
|
||||||
|
**其实这些都是误解。如果你知道去哪里和如何找工程师,并为他们提供一个愉快的工作环境,你就可以找到技术非常精湛的工程师,来帮你解决那些最棘手的技术债务问题。**
|
||||||
|
|
||||||
|
“每次在会议上,我们都会问现场的同事‘谁喜欢去在遗留代码上工作?’每次只有不到 10% 的与会者会举手。”Goulet 说。“但是我跟这些人交流后,我发现这些工程师恰好是喜欢最具挑战性工作的人才。”
|
||||||
|
|
||||||
|
有一位客户来寻求她的帮助,他们使用国产的数据库,没有任何相关文档,也没有一种有效的方法来弄清楚他们公司的产品架构。她称修理这种情况的一类工程师为“修正者”。在 Corgibytes 公司,她有一支这样的修正者团队由她支配,热衷于通过研究二进制代码来解决技术问题。
|
||||||
|
|
||||||
|
![](https://s3.amazonaws.com/marquee-test-akiaisur2rgicbmpehea/BeX5wWrESmCTaJYsuKhW_Screen%20Shot%202016-08-11%20at%209.17.04%20AM.png)
|
||||||
|
|
||||||
|
那么,如何才能找到这些技术人才呢? Goulet 尝试过各种各样的方法,其中有一些方法还是富有成效的。
|
||||||
|
|
||||||
|
她创办了一个社区网站 [legacycode.rocks][49] 并且在招聘启示上写道:“长期招聘那些喜欢重构遗留代码的另类开发人员...如果你以从事处理遗留代码的工作为自豪,欢迎加入!”
|
||||||
|
|
||||||
|
“我开始收到很多人发来邮件说,‘噢,天呐,我也属于这样的开发人员!’”她说。“只需要发布这条信息,并且告诉他们这份工作是非常有意义的,就吸引了合适的人才。”
|
||||||
|
|
||||||
|
在招聘的过程中,她也会使用持续性交付的经验来回答那些另类开发者想知道的信息:包括详细的工作内容以及明确的要求。“我这么做的原因是因为我讨厌重复性工作。如果我收到多封邮件来咨询同一个问题,我会把答案发布在网上,我感觉自己更像是在写说明文档一样。”
|
||||||
|
|
||||||
|
但是随着时间的推移,她发现可以重新定义招聘流程来帮助她识别出更出色的候选人。比如说,她在应聘要求中写道,“公司 CEO 将会重新审查你的简历,因此请确保求职信中致意时不用写明性别。所有以‘尊敬的先生’或‘先生’开头的信件将会被当垃圾处理掉”。这些只是她的招聘初期策略。
|
||||||
|
|
||||||
|
“我开始这么做是因为很多申请人把我当成男性,因为我是一家软件公司的男性 CEO,我必须是男性!?”Goulet 说。“所以,有一天我想我应该它当作应聘要求放到网上,看有多少人注意到这个问题。令我惊讶的是,这让我过滤掉一些不太严谨的申请人。还突显出了很多擅于从事遗留代码方面工作的人。”
|
||||||
|
|
||||||
|
Goulet 想起一个应聘者发邮件给我说,“我查看了你们网站的代码(我喜欢这个网站,这也是我的工作)。你们的网站架构很奇特,好像是用 PHP 写的,但是你们却运行在用 Ruby 语言写的 Jekyll 下。我真的很好奇那是什么呢。”
|
||||||
|
|
||||||
|
Goulet 从她的设计师那里得知,原来,在 HTML、CSS 和 JavaScript 文件中有一个未使用的 PHP 类名,她一直想解决这个问题,但是一直没机会。Goulet 的回复是:“你正在找工作吗?”
|
||||||
|
|
||||||
|
另外一名候选人注意到她曾经在一篇说明文档中使用 CTO 这个词,但是她的团队里并没有这个头衔(她的合作伙伴是 Chief Code Whisperer)。这些注重细节、充满求知欲、积极主动的候选者更能引起她的注意。
|
||||||
|
|
||||||
|
> **代码修正者不仅需要注重细节,而且这也是他们必备的品质。**
|
||||||
|
|
||||||
|
让人吃惊的是,Goulet 从来没有为招募最优秀的代码修正者而感到厌烦过。“大多数人都是通过我们的网站直接投递简历,但是当我们想扩大招聘范围的时候,我们会通过 [PowerToFly][48] 和 [WeWorkRemotely][47] 网站进行招聘。我现在确实不需要招募新人马了。他们需要经历一段很艰难的时期才能理解代码修正者的意义是什么。”
|
||||||
|
|
||||||
|
如果他们通过首轮面试,Goulet 将会让候选者阅读一篇 Arlo Belshee 写的文章“[<ruby>命名是一个过程<rt>Naming is a Process</rt></ruby>][46]”。它讲的是非常详细的处理遗留代码的的过程。她最经典的指导方法是:“阅读完这段代码并且告诉我,你是怎么理解的。”
|
||||||
|
|
||||||
|
她将找出对问题的理解很深刻并且也愿意接受文章里提出的观点的候选者。这对于区分有深刻理解的候选者和仅仅想获得工作的候选者来说,是极其有用的办法。她强烈要求候选者找出一段与他操作相关的代码,来证明他是充满激情的、有主见的及善于分析问题的人。
|
||||||
|
|
||||||
|
最后,她会让候选者跟公司里当前的团队成员一起使用 [Exercism.io][45] 工具进行编程。这是一个开源项目,它允许开发者学习如何在不同的编程语言环境下使用一系列的测试驱动开发的练习进行编程。结对编程课程的第一部分允许候选者选择其中一种语言来使用。下一个练习中,面试官可以选择一种语言进行编程。他们总能看到那些人处理异常的方法、随机应便的能力以及是否愿意承认某些自己不了解的技术。
|
||||||
|
|
||||||
|
“当一个人真正的从执业者转变为大师的时候,他会毫不犹豫的承认自己不知道的东西,”Goulet说。
|
||||||
|
|
||||||
|
让他们使用自己不熟悉的编程语言来写代码,也能衡量其坚韧不拔的毅力。“我们想听到某个人说,‘我会深入研究这个问题直到彻底解决它。’也许第二天他们仍然会跑过来跟我们说,‘我会一直留着这个问题直到我找到答案为止。’那是作为一个成功的修正者表现出来的一种气质。”
|
||||||
|
|
||||||
|
> **产品开发人员在我们这个行业很受追捧,因此很多公司也想让他们来做维护工作。这是一个误解。最优秀的维护修正者并不是最好的产品开发工程师。**
|
||||||
|
|
||||||
|
如果一个有天赋的修正者在眼前,Goulet 懂得如何让他走向成功。下面是如何让这种类型的开发者感到幸福及高效工作的一些方式:
|
||||||
|
|
||||||
|
* 给他们高度的自主权。把问题解释清楚,然后安排他们去完成,但是永不命令他们应该如何去解决问题。
|
||||||
|
* 如果他们要求升级他们的电脑配置和相关工具,尽管去满足他们。他们明白什么样的需求才能最大限度地提高工作效率。
|
||||||
|
* 帮助他们[避免分心][39]。他们喜欢全身心投入到某一个任务直至完成。
|
||||||
|
|
||||||
|
总之,这些方法已经帮助 Corgibytes 公司培养出二十几位对遗留代码充满激情的专业开发者。
|
||||||
|
|
||||||
|
### 稳定期没什么不好
|
||||||
|
|
||||||
|
大多数创业公司都都不想跳过他们的成长期。一些公司甚至认为成长期应该是永无止境的。而且,他们觉得也没这个必要跳过成长期,即便他们已经进入到了下一个阶段:稳定期。**完全进入到稳定期意味着你拥有人力资源及管理方法来创造技术财富,同时根据优先权适当支出。**
|
||||||
|
|
||||||
|
“在成长期和稳定期之间有个转折点,就是维护人员必须要足够壮大,并且相对于专注新功能的产品开发人员,你开始更公平的对待维护人员,”Goulet 说。“你们公司的产品开发完成了。现在你得让他们更加稳定地运行。”
|
||||||
|
|
||||||
|
这就意味着要把公司更多的预算分配到产品维护及现代化方面。“你不应该把产品维护当作是一个不值得关注的项目,”她说。“这必须成为你们公司固有的一种企业文化 —— 这将帮助你们公司将来取得更大的成功。“
|
||||||
|
|
||||||
|
最终,你通过这些努力创建的技术财富,将会为你的团队带来一大批全新的开发者:他们就像侦查兵一样,有充足的时间和资源去探索新的领域,挖掘新客户资源并且给公司创造更多的机遇。当你们在新的市场领域做得更广泛并且不断取得进展 —— 那么你们公司已经真正地进入到繁荣发展的状态了。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://firstround.com/review/forget-technical-debt-heres-how-to-build-technical-wealth/
|
||||||
|
|
||||||
|
作者:[http://firstround.com/][a]
|
||||||
|
译者:[rusking](https://github.com/rusking)
|
||||||
|
校对:[jasminepeng](https://github.com/jasminepeng)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://firstround.com/
|
||||||
|
[1]:http://corgibytes.com/blog/2016/04/15/inception-layers/
|
||||||
|
[2]:http://www.courageworks.com/
|
||||||
|
[3]:http://corgibytes.com/blog/2016/08/02/how-we-use-daily-journals/
|
||||||
|
[4]:https://www.industriallogic.com/blog/modern-agile/
|
||||||
|
[5]:http://mobprogramming.org/
|
||||||
|
[6]:http://exercism.io/
|
||||||
|
[7]:http://arlobelshee.com/good-naming-is-a-process-not-a-single-step/
|
||||||
|
[8]:https://weworkremotely.com/
|
||||||
|
[9]:https://www.powertofly.com/
|
||||||
|
[10]:http://legacycode.rocks/
|
||||||
|
[11]:https://www.docker.com/
|
||||||
|
[12]:http://legacycoderocks.libsyn.com/technical-wealth-with-declan-wheelan
|
||||||
|
[13]:https://www.agilealliance.org/resources/initiatives/technical-debt/
|
||||||
|
[14]:https://en.wikipedia.org/wiki/Behavior-driven_development
|
||||||
|
[15]:https://en.wikipedia.org/wiki/Conway%27s_law
|
||||||
|
[16]:https://www.amazon.com/Working-Effectively-Legacy-Michael-Feathers/dp/0131177052
|
||||||
|
[17]:http://corgibytes.com/
|
||||||
|
[18]:https://www.linkedin.com/in/andreamgoulet
|
||||||
|
[19]:http://corgibytes.com/blog/2016/04/15/inception-layers/
|
||||||
|
[20]:http://www.courageworks.com/
|
||||||
|
[21]:http://corgibytes.com/blog/2016/08/02/how-we-use-daily-journals/
|
||||||
|
[22]:https://www.industriallogic.com/blog/modern-agile/
|
||||||
|
[23]:http://mobprogramming.org/
|
||||||
|
[24]:http://mobprogramming.org/
|
||||||
|
[25]:http://exercism.io/
|
||||||
|
[26]:http://arlobelshee.com/good-naming-is-a-process-not-a-single-step/
|
||||||
|
[27]:https://weworkremotely.com/
|
||||||
|
[28]:https://www.powertofly.com/
|
||||||
|
[29]:http://legacycode.rocks/
|
||||||
|
[30]:https://www.docker.com/
|
||||||
|
[31]:http://legacycoderocks.libsyn.com/technical-wealth-with-declan-wheelan
|
||||||
|
[32]:https://www.agilealliance.org/resources/initiatives/technical-debt/
|
||||||
|
[33]:https://en.wikipedia.org/wiki/Behavior-driven_development
|
||||||
|
[34]:https://en.wikipedia.org/wiki/Conway%27s_law
|
||||||
|
[35]:https://www.amazon.com/Working-Effectively-Legacy-Michael-Feathers/dp/0131177052
|
||||||
|
[36]:https://www.amazon.com/Working-Effectively-Legacy-Michael-Feathers/dp/0131177052
|
||||||
|
[37]:http://corgibytes.com/
|
||||||
|
[38]:https://www.linkedin.com/in/andreamgoulet
|
||||||
|
[39]:http://corgibytes.com/blog/2016/04/15/inception-layers/
|
||||||
|
[40]:http://www.courageworks.com/
|
||||||
|
[41]:http://corgibytes.com/blog/2016/08/02/how-we-use-daily-journals/
|
||||||
|
[42]:https://www.industriallogic.com/blog/modern-agile/
|
||||||
|
[43]:http://corgibytes.com/blog/2016/08/02/how-we-use-daily-journals/
|
||||||
|
[44]:http://mobprogramming.org/
|
||||||
|
[45]:http://exercism.io/
|
||||||
|
[46]:http://arlobelshee.com/good-naming-is-a-process-not-a-single-step/
|
||||||
|
[47]:https://weworkremotely.com/
|
||||||
|
[48]:https://www.powertofly.com/
|
||||||
|
[49]:http://legacycode.rocks/
|
||||||
|
[50]:https://www.docker.com/
|
||||||
|
[51]:http://legacycoderocks.libsyn.com/technical-wealth-with-declan-wheelan
|
||||||
|
[52]:https://www.agilealliance.org/resources/initiatives/technical-debt/
|
||||||
|
[53]:https://en.wikipedia.org/wiki/Behavior-driven_development
|
||||||
|
[54]:https://en.wikipedia.org/wiki/Conway%27s_law
|
||||||
|
[56]:https://www.amazon.com/Working-Effectively-Legacy-Michael-Feathers/dp/0131177052
|
||||||
|
[57]:http://corgibytes.com/
|
||||||
|
[58]:https://www.linkedin.com/in/andreamgoulet
|
@ -1,21 +1,15 @@
|
|||||||
|
如何在 CentOS 7 用 cPanel 配置 Nginx 反向代理
|
||||||
|
|
||||||
【flankershen翻译中】
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
How to Set Nginx as Reverse Proxy on Centos7 CPanel
|
|
||||||
================================================================================
|
================================================================================
|
||||||
|
|
||||||
Nginx is one of the fastest and most powerful web-server. It is known for its high performance and low resource utilization. It can be installed as both a standalone and a Reverse Proxy Web-server. In this article, I'm discussing about the installation of Nginx as a reverse proxy along with Apache on a CPanel server with latest CentOS 7 installed.
|
Nginx 是最快和最强大的 Web 服务器之一,以其高性能和低资源占用率而闻名。它既可以被安装为一个独立的 Web 服务器,也可以安装成反向代理 Web 服务器。在这篇文章,我将讨论在安装了 cPanel 管理系统的 Centos 7 服务器上安装 Nginx 作为 Apache 的反向代理服务器。
|
||||||
|
|
||||||
Nginx as a reverse proxy will work as a frontend webserver serving static contents along with Apache serving the dynamic files in backend. This setup will boost up the overall server performance.
|
Nginx 作为前端服务器用反向代理为静态文件提供服务,Apache 作为后端为动态文件提供服务。这个设置将整体提高服务器的性能。
|
||||||
|
|
||||||
Let's walk through the installation steps for Nginx as reverse proxy in CentOS7 x86_64 bit server with cPanel 11.52 installed.
|
让我们过一遍在已经安装好 cPanel 11.52 的 CentOS 7 x86_64 服务器上配置 Nginx 作为反向代理的安装过程。
|
||||||
|
|
||||||
First of all, we need to install the EPEL repo to start-up with the process.
|
首先,我们需要安装 EPEL 库来启动这个进程
|
||||||
|
|
||||||
### Step 1: Install the EPEL repo. ###
|
### 第一步: 安装 EPEL 库###
|
||||||
|
|
||||||
root@server1 [/usr]# yum -y install epel-release
|
root@server1 [/usr]# yum -y install epel-release
|
||||||
Loaded plugins: fastestmirror, tsflags, universal-hooks
|
Loaded plugins: fastestmirror, tsflags, universal-hooks
|
||||||
@ -31,13 +25,13 @@ First of all, we need to install the EPEL repo to start-up with the process.
|
|||||||
|
|
||||||
Dependencies Resolved
|
Dependencies Resolved
|
||||||
|
|
||||||
===============================================================================================================================================
|
========================================================================================
|
||||||
Package Arch Version Repository Size
|
Package Arch Version Repository Size
|
||||||
===============================================================================================================================================
|
========================================================================================
|
||||||
Installing:
|
Installing:
|
||||||
epel-release noarch 7-5 extras 14 k
|
epel-release noarch 7-5 extras 14 k
|
||||||
|
|
||||||
### Step 2: After installing the repo, we can start with the installation of the nDeploy RPM repo for CentOS to install our required nDeploy Webstack and Nginx plugin. ###
|
### 第二步: 可以安装 nDeploy 的 CentOS RPM 库来安装我们所需的 nDeploy Web 类软件和 Nginx 插件 ###
|
||||||
|
|
||||||
root@server1 [/usr]# yum -y install http://rpm.piserve.com/nDeploy-release-centos-1.0-1.noarch.rpm
|
root@server1 [/usr]# yum -y install http://rpm.piserve.com/nDeploy-release-centos-1.0-1.noarch.rpm
|
||||||
Loaded plugins: fastestmirror, tsflags, universal-hooks
|
Loaded plugins: fastestmirror, tsflags, universal-hooks
|
||||||
@ -51,13 +45,13 @@ First of all, we need to install the EPEL repo to start-up with the process.
|
|||||||
|
|
||||||
Dependencies Resolved
|
Dependencies Resolved
|
||||||
|
|
||||||
===============================================================================================================================================
|
========================================================================================
|
||||||
Package Arch Version Repository Size
|
Package Arch Version Repository Size
|
||||||
===============================================================================================================================================
|
========================================================================================
|
||||||
Installing:
|
Installing:
|
||||||
nDeploy-release-centos noarch 1.0-1 /nDeploy-release-centos-1.0-1.noarch 110
|
nDeploy-release-centos noarch 1.0-1 /nDeploy-release-centos-1.0-1.noarch 110
|
||||||
|
|
||||||
### Step 3: Install the nDeploy and Nginx nDeploy plugins. ###
|
### 第三步:安装 nDeploy 和 Nginx nDeploy 插件 ###
|
||||||
|
|
||||||
root@server1 [/usr]# yum --enablerepo=ndeploy install nginx-nDeploy nDeploy
|
root@server1 [/usr]# yum --enablerepo=ndeploy install nginx-nDeploy nDeploy
|
||||||
Loaded plugins: fastestmirror, tsflags, universal-hooks
|
Loaded plugins: fastestmirror, tsflags, universal-hooks
|
||||||
@ -70,9 +64,9 @@ First of all, we need to install the EPEL repo to start-up with the process.
|
|||||||
|
|
||||||
Dependencies Resolved
|
Dependencies Resolved
|
||||||
|
|
||||||
===============================================================================================================================================
|
========================================================================================
|
||||||
Package Arch Version Repository Size
|
Package Arch Version Repository Size
|
||||||
===============================================================================================================================================
|
========================================================================================
|
||||||
Installing:
|
Installing:
|
||||||
nDeploy noarch 2.0-11.el7 ndeploy 80 k
|
nDeploy noarch 2.0-11.el7 ndeploy 80 k
|
||||||
nginx-nDeploy x86_64 1.8.0-34.el7 ndeploy 36 M
|
nginx-nDeploy x86_64 1.8.0-34.el7 ndeploy 36 M
|
||||||
@ -84,12 +78,12 @@ First of all, we need to install the EPEL repo to start-up with the process.
|
|||||||
python-lxml x86_64 3.2.1-4.el7 base 758 k
|
python-lxml x86_64 3.2.1-4.el7 base 758 k
|
||||||
|
|
||||||
Transaction Summary
|
Transaction Summary
|
||||||
===============================================================================================================================================
|
========================================================================================
|
||||||
Install 2 Packages (+5 Dependent packages)
|
Install 2 Packages (+5 Dependent packages)
|
||||||
|
|
||||||
With these steps, we've completed with the installation of Nginx plugin in our server. Now we need to configure Nginx as reverse proxy and create the virtualhost for the existing cPanel user accounts. For that we can run the following script.
|
通过以上这些步骤,我们完成了在我们的服务器上 Nginx 插件的安装。现在我们可以配置 Nginx 作为反向代理和为已有的 cPanel 用户账户创建虚拟主机,为此我们可以运行如下脚本。
|
||||||
|
|
||||||
### Step 4: To enable Nginx as a front end Web Server and create the default configuration files. ###
|
### 第四步:启动 Nginx 作为默认的前端 Web 服务器,并创建默认的配置文件###
|
||||||
|
|
||||||
root@server1 [/usr]# /opt/nDeploy/scripts/cpanel-nDeploy-setup.sh enable
|
root@server1 [/usr]# /opt/nDeploy/scripts/cpanel-nDeploy-setup.sh enable
|
||||||
Modifying apache http and https port in cpanel
|
Modifying apache http and https port in cpanel
|
||||||
@ -101,9 +95,9 @@ With these steps, we've completed with the installation of Nginx plugin in our s
|
|||||||
ConfGen:: saheetha
|
ConfGen:: saheetha
|
||||||
ConfGen:: satest
|
ConfGen:: satest
|
||||||
|
|
||||||
As you can see these script will modify the Apache port from 80 to another port to make Nginx run as a front end web server and create the virtual host configuration files for the existing cPanel accounts. Once it is done, confirm the status of both Apache and Nginx.
|
你可以看到这个脚本将修改 Apache 的端口从 80 到另一个端口来让 Nginx 作为前端 Web 服务器,并为现有的 cPanel 用户创建虚拟主机配置文件。一旦完成,确认 Apache 和 Nginx 的状态。
|
||||||
|
|
||||||
### Apache Status: ###
|
#### Apache 状态:####
|
||||||
|
|
||||||
root@server1 [/var/run/httpd]# systemctl status httpd
|
root@server1 [/var/run/httpd]# systemctl status httpd
|
||||||
● httpd.service - Apache Web Server
|
● httpd.service - Apache Web Server
|
||||||
@ -118,7 +112,7 @@ As you can see these script will modify the Apache port from 80 to another port
|
|||||||
Jan 18 06:34:23 server1.centos7-test.com apachectl[25606]: httpd (pid 24760) already running
|
Jan 18 06:34:23 server1.centos7-test.com apachectl[25606]: httpd (pid 24760) already running
|
||||||
Jan 18 06:34:23 server1.centos7-test.com systemd[1]: Started Apache Web Server.
|
Jan 18 06:34:23 server1.centos7-test.com systemd[1]: Started Apache Web Server.
|
||||||
|
|
||||||
### Nginx Status: ###
|
#### Nginx 状态:####
|
||||||
|
|
||||||
root@server1 [~]# systemctl status nginx
|
root@server1 [~]# systemctl status nginx
|
||||||
● nginx.service - nginx-nDeploy - high performance web server
|
● nginx.service - nginx-nDeploy - high performance web server
|
||||||
@ -137,7 +131,7 @@ As you can see these script will modify the Apache port from 80 to another port
|
|||||||
Jan 17 17:18:29 server1.centos7-test.com nginx[3804]: nginx: configuration file /etc/nginx/nginx.conf test is successful
|
Jan 17 17:18:29 server1.centos7-test.com nginx[3804]: nginx: configuration file /etc/nginx/nginx.conf test is successful
|
||||||
Jan 17 17:18:29 server1.centos7-test.com systemd[1]: Started nginx-nDeploy - high performance web server.
|
Jan 17 17:18:29 server1.centos7-test.com systemd[1]: Started nginx-nDeploy - high performance web server.
|
||||||
|
|
||||||
Nginx act as a frontend webserver running on port 80 and Apache configuration is modified to listen on http port 9999 and https port 4430. Please see their status below:
|
Nginx 作为前端服务器运行在 80 端口,Apache 配置被更改为监听 http 端口 9999 和 https 端口 4430。请看他们的情况:
|
||||||
|
|
||||||
root@server1 [/usr/local/src]# netstat -plan | grep httpd
|
root@server1 [/usr/local/src]# netstat -plan | grep httpd
|
||||||
tcp 0 0 0.0.0.0:4430 0.0.0.0:* LISTEN 17270/httpd
|
tcp 0 0 0.0.0.0:4430 0.0.0.0:* LISTEN 17270/httpd
|
||||||
@ -151,13 +145,13 @@ Nginx act as a frontend webserver running on port 80 and Apache configuration is
|
|||||||
tcp 0 0 127.0.0.1:80 0.0.0.0:* LISTEN 17802/nginx: master
|
tcp 0 0 127.0.0.1:80 0.0.0.0:* LISTEN 17802/nginx: master
|
||||||
tcp 0 0 45.79.183.73:80 0.0.0.0:* LISTEN 17802/nginx: master
|
tcp 0 0 45.79.183.73:80 0.0.0.0:* LISTEN 17802/nginx: master
|
||||||
|
|
||||||
The virtualhost entries created for the existing users as located in the folder "**/etc/nginx/sites-enabled**". This file path is included in the Nginx main configuration file.
|
为已有用户创建的虚拟主机的配置文件在 "**/etc/nginx/sites-enabled**"。 这个文件路径包含了 Nginx 主要配置文件。
|
||||||
|
|
||||||
root@server1 [/etc/nginx/sites-enabled]# ll | grep .conf
|
root@server1 [/etc/nginx/sites-enabled]# ll | grep .conf
|
||||||
-rw-r--r-- 1 root root 311 Jan 17 09:02 saheetha.com.conf
|
-rw-r--r-- 1 root root 311 Jan 17 09:02 saheetha.com.conf
|
||||||
-rw-r--r-- 1 root root 336 Jan 17 09:02 saheethastest.com.conf
|
-rw-r--r-- 1 root root 336 Jan 17 09:02 saheethastest.com.conf
|
||||||
|
|
||||||
### Sample Vhost for a domain: ###
|
#### 一个域名的示例虚拟主机:###
|
||||||
|
|
||||||
server {
|
server {
|
||||||
|
|
||||||
@ -173,7 +167,7 @@ The virtualhost entries created for the existing users as located in the folder
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
We can confirm the working of the web server status by calling a website in the browser. Please see the web server information on my server after the installation.
|
我们可以启动浏览器查看网站来确定 Web 服务器的工作状态。安装后,请阅读服务器上的 web 服务信息。
|
||||||
|
|
||||||
root@server1 [/home]# ip a | grep -i eth0
|
root@server1 [/home]# ip a | grep -i eth0
|
||||||
3: eth0: <BROADCAST,MULTICAST,UP,LOWER_UP> mtu 1500 qdisc pfifo_fast state UP qlen 1000
|
3: eth0: <BROADCAST,MULTICAST,UP,LOWER_UP> mtu 1500 qdisc pfifo_fast state UP qlen 1000
|
||||||
@ -183,24 +177,24 @@ We can confirm the working of the web server status by calling a website in the
|
|||||||
|
|
||||||
![webserver-status](http://blog.linoxide.com/wp-content/uploads/2016/01/webserver.png)
|
![webserver-status](http://blog.linoxide.com/wp-content/uploads/2016/01/webserver.png)
|
||||||
|
|
||||||
Nginx will create the virtual host automatically for any newly created accounts in cPanel. With these simple steps we can configure Nginx as reverse proxy on a CentOS 7/CPanel server.
|
Nginx 将会为任何最新在 cPanel 中创建的账户创建虚拟主机。通过这些简单的的步骤,我们能够在一台 CentOS 7 / cPanel 的服务器上配置 Nginx 作为反向代理。
|
||||||
|
|
||||||
### Advantages of Nginx as Reverse Proxy: ###
|
### Nginx 作为反向代理的优势###
|
||||||
|
|
||||||
1. Easy to install and configure
|
1. 便于安装和配置。
|
||||||
2. Performance and efficiency
|
2. 效率高、性能好。
|
||||||
3. Prevent DDOS attacks
|
3. 防止 Ddos 攻击。
|
||||||
4. Allows .htaccess PHP rewrite rules
|
4. 支持使用 .htaccess 作为 PHP 的重写规则。
|
||||||
|
|
||||||
I hope this article is useful for you guys. Thank you for referring to this. I would appreciate your valuable comments and suggestions on this for further improvements.
|
我希望这篇文章对你们有用。感谢你看它。我非常高兴收到你的宝贵意见和建议,并进一步改善。
|
||||||
|
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
via: http://linoxide.com/linux-how-to/set-nginx-reverse-proxy-centos-7-cpanel/
|
via: http://linoxide.com/linux-how-to/set-nginx-reverse-proxy-centos-7-cpanel/
|
||||||
|
|
||||||
作者:[Saheetha Shameer][a]
|
作者:[Saheetha Shameer][a]
|
||||||
译者:[译者ID](https://github.com/译者ID)
|
译者:[bestony](https://github.com/bestony)
|
||||||
校对:[校对者ID](https://github.com/校对者ID)
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
@ -0,0 +1,73 @@
|
|||||||
|
一位跨平台开发者的自白
|
||||||
|
=============================================
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/images/business/business_clouds.png?itok=cucHuJnU)
|
||||||
|
|
||||||
|
[Andreia Gaita][1] 在 OSCON 开源大会上发表了一个题为[跨平台开发者的自白][2]的演讲。她长期从事于开源工作,并且为 [Mono][3] 工程(LCTT 译注:一个致力于开创 .NET 在 Linux 上使用的开源工程)做着贡献,主要以 C#/C++ 开发。Andreia 任职于 GitHub,她的工作是专注构建 Visual Studio 的 GitHub 扩展管理器。
|
||||||
|
|
||||||
|
我在她发表演讲前就迫不及待的想要问她一些关于跨平台开发的事,问问她作为一名跨平台开发者在这 16 年之中学习到了什么。
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/images/life/Interview%20banner%20Q%26A.png)
|
||||||
|
|
||||||
|
**在你开发跨平台代码中,你使用过的最简单的和最难的代码语言是什么?**
|
||||||
|
|
||||||
|
我很少讨论某种语言的好坏,更多是讨论是那些语言有哪些库和工具。语言的编译器、解释器以及构建系统决定了用它们做跨平台开发的难易程度(或者它们是否可能做跨平台开发),可用的 UI 库和对本地系统的访问能力决定了与该操作系统集成的紧密程度。按照我的观点,我认为 C# 最适合完成跨平台开发工作。这种语言自身包括了允许快速的本地调用和精确的内存映射的功能,如果你希望你的代码能够与系统和本地函数库进行交互就需要这些功能。而当我需要非常特殊的系统功能时,我就会切换到 C 或者 C++。
|
||||||
|
|
||||||
|
**你使用的跨平台开发工具或者抽象层有哪些?**
|
||||||
|
|
||||||
|
我的大部分跨平台工作都是为其它需要开发跨平台应用的人开发工具、库和绑定(binding),一般是用 MONO/C# 和 C/C++。在抽象的层面我用的不多,更多是在 glib 库和友元(friends)方面。大多数时候,我用 Mono 去完成各种跨平台应用的,包括 UI,或者偶然在游戏开发中用到 Unity3D 的部分。我经常使用 Electron(LCTT 译注:Atom 编辑器的兄弟项目,可以用 Electron 开发桌面应用)。
|
||||||
|
|
||||||
|
**你接触过哪些构建系统?它们之间的区别是由于语言还是平台的不同?**
|
||||||
|
|
||||||
|
我试着选择适合我使用的语言的构建系统。那样,就会很少遇到让我头疼的问题(希望如此)。它需要支持平台和体系结构间的选择、构建输出结果位置可以智能些(多个并行构建),以及易配置性等。大多数时候,我的项目会结合使用 C/C++ 和 C#,我要从同一源代码同时构建不同的配置环境(调试、发布、Windows、OSX、Linux、Android、iOS 等等),这通常需要为每个构建的输出结果选择带有不同参数的不同编译器。构建系统可以帮我做到这一切而不用让我(太)费心。我时常尝试着用不同的构建系统,看看有些什么新的变化,但是最终,我还是回到了使用 makefile 的情况,并结合使用 shell 和批处理脚本或 Perl 脚本来完成工作(因为如果我希望用户来构建我的软件,我还是最好选择一种到处都可以用的命令行脚本语言)。
|
||||||
|
|
||||||
|
**你怎样平衡在这种使用统一的用户界面下提供原生的外观和体验的需求呢?**
|
||||||
|
|
||||||
|
跨平台的用户界面的实现很困难。在过去几年中我已经使用了一些跨平台 GUI,并且我认为这些事情上并没有最优解。基本上有两种选择。你可以选择一个跨平台工具去做一个并不是完全适合你所有支持的平台的 UI,但是代码库比较小,维护成本比较低。或者你可以选择去开发针对平台的 UI,那样看起来更原生,集成的也更好,但是需要更大的代码库和更高的维护成本。这种决定完全取决于 APP 的类型、它有多少功能、你有多少资源,以及你要把它运行在多少平台上?
|
||||||
|
|
||||||
|
最后,我认为用户比较接受这种“一个 UI 打通关”了,就比如 Electron 框架。我有个 Chromium+C+C# 的框架侧项目,有一天我希望可以用 C# 构建 Electron 型的 app,这样的话我就可以做到两全其美了。
|
||||||
|
|
||||||
|
**构建/打包系统的依赖性对你有影响吗 ?**
|
||||||
|
|
||||||
|
我依赖的使用方面很保守,我被崩溃的 ABI(LCTT 译注:应用程序二进制接口)、冲突的符号、以及丢失的包等问题困扰了太多次。我决定我要针对的操作系统版本,并选择最低的公有部分来使问题最小化。通常这就意味着有五种不同的 Xcode 和 OSX 框架库,要在同样的机器上相应安装五种不同的 Visual Studio 版本,多种 clang(LCTT 译注:C语言、C++、Object-C、C++ 语言的轻量级编译器)和 gcc 版本,一系列的运行着各种发行版的虚拟机。如果我不能确定我要使用的操作系统的包的状态,我有时就会静态连接库,有时会子模块化依赖以确保它们一直可用。大多时候,我会避免这些很棘手的问题,除非我非常需要使用他们。
|
||||||
|
|
||||||
|
**你使用持续集成(CI)、代码审查以及相关的工具吗?**
|
||||||
|
|
||||||
|
基本每天都用。这是保持高效的唯一方式。我在一个项目中做的第一件事情是配置跨平台构建脚本,保证每件事尽可能自动化完成。当你面向多平台开发的时候,持续集成是至关重要的。没有人能在一个机器上构建各种平台的不同组合,并且一旦你的构建过程没有包含所有的平台,你就不会注意到你搞砸的事情。在一个共享式的多平台代码库中,不同的人拥有不同的平台和功能,所以保证质量的唯一的方法是跨团队代码审查结合持续集成和其他分析工具。这不同于其他的软件项目,如果不使用相关的工具就会面临失败。
|
||||||
|
|
||||||
|
**你依赖于自动构建测试,或者倾向于在每个平台上构建并且进行本地测试吗?**
|
||||||
|
|
||||||
|
对于不包括 UI 的工具和库,我通常使用自动构建测试。如果有 UI,两种方法我都会用到——针对已有的 GUI 工具的可靠的、可脚本化的 UI 自动化少到几乎没有,所以我要么我去针对我要跨我所支持的平台创建 UI 自动化工具,要么手动进行测试。如果一个项目使用了定制的 UI 库(比如说一个类似 Unity3D 的 OpenGL UI),开发可编程的自动化工具并自动化大多数工作就相当容易。不过,没有什么东西会像人一样双击就测试出问题。
|
||||||
|
|
||||||
|
**如果你要做跨平台开发,你喜欢用跨编辑器的构建系统,比如在 Windows 上使用 Visual Studio,在 Linux 上使用 Qt Creator,在 Mac 上使用 XCode 吗?还是你更趋向于使用 Eclipse 这样的可以在所有平台上使用的单一平台?**
|
||||||
|
|
||||||
|
我喜欢使用跨编辑器的构建系统。我更喜欢在不同的IDE上保存项目文件(这样可以使增加 IDE 变得更容易),通过使用构建脚本让 IDE 在它们支持的平台上去构建。对于一个开发者来说编辑器是最重要的工具,学习它们是需要花费时间和精力的,而它们是不可相互替代的。我有我自己喜欢的编辑器和工具,每个人也可以使用他们最喜爱的工具。
|
||||||
|
|
||||||
|
**在跨平台开发的时候,你更喜欢使用什么样的编辑器、开发环境和 IDE 呢?**
|
||||||
|
|
||||||
|
跨平台开发者被限制在只能选择可以在多数平台上工作的所共有的不多选择之一。我爱用 Visual Studio,但是我不能依赖它完成除 Windows 平台之外的工作(你可能不想让 Windows 成为你的主要的交叉编译平台),所以我不会使用它作为我的主要 IDE。即使可以,跨平台开发者的核心技能也是尽可能的了解和使用大量的平台。这就意味着必须很熟悉它们——使用该平台上的编辑器和库,了解这种操作系统及其适用场景、运行方式以及它的局限性等。做这些事情就需要头脑清醒(我的捷径是加强记忆),我必须依赖跨平台的编辑器。所以我使用 Emacs 和 Sublime。
|
||||||
|
|
||||||
|
**你之前和现在最喜欢的跨平台项目是什么?**
|
||||||
|
|
||||||
|
我一直很喜欢 Mono,并且得心应手,其它的项目大部分都是以某种方式围绕着它进行的。Gluezilla 是我在多年前开发的一个 Mozilla 绑定(binding),可以把 C# 开发的应用嵌入到 Web 浏览器页面中,并且看起来很有特色。我开发过一个 Winform 应用,它是在 linux 上开发的,它可以在 Windows 上运行在一个 Mozilla 浏览器页面里嵌入的 GTK 视图中。CppSharp 项目(以前叫做 Cxxi,更早时叫做 CppInterop)是一个我开始为 C++ 库生成 C# 绑定(binding)的项目,这样就可以在 C# 中调用、创建实例、子类化 C++ 类。这样,它在运行的时候就能够检测到所使用的平台,以及用来创建本地运行库的是什么编译器,并为它生成正确的 C# 绑定(binding)。这多么有趣啊。
|
||||||
|
|
||||||
|
**你怎样看跨平台开发的未来趋势呢?**
|
||||||
|
|
||||||
|
我们构建本地应用程序的方式已经改变了,我感觉在各种桌面操作系统的明显差异在慢慢变得模糊;所以构建跨平台的应用程序将会更加容易,而且对系统的集成也不需要完全本地化。不好的是,这可能意味着应用程序易用性更糟,并且在发挥操作系统特性方面所能做的更少。库、工具以及运行环境的跨平台开发是一种我们知道怎样做的更好,但是跨平台应用程序的开发仍然需要我们的努力。
|
||||||
|
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://opensource.com/business/16/5/oscon-interview-andreia-gaita
|
||||||
|
|
||||||
|
作者:[Marcus D. Hanwell][a]
|
||||||
|
译者:[vim-kakali](https://github.com/vim-kakali)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创翻译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://opensource.com/users/mhanwell
|
||||||
|
[1]: https://twitter.com/sh4na
|
||||||
|
[2]: http://conferences.oreilly.com/oscon/open-source-us/public/schedule/detail/48702
|
||||||
|
[3]: http://www.mono-project.com/
|
||||||
|
|
85
published/201609/20160506 Setup honeypot in Kali Linux.md
Normal file
85
published/201609/20160506 Setup honeypot in Kali Linux.md
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
在 Kali Linux 环境下设置蜜罐
|
||||||
|
=========================
|
||||||
|
|
||||||
|
Pentbox 是一个包含了许多可以使渗透测试工作变得简单流程化的工具的安全套件。它是用 Ruby 编写并且面向 GNU / Linux,同时也支持 Windows、MacOS 和其它任何安装有 Ruby 的系统。在这篇短文中我们将讲解如何在 Kali Linux 环境下设置蜜罐。如果你还不知道什么是蜜罐(honeypot),“蜜罐是一种计算机安全机制,其设置用来发现、转移、或者以某种方式,抵消对信息系统的非授权尝试。"
|
||||||
|
|
||||||
|
### 下载 Pentbox:
|
||||||
|
|
||||||
|
在你的终端中简单的键入下面的命令来下载 pentbox-1.8。
|
||||||
|
|
||||||
|
```
|
||||||
|
root@kali:~# wget http://downloads.sourceforge.net/project/pentbox18realised/pentbox-1.8.tar.gz
|
||||||
|
```
|
||||||
|
|
||||||
|
![](https://www.blackmoreops.com/wp-content/uploads/2016/05/Set-up-a-honeypot-in-Kali-Linux-blackMORE-Ops-1.jpg)
|
||||||
|
|
||||||
|
### 解压 pentbox 文件
|
||||||
|
|
||||||
|
使用如下命令解压文件:
|
||||||
|
|
||||||
|
```
|
||||||
|
root@kali:~# tar -zxvf pentbox-1.8.tar.gz
|
||||||
|
```
|
||||||
|
|
||||||
|
![](https://www.blackmoreops.com/wp-content/uploads/2016/05/Set-up-a-honeypot-in-Kali-Linux-blackMORE-Ops-2.jpg)
|
||||||
|
|
||||||
|
### 运行 pentbox 的 ruby 脚本
|
||||||
|
|
||||||
|
改变目录到 pentbox 文件夹:
|
||||||
|
|
||||||
|
```
|
||||||
|
root@kali:~# cd pentbox-1.8/
|
||||||
|
```
|
||||||
|
|
||||||
|
![](https://www.blackmoreops.com/wp-content/uploads/2016/05/Set-up-a-honeypot-in-Kali-Linux-blackMORE-Ops-3.jpg)
|
||||||
|
|
||||||
|
使用下面的命令来运行 pentbox:
|
||||||
|
|
||||||
|
```
|
||||||
|
root@kali:~# ./pentbox.rb
|
||||||
|
```
|
||||||
|
|
||||||
|
![](https://www.blackmoreops.com/wp-content/uploads/2016/05/Set-up-a-honeypot-in-Kali-Linux-blackMORE-Ops-4.jpg)
|
||||||
|
|
||||||
|
### 设置一个蜜罐
|
||||||
|
|
||||||
|
使用选项 2 (Network Tools) 然后是其中的选项 3 (Honeypot)。
|
||||||
|
|
||||||
|
![](https://www.blackmoreops.com/wp-content/uploads/2016/05/Set-up-a-honeypot-in-Kali-Linux-blackMORE-Ops-5.jpg)
|
||||||
|
|
||||||
|
完成让我们执行首次测试,选择其中的选项 1 (Fast Auto Configuration)
|
||||||
|
|
||||||
|
![](https://www.blackmoreops.com/wp-content/uploads/2016/05/Set-up-a-honeypot-in-Kali-Linux-blackMORE-Ops-6.jpg)
|
||||||
|
|
||||||
|
这样就在 80 端口上开启了一个蜜罐。打开浏览器并且打开链接 http://192.168.160.128 (这里的 192.168.160.128 是你自己的 IP 地址。)你应该会看到一个 Access denied 的报错。
|
||||||
|
|
||||||
|
|
||||||
|
![](https://www.blackmoreops.com/wp-content/uploads/2016/05/Set-up-a-honeypot-in-Kali-Linux-blackMORE-Ops-7.jpg)
|
||||||
|
|
||||||
|
并且在你的终端应该会看到 “HONEYPOT ACTIVATED ON PORT 80” 和跟着的 “INTRUSION ATTEMPT DETECTED”。
|
||||||
|
|
||||||
|
![](https://www.blackmoreops.com/wp-content/uploads/2016/05/Set-up-a-honeypot-in-Kali-Linux-blackMORE-Ops-8.jpg)
|
||||||
|
|
||||||
|
现在,如果你在同一步选择了选项 2 (Manual Configuration), 你应该看见更多的其它选项:
|
||||||
|
|
||||||
|
![](https://www.blackmoreops.com/wp-content/uploads/2016/05/Set-up-a-honeypot-in-Kali-Linux-blackMORE-Ops-9.jpg)
|
||||||
|
|
||||||
|
执行相同的步骤但是这次选择 22 端口 (SSH 端口)。接着在你家里的路由器上做一个端口转发,将外部的 22 端口转发到这台机器的 22 端口上。或者,把这个蜜罐设置在你的云端服务器的一个 VPS 上。
|
||||||
|
|
||||||
|
你将会被有如此多的机器在持续不断地扫描着 SSH 端口而震惊。 你知道你接着应该干什么么? 你应该黑回它们去!桀桀桀!
|
||||||
|
|
||||||
|
如果视频是你的菜的话,这里有一个设置蜜罐的视频:
|
||||||
|
|
||||||
|
<https://youtu.be/NufOMiktplA>
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://www.blackmoreops.com/2016/05/06/setup-honeypot-in-kali-linux/
|
||||||
|
|
||||||
|
作者:[blackmoreops.com][a]
|
||||||
|
译者:[wcnnbdk1](https://github.com/wcnnbdk1)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: blackmoreops.com
|
@ -0,0 +1,89 @@
|
|||||||
|
对比 Android 和 iPhone 的优缺点
|
||||||
|
===================================
|
||||||
|
|
||||||
|
>当我们比较 Android 与 iPhone 的时候,很显然 Android 具有一定的优势,而 iPhone 则在一些关键方面更好。但是,究竟哪个比较好呢?
|
||||||
|
|
||||||
|
对 Android 与 iPhone 比较是个个人的问题。
|
||||||
|
|
||||||
|
就好比我来说,我两个都用。我深知这两个平台的优缺点。所以,我决定分享我关于这两个移动平台的观点。另外,然后谈谈我对新的 Ubuntu 移动平台的印象和它的优势。
|
||||||
|
|
||||||
|
### iPhone 的优点
|
||||||
|
|
||||||
|
虽然这些天我是个十足的 Android 用户,但我必须承认 iPhone 在某些方面做的是不错。首先,苹果公司在他们的设备更新方面有更好的成绩。这对于运行着 iOS 的旧设备来说尤其是这样。反观 Android ,如果不是谷歌亲生的 Nexus,它最好也不过是一个更高端的运营商支持的手机,你将发现它们的更新少的可怜或者根本没有。
|
||||||
|
|
||||||
|
其中 iPhone 做得很好的另一个领域是应用程序的可用性。展开来说,iPhone 应用程序几乎总是有一个简洁的外观。这并不是说 Android 应用程序就是丑陋的,而是,它们可能只是没有像 iOS 上一样的保持不变的操控习惯和一以贯之的用户体验。两个典型的例子, [Dark Sky][1] (天气)和 [Facebook Paper][2] 很好表现了 iOS 独有的布局。
|
||||||
|
|
||||||
|
再有就是备份过程。 Android 可以备份,默认情况下是备份到谷歌。但是对应用数据起不了太大作用。对比 iPhone ,iCloud 基本上可以对你的 iOS 设备进行了完整备份。
|
||||||
|
|
||||||
|
### iPhone 令我失望的地方
|
||||||
|
|
||||||
|
对 iPhone 来说,最无可争辩的问题是它的硬件限制要比软件限制更大,换句话来说,就是存储容量问题。
|
||||||
|
|
||||||
|
你看,对于大多数 Android 手机,我可以买一个容量较小的手机,然后以后可以添加 SD 卡。这意味着两件事:第一,我可以使用 SD 卡来存储大量的媒体文件。其次,我甚至可以用 SD 卡来存储“一些”我的应用程序。而苹果完全不能这么做。
|
||||||
|
|
||||||
|
另一个 iPhone 让我失望的地方是它提供的选择很少。备份您的设备?希望你喜欢 iTunes 或 iCloud 吧。但对一些像我一样用 Linux 的人,那就意味着,我唯一的选择便是使用 iCloud。
|
||||||
|
|
||||||
|
要公平的说,如果你愿意越狱,你的 iPhone 还有一些其他解决方案的,但这并不是这篇文章所讲的。 Android 的 解锁 root 也一样。本文章针对的是两个平台的原生设置。
|
||||||
|
|
||||||
|
最后,让我们不要忘记这件看起来很小的事—— [iTunes 会删掉用户的音乐][3],因为它认为和苹果音乐的内容重复了……或者因为一些其它的类似规定。这不是 iPhone 特有的情况?我不同意,因为那些音乐最终就是在 iPhone 上没有了。我能十分肯定地这么说,是因为不管在哪里我都不会说这种谎话。
|
||||||
|
|
||||||
|
![](http://www.datamation.com/imagesvr_ce/5552/mobile-abstract-icon-200x150.jpg)
|
||||||
|
|
||||||
|
*Android 和 iPhone 的对决取决于什么功能对你来说最重要。*
|
||||||
|
|
||||||
|
### Android 的优点
|
||||||
|
|
||||||
|
Android 给我最大的好处就是 iPhone 所提供不了的:选择。这包括对应用程序、设备以及手机是整体如何工作的选择。
|
||||||
|
|
||||||
|
我爱桌面小工具!对于 iPhone 用户来说,它们也许看上去很蠢。但我可以告诉你,它们可以让我不用打开应用程序就可以看到所需的数据,而无需额外的麻烦。另一个类似的功能,我喜欢安装定制的桌面界面,而不是我的手机默认的那个!
|
||||||
|
|
||||||
|
最后,我可以通过像 [Airdroid][4] 和 [Tasker][5] 这样的工具给我的智能手机添加计算机级的完整功能。AirDroid 可以让我把我的 Android 手机当成带有一个文件管理和通信功能的计算机——这可以让我可以轻而易举的使用鼠标和键盘。Tasker 更厉害,我可以用它让我手机根据环境变得可联系或不可联系,当我设置好了之后,当我到会议室之后我的手机就会自己进入会议模式,甚至变成省电模式。我还可以设置它当我到达特定的目的地时自动启动某个应用程序。
|
||||||
|
|
||||||
|
### Android 让我失望的地方
|
||||||
|
|
||||||
|
Android 备份选项仅限于特定的用户数据,而不是手机的完整克隆。如果不解锁 root,要么你只能听之任之,要么你必须用 Android SDK 来解决。期望普通用户会解锁 root 或运行 SDK 来完成备份(我的意思是一切都备份)显然是一个笑话。
|
||||||
|
|
||||||
|
是的,谷歌的备份服务会备份谷歌应用程序的数据、以及其他相关的自定义设置。但它是远不及我们所看到的苹果服务一样完整。为了完成类似于苹果那样的功能,我发现你就要么必须解锁 root ,要么将其连接到一个在 PC 机上使用一些不知道是什么的软件来干这个。
|
||||||
|
|
||||||
|
不过,公平的说,我知道使用 Nexus 的人能从该设备特有的[完整备份服务][6]中得到帮助。对不起,但是谷歌的默认备份方案是不行的。对于通过在 PC 上使用 adb (Android Debug Bridge) 来备份也是一样的——不会总是如预期般的恢复。
|
||||||
|
|
||||||
|
等吧,它会变好的。经过了很多失败的失望和挫折之后,我发现有一个应用程序,看起来它“可能”提供了一点点微小的希望,它叫 Helium 。它不像我发现的其他应用程序那样拥有误导性的和令人沮丧的局限性,[Helium][7] 最初看起来像是谷歌应该一直提供的备份应用程序——注意,只是“看起来像”。可悲的是,它绊了我一跤。第一次运行时,我不仅需要将它连接到我的计算机上,甚至使用他们提供的 Linux 脚本都不工作。在删除他们的脚本后,我弄了一个很好的老式 adb 来备份到我的 Linux PC 上。你可能要知道的是:你需要在开发工具里打开一箩筐东西,而且如果你运行 Twilight 应用的话还要关闭它。当 adb 的备份选项在我的手机上不起作用时,我花了一点时间把这个搞好了。
|
||||||
|
|
||||||
|
最后,Android 为非 root 用户也提供了可以轻松备份一些如联系人、短信等简单东西的选择。但是,要深度手机备份的话,以我经验还是通过有线连接和 adb。
|
||||||
|
|
||||||
|
### Ubuntu 能拯救我们吗?
|
||||||
|
|
||||||
|
在手机领域,通过对这两大玩家之间的优劣比较,我们很期望从 Ubuntu 看到更好的表现。但是,迄今为止,它的表现相当低迷。
|
||||||
|
|
||||||
|
我喜欢开发人员基于这个操作系统正在做的那些努力,我当然想在 iPhone 和 Android 手机之外有第三种选择。但是不幸的是,它在手机和平板上并不受欢迎,而且由于硬件的低端以及在 YouTube 上的糟糕的演示,有很多不好的传闻。
|
||||||
|
|
||||||
|
公平来说,我在以前也用过 iPhone 和 Android 的低端机,所以这不是对 Ubuntu 的挖苦。但是它要表现出准备与 iPhone 和 Android 相竞争的功能生态时,那就另说了,这还不是我现在特别感兴趣的东西。在以后的日子里,也许吧,我会觉得 Ubuntu 手机可以满足我的需要了。
|
||||||
|
|
||||||
|
###Android 与 iPhone 之争:为什么 Android 将终究赢得胜利
|
||||||
|
|
||||||
|
忽视 Android 那些痛苦的缺点,它起码给我了选择。它并没有把我限制在只有两种备份数据的方式上。是的,一些 Android 的限制事实上是由于它的关注点在于让我选择如何处理我的数据。但是,我可以选择我自己的设备,想加内存就加内存。Android 可以我让做很多很酷的东西,而 iPhone 根本就没有能力这些事情。
|
||||||
|
|
||||||
|
从根本上来说, Android 给非 root 用户提供了访问手机功能的更大自由。无论是好是坏,这是人们想要的一种自由。现在你们其中有很多 iPhone 的粉丝应该感谢像 [libimobiledevice][8] 这样项目带来的影响。看看苹果阻止 Linux 用户所做的事情……然后问问你自己:作为一个 Linux 用户这真的值得吗?
|
||||||
|
|
||||||
|
发表下评论吧,分享你对 iPhone 、Android 或 Ubuntu 的看法。
|
||||||
|
|
||||||
|
------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.datamation.com/mobile-wireless/android-vs.-iphone-pros-and-cons.html
|
||||||
|
|
||||||
|
作者:[Matt Hartley][a]
|
||||||
|
译者:[jovov](https://github.com/jovov)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创翻译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.datamation.com/author/Matt-Hartley-3080.html
|
||||||
|
[1]: http://darkskyapp.com/
|
||||||
|
[2]: https://www.facebook.com/paper/
|
||||||
|
[3]: https://blog.vellumatlanta.com/2016/05/04/apple-stole-my-music-no-seriously/
|
||||||
|
[4]: https://www.airdroid.com/
|
||||||
|
[5]: http://tasker.dinglisch.net/
|
||||||
|
[6]: https://support.google.com/nexus/answer/2819582?hl=en
|
||||||
|
[7]: https://play.google.com/store/apps/details?id=com.koushikdutta.backup&hl=en
|
||||||
|
[8]: http://www.libimobiledevice.org/
|
||||||
|
|
@ -1,24 +1,25 @@
|
|||||||
Linux 将成为21世纪汽车的主要操作系统
|
Linux 将成为 21 世纪汽车的主要操作系统
|
||||||
===============================================================
|
===============================================================
|
||||||
|
|
||||||
>汽车可不单单是由引擎和华丽外壳组成的,汽车里还有许许多多的计算部件,而 Linux 就在它们里面跑着。
|
> 汽车可不单单是由引擎和华丽外壳组成的,汽车里还有许许多多的计算部件,而 Linux 就在它们里面跑着。
|
||||||
|
|
||||||
Linux 不只运行在你的服务器和手机(安卓)上。它还运行在你的车里。当然了,没人会因为某个车载系统而去买辆车。但是 Linux 已经为像丰田,日产,捷豹路虎,马自达,三菱,斯巴鲁,这些大型汽车制造商改进了信息娱乐系统、平视显示以及其联网汽车(connected car)的 4G 与 Wi-Fi 系统,而且 [Linux 即将登陆福特汽车][1]。
|
Linux 不只运行在你的服务器和手机(安卓)上。它还运行在你的车里。当然了,没人会因为某个车载系统而去买辆车。但是 Linux 已经为像丰田、日产、捷豹路虎这些大型汽车制造商提供了信息娱乐系统、平视显示以及其联网汽车(connected car)的 4G 与 Wi-Fi 系统,而且 [Linux 即将登陆福特汽车][1]、马自达、三菱、斯巴鲁。
|
||||||
|
|
||||||
![](http://zdnet4.cbsistatic.com/hub/i/2016/05/10/743f0c14-6458-4d1e-8723-d2d94d0d0e69/c297b7d52e27e97d8721d4cb46bb371b/agl-logo.jpg)
|
![](http://zdnet4.cbsistatic.com/hub/i/2016/05/10/743f0c14-6458-4d1e-8723-d2d94d0d0e69/c297b7d52e27e97d8721d4cb46bb371b/agl-logo.jpg)
|
||||||
>如今,所有的 Linux 和开源汽车软件的成果都已经在 Automotive Grade Linux 项目下统一标准化了。
|
|
||||||
|
|
||||||
传统软件公司也进入了移动物联网领域。 Movimento, Oracle, Qualcomm, Texas Instruments, UIEvolution 和 VeriSilicon 都已经 [加入 Automotive Grade Linux(AGL)项目][2]。 [AGL][3] 是一个相互协作的开源项目,志在于为联网汽车打造一个基于 Linux 的通用软件栈。
|
*如今,所有的 Linux 和开源汽车软件的成果都已经在 Linux 基金会的 Automotive Grade Linux (AGL)项目下统一标准化了。*
|
||||||
|
|
||||||
|
传统软件公司也进入了移动物联网领域。 Movimento、甲骨文、高通、Texas Instruments、UIEvolution 和 VeriSilicon 都已经[加入 Automotive Grade Linux(AGL)项目][2]。 [AGL][3] 是一个相互协作的开源项目,志在于为联网汽车打造一个基于 Linux 的通用软件栈。
|
||||||
|
|
||||||
“随着联网汽车技术和信息娱乐系统需求的快速增长,AGL 过去几年中得到了极大的发展,” Linux 基金会汽车总经理 Dan Cauchy 如是说。
|
“随着联网汽车技术和信息娱乐系统需求的快速增长,AGL 过去几年中得到了极大的发展,” Linux 基金会汽车总经理 Dan Cauchy 如是说。
|
||||||
|
|
||||||
Cauchy 又补充道,“我们的会员基础不单单只是迅速壮大,而且通过横跨不同的业界实现多元化,从半导体和车载软件到 IoT 和连接云服务。这是一个明显的迹象,即联网汽车的革命已经间接影响到许多行业纵向市场。”
|
Cauchy 又补充道,“我们的会员基础不单单只是迅速壮大,而且通过横跨不同的业界实现了多元化,从半导体和车载软件到 IoT 和连接云服务。这是一个明显的迹象,即联网汽车的革命已经间接影响到许多行业纵向市场。”
|
||||||
|
|
||||||
这些公司在 AGL 发布了新的 AGL Unified Code Base (UCB) 之后加入了 AGL 项目。这个新的 Linux 发行版基于 AGL 和另外两个汽车开源项目: [Tizen][4] 和 [GENIVI Alliance][5] 。 UCB 是第二代 Linux 汽车系统。它从底层开始开发,一直到特定的汽车应用软件。它能处理导航,通信,安全,安保和信息娱乐系统,
|
这些公司在 AGL 发布了新的 AGL Unified Code Base (UCB) 之后加入了 AGL 项目。这个新的 Linux 发行版基于 AGL 和另外两个汽车开源项目: [Tizen][4] 和 [GENIVI Alliance][5] 。 UCB 是第二代 Linux 汽车系统。它从底层开始开发,一直到特定的汽车应用软件。它能处理导航、通信、安全、安保和信息娱乐系统。
|
||||||
|
|
||||||
“汽车行业需要一个标准的开源系统和框架来让汽车制造商和供应商能够快速地将类似智能手机的功能带入到汽车中来。” Cauchy 说。“这个新的发行版将 AGL, Tizen, GENIVI 项目和相关开源代码中的精华部分整合进 AGL Unified Code Base ( UCB ) 中,使得汽车制造商能够利用一个通用平台进行快速创新。 在汽车中采用基于 Linux 的系统来实现所有功能时, AGL 的 UCB 发行版将扮演一个重大的角色。”
|
“汽车行业需要一个标准的开源系统和框架来让汽车制造商和供应商能够快速地将类似智能手机的功能带入到汽车中来。” Cauchy 说。“这个新的发行版将 AGL、Tizen、GENIVI 项目和相关开源代码中的精华部分整合进 AGL Unified Code Base (UCB)中,使得汽车制造商能够利用一个通用平台进行快速创新。 在汽车中采用基于 Linux 的系统来实现所有功能时, AGL 的 UCB 发行版将扮演一个重大的角色。”
|
||||||
|
|
||||||
他说得对。自从 2016 年 1 月发布以来,已有四个汽车公司和十个新的软件厂商加入了 AGL。Esso, 如今的 Exxon, 曾让 “把老虎装入油箱” 这条广告语出了名。我怀疑 “把企鹅装到引擎盖下” 这样的广告语是否也会变得家喻户晓,但是它却道出了事实。 Linux 正在慢慢成为 21 世纪汽车的主要操作系统。
|
他说得对。自从 2016 年 1 月发布以来,已有四个汽车公司和十个新的软件厂商加入了 AGL。Esso,如今的 Exxon, 曾让 “把老虎装入油箱” 这条广告语出了名。我怀疑 “把企鹅装到引擎盖下” 这样的广告语是否也会变得家喻户晓,但是它却道出了事实。 Linux 正在慢慢成为 21 世纪汽车的主要操作系统。
|
||||||
|
|
||||||
------------------------------------------------------------------------------
|
------------------------------------------------------------------------------
|
||||||
|
|
||||||
@ -26,7 +27,7 @@ via: http://www.zdnet.com/article/the-linux-in-your-car-movement-gains-momentum/
|
|||||||
|
|
||||||
作者:[Steven J. Vaughan-Nichols][a]
|
作者:[Steven J. Vaughan-Nichols][a]
|
||||||
译者:[XLCYun](https://github.com/XLCYun)
|
译者:[XLCYun](https://github.com/XLCYun)
|
||||||
校对:[校对者ID](https://github.com/校对者ID)
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创翻译,[Linux中国](https://linux.cn/) 荣誉推出
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创翻译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
@ -0,0 +1,74 @@
|
|||||||
|
使用 Python 和 Asyncio 编写在线多人游戏(一)
|
||||||
|
===================================================================
|
||||||
|
|
||||||
|
你在 Python 中用过异步编程吗?本文中我会告诉你怎样做,而且用一个[能工作的例子][1]来展示它:这是一个流行的贪吃蛇游戏,而且是为多人游戏而设计的。
|
||||||
|
|
||||||
|
- [游戏入口在此,点此体验][2]。
|
||||||
|
|
||||||
|
###1、简介
|
||||||
|
|
||||||
|
在技术和文化领域,大规模多人在线游戏(MMO)毋庸置疑是我们当今世界的潮流之一。很长时间以来,为一个 MMO 游戏写一个服务器这件事总是会涉及到大量的预算与复杂的底层编程技术,不过在最近这几年,事情迅速发生了变化。基于动态语言的现代框架允许在中档的硬件上面处理大量并发的用户连接。同时,HTML5 和 WebSockets 标准使得创建基于实时图形的游戏的直接运行至浏览器上的客户端成为可能,而不需要任何的扩展。
|
||||||
|
|
||||||
|
对于创建可扩展的非堵塞性的服务器来说,Python 可能不是最受欢迎的工具,尤其是和在这个领域里最受欢迎的 Node.js 相比而言。但是最近版本的 Python 正在改变这种现状。[asyncio][3] 的引入和一个特别的 [async/await][4] 语法使得异步代码看起来像常规的阻塞代码一样,这使得 Python 成为了一个值得信赖的异步编程语言,所以我将尝试利用这些新特点来创建一个多人在线游戏。
|
||||||
|
|
||||||
|
###2、异步
|
||||||
|
|
||||||
|
一个游戏服务器应该可以接受尽可能多的用户并发连接,并实时处理这些连接。一个典型的解决方案是创建线程,然而在这种情况下并不能解决这个问题。运行上千的线程需要 CPU 在它们之间不停的切换(这叫做上下文切换),这将导致开销非常大,效率很低下。更糟糕的是使用进程来实现,因为,不但如此,它们还会占用大量的内存。在 Python 中,甚至还有一个问题,Python 的解释器(CPython)并不是针对多线程设计的,相反它主要针对于单线程应用实现最大的性能。这就是为什么它使用 GIL(global interpreter lock),这是一个不允许同时运行多线程 Python 代码的架构,以防止同一个共享对象出现使用不可控。正常情况下,在当前线程正在等待的时候,解释器会转换到另一个线程,通常是等待一个 I/O 的响应(举例说,比如等待 Web 服务器的响应)。这就允许在你的应用中实现非阻塞 I/O 操作,因为每一个操作仅仅阻塞一个线程而不是阻塞整个服务器。然而,这也使得通常的多线程方案变得几近无用,因为它不允许你并发执行 Python 代码,即使是在多核心的 CPU 上也是这样。而与此同时,在一个单一线程中拥有非阻塞 I/O 是完全有可能的,因而消除了经常切换上下文的需要。
|
||||||
|
|
||||||
|
实际上,你可以用纯 Python 代码来实现一个单线程的非阻塞 I/O。你所需要的只是标准的 [select][5] 模块,这个模块可以让你写一个事件循环来等待未阻塞的 socket 的 I/O。然而,这个方法需要你在一个地方定义所有 app 的逻辑,用不了多久,你的 app 就会变成非常复杂的状态机。有一些框架可以简化这个任务,比较流行的是 [tornade][6] 和 [twisted][7]。它们被用来使用回调方法实现复杂的协议(这和 Node.js 比较相似)。这种框架运行在它自己的事件循环中,按照定义的事件调用你的回调函数。并且,这或许是一些情况的解决方案,但是它仍然需要使用回调的方式编程,这使你的代码变得碎片化。与写同步代码并且并发地执行多个副本相比,这就像我们在普通的线程上做的一样。在单个线程上这为什么是不可能的呢?
|
||||||
|
|
||||||
|
这就是为什么出现微线程(microthread)概念的原因。这个想法是为了在一个线程上并发执行任务。当你在一个任务中调用阻塞的方法时,有一个叫做“manager” (或者“scheduler”)的东西在执行事件循环。当有一些事件准备处理的时候,一个 manager 会转移执行权给一个任务,并等着它执行完毕。任务将一直执行,直到它遇到一个阻塞调用,然后它就会将执行权返还给 manager。
|
||||||
|
|
||||||
|
> 微线程也称为轻量级线程(lightweight threads)或绿色线程(green threads)(来自于 Java 中的一个术语)。在伪线程中并发执行的任务叫做 tasklets、greenlets 或者协程(coroutines)。
|
||||||
|
|
||||||
|
Python 中的微线程最早的实现之一是 [Stackless Python][8]。它之所以这么知名是因为它被用在了一个叫 [EVE online][9] 的非常有名的在线游戏中。这个 MMO 游戏自称说在一个持久的“宇宙”中,有上千个玩家在做不同的活动,这些都是实时发生的。Stackless 是一个独立的 Python 解释器,它代替了标准的函数栈调用,并且直接控制程序运行流程来减少上下文切换的开销。尽管这非常有效,这个解决方案不如在标准解释器中使用“软”库更流行,像 [eventlet][10] 和 [gevent][11] 的软件包配备了修补过的标准 I/O 库,I/O 函数会将执行权传递到内部事件循环。这使得将正常的阻塞代码转变成非阻塞的代码变得简单。这种方法的一个缺点是从代码上看这并不分明,它的调用是非阻塞的。新版本的 Python 引入了本地协程作为生成器的高级形式。在 Python 的 3.4 版本之后,引入了 asyncio 库,这个库依赖于本地协程来提供单线程并发。但是仅仅到了 Python 3.5 ,协程就变成了 Python 语言的一部分,使用新的关键字 async 和 await 来描述。这是一个简单的例子,演示了使用 asyncio 来运行并发任务。
|
||||||
|
|
||||||
|
```
|
||||||
|
import asyncio
|
||||||
|
|
||||||
|
async def my_task(seconds):
|
||||||
|
print("start sleeping for {} seconds".format(seconds))
|
||||||
|
await asyncio.sleep(seconds)
|
||||||
|
print("end sleeping for {} seconds".format(seconds))
|
||||||
|
|
||||||
|
all_tasks = asyncio.gather(my_task(1), my_task(2))
|
||||||
|
loop = asyncio.get_event_loop()
|
||||||
|
loop.run_until_complete(all_tasks)
|
||||||
|
loop.close()
|
||||||
|
```
|
||||||
|
|
||||||
|
我们启动了两个任务,一个睡眠 1 秒钟,另一个睡眠 2 秒钟,输出如下:
|
||||||
|
|
||||||
|
```
|
||||||
|
start sleeping for 1 seconds
|
||||||
|
start sleeping for 2 seconds
|
||||||
|
end sleeping for 1 seconds
|
||||||
|
end sleeping for 2 seconds
|
||||||
|
```
|
||||||
|
|
||||||
|
正如你所看到的,协程不会阻塞彼此——第二个任务在第一个结束之前启动。这发生的原因是 asyncio.sleep 是协程,它会返回执行权给调度器,直到时间到了。
|
||||||
|
|
||||||
|
在下一节中,我们将会使用基于协程的任务来创建一个游戏循环。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://7webpages.com/blog/writing-online-multiplayer-game-with-python-asyncio-getting-asynchronous/
|
||||||
|
|
||||||
|
作者:[Kyrylo Subbotin][a]
|
||||||
|
译者:[xinglianfly](https://github.com/xinglianfly)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://7webpages.com/blog/writing-online-multiplayer-game-with-python-asyncio-getting-asynchronous/
|
||||||
|
[1]: http://snakepit-game.com/
|
||||||
|
[2]: http://snakepit-game.com/
|
||||||
|
[3]: https://docs.python.org/3/library/asyncio.html
|
||||||
|
[4]: https://docs.python.org/3/whatsnew/3.5.html#whatsnew-pep-492
|
||||||
|
[5]: https://docs.python.org/2/library/select.html
|
||||||
|
[6]: http://www.tornadoweb.org/
|
||||||
|
[7]: http://twistedmatrix.com/
|
||||||
|
[8]: http://www.stackless.com/
|
||||||
|
[9]: http://www.eveonline.com/
|
||||||
|
[10]: http://eventlet.net/
|
||||||
|
[11]: http://www.gevent.org/
|
@ -0,0 +1,59 @@
|
|||||||
|
Linux 发行版们应该禁用 IPv4 映射的 IPv6 地址吗?
|
||||||
|
=============================================
|
||||||
|
|
||||||
|
从各方面来看,互联网向 IPv6 的过渡是件很缓慢的事情。不过在最近几年,可能是由于 IPv4 地址资源的枯竭,IPv6 的使用处于[上升态势][1]。相应的,开发者也有兴趣确保软件能在 IPv4 和 IPv6 下工作。但是,正如近期 OpenBSD 邮件列表的讨论所关注的,一个使得向 IPv6 转换更加轻松的机制设计同时也可能导致网络更不安全——并且 Linux 发行版们的默认配置可能并不安全。
|
||||||
|
|
||||||
|
### 地址映射
|
||||||
|
|
||||||
|
IPv6 在很多方面看起来可能很像 IPv4,但它是一个不同地址空间的不同的协议。服务器程序想要接受使用二者之中任意一个协议的连接,必须给两个不同的地址族分别打开一个套接字——IPv4 的 `AF_INET` 和 IPv6 的 `AF_INET6`。特别是一个程序希望在主机的使用两种地址协议的任意接口都接受连接的话,需要创建一个绑定到全零通配符地址(`0.0.0.0`)的 `AF_INET` 套接字和一个绑定到 IPv6 等效地址(写作 `::`)的 `AF_INET6` 套接字。它必须在两个套接字上都监听连接——或者有人会这么认为。
|
||||||
|
|
||||||
|
多年前,在 [RFC 3493][2],IETF 指定了一个机制,程序可以使用一个单独的 IPv6 套接字工作在两个协议之上。有了一个启用这个行为的套接字,程序只需要绑定到 `::` 来在所有接口上接受使用这两个协议连接。当创建了一个 IPv4 连接到该绑定端口,源地址会像 [RFC 2373][3] 中描述的那样映射到 IPv6。所以,举个例子,一个使用了这个模式的程序会将一个 `192.168.1.1` 的传入连接看作来自 `::ffff:192.168.1.1`(这个混合的写法就是这种地址通常的写法)。程序也能通过相同的映射方法打开一个到 IPv4 地址的连接。
|
||||||
|
|
||||||
|
RFC 要求这个行为要默认实现,所以大多数系统这么做了。不过也有些例外,OpenBSD 就是其中之一;在那里,希望在两种协议下工作的程序能做的只能是创建两个独立的套接字。但一个在 Linux 中打开两个套接字的程序会遇到麻烦:IPv4 和 IPv6 套接字都会尝试绑定到 IPv4 地址,所以不论是哪个后者都会失败。换句话说,一个绑定到 `::` 指定端口的套接字的程序会同时绑定到 IPv6 `::` 和 IPv4 `0.0.0.0` 地址的那个端口上。如果程序之后尝试绑定一个 IPv4 套接字到 `0.0.0.0` 的相同端口上时,这个操作会失败,因为这个端口已经被绑定了。
|
||||||
|
|
||||||
|
当然有个办法可以解决这个问题;程序可以调用 `setsockopt()` 来打开 `IPV6_V6ONLY` 选项。一个打开两个套接字并且设置了 `IPV6_V6ONLY` 的程序应该可以在所有的系统间移植。
|
||||||
|
|
||||||
|
读者们可能对不是每个程序都能正确处理这一问题没那么震惊。事实证明,这些程序的其中之一是网络时间协议(Network Time Protocol)的 [OpenNTPD][4] 实现。Brent Cook 最近给上游 OpenNTPD 源码[提交了一个小补丁][5],添加了必要的 `setsockopt()` 调用,它也被提交到了 OpenBSD 中了。不过那个补丁看起来不大可能被接受,最可能是因为 OpenBSD 式的理由(LCTT 译注:如前文提到的,OpenBSD 并不受这个问题的影响)。
|
||||||
|
|
||||||
|
### 安全担忧
|
||||||
|
|
||||||
|
正如上文所提到,OpenBSD 根本不支持 IPv4 映射的 IPv6 套接字。即使一个程序试着通过将 `IPV6_V6ONLY` 选项设置为 0 显式地启用地址映射,它的作者会感到沮丧,因为这个设置在 OpenBSD 系统中无效。这个决定背后的原因是这个映射带来了一些安全担忧。攻击打开的接口的攻击类型有很多种,但它们最后都会回到规定的两个途径到达相同的端口,每个端口都有它自己的控制规则。
|
||||||
|
|
||||||
|
任何给定的服务器系统可能都设置了防火墙规则,描述端口的允许访问权限。也许还会有适当的机制,比如 TCP wrappers 或一个基于 BPF 的过滤器,或一个网络上的路由器可以做连接状态协议过滤。结果可能是导致防火墙保护和潜在的所有类型的混乱连接之间的缺口导致同一 IPv4 地址可以通过两个不同的协议到达。如果地址映射是在网络边界完成的,情况甚至会变得更加复杂;参看[这个 2003 年的 RFC 草案][6],它描述了如果映射地址在主机之间传播,一些随之而来的其它攻击场景。
|
||||||
|
|
||||||
|
改变系统和软件正确地处理 IPv4 映射的 IPv6 地址当然可以实现。但那增加了系统的整体复杂度,并且可以确定这个改动没有实际地完整实现到它应该实现的范围内。如同 Theo de Raadt [说的][7]:
|
||||||
|
|
||||||
|
> **有时候人们将一个糟糕的想法放进了 RFC。之后他们发现这个想法是不可能的就将它丢回垃圾箱了。结果就是概念变得如此复杂,每个人都得在管理和编码方面是个全职专家。**
|
||||||
|
|
||||||
|
我们也根本不清楚这些全职专家有多少在实际配置使用 IPv4 映射的 IPv6 地址的系统和网络。
|
||||||
|
|
||||||
|
有人可能会说,尽管 IPv4 映射的 IPv6 地址造成了安全危险,更改一下程序让它在实现了地址映射的系统上关闭地址映射应该没什么危害。但 Theo [认为][8]不应该这么做,有两个理由。第一个是有许多破旧的程序,它们永远不会被修复。而实际的原因是给发行版们施加压力去默认关闭地址映射。正如他说的:“**最终有人会理解这个危害是系统性的,并更改系统默认行为使之‘secure by default’**。”
|
||||||
|
|
||||||
|
### Linux 上的地址映射
|
||||||
|
|
||||||
|
在 Linux 系统,地址映射由一个叫做 `net.ipv6.bindv6only` 的 sysctl 开关控制;它默认设置为 0(启用地址映射)。管理员(或发行版们)可以通过将它设置为 1 来关闭地址映射,但在部署这样一个系统到生产环境之前最好确认软件都能正常工作。一个快速调查显示没有哪个主要发行版改变这个默认值;Debian 在 2009 年的 “squeeze” 中[改变了这个默认值][9],但这个改动破坏了足够多的软件包(比如[任何包含 Java 的程序][10]),[在经过了几次的 Debian 式的讨论之后][11],它恢复到了原来的设置。看上去不少程序依赖于默认启用地址映射。
|
||||||
|
|
||||||
|
OpenBSD 有以“secure by default”的名义打破其核心系统之外的东西的传统;而 Linux 发行版们则更倾向于难以作出这样的改变。所以那些一般不愿意收到他们用户的不满的发行版们,不太可能很快对 bindv6only 的默认设置作出改变。好消息是这个功能作为默认已经很多年了,但很难找到被利用的例子。但是,正如我们都知道的,谁都无法保证这样的利用不可能发生。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://lwn.net/Articles/688462/
|
||||||
|
|
||||||
|
作者:[Jonathan Corbet][a]
|
||||||
|
译者:[alim0x](https://github.com/alim0x)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://lwn.net/
|
||||||
|
[1]: https://www.google.com/intl/en/ipv6/statistics.html
|
||||||
|
[2]: https://tools.ietf.org/html/rfc3493#section-3.7
|
||||||
|
[3]: https://tools.ietf.org/html/rfc2373#page-10
|
||||||
|
[4]: https://github.com/openntpd-portable/
|
||||||
|
[5]: https://lwn.net/Articles/688464/
|
||||||
|
[6]: https://tools.ietf.org/html/draft-itojun-v6ops-v4mapped-harmful-02
|
||||||
|
[7]: https://lwn.net/Articles/688465/
|
||||||
|
[8]: https://lwn.net/Articles/688466/
|
||||||
|
[9]: https://lists.debian.org/debian-devel/2009/10/msg00541.html
|
||||||
|
[10]: https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=560056
|
||||||
|
[11]: https://lists.debian.org/debian-devel/2010/04/msg00099.html
|
@ -1,37 +1,37 @@
|
|||||||
用Python实现Python解释器
|
用 Python 实现 Python 解释器
|
||||||
===
|
===
|
||||||
|
|
||||||
_Allison是Dropbox的工程师,在那里她维护着世界上最大的由Python客户组成的网络。在Dropbox之前,她是Recurse Center的导师, 曾在纽约写作。在北美的PyCon做过关于Python内部机制的演讲,并且她喜欢奇怪的bugs。她的博客地址是[akaptur.com](http://akaptur.com)._
|
_Allison 是 Dropbox 的工程师,在那里她维护着这个世界上最大的 Python 客户端网络之一。在去 Dropbox 之前,她是 Recurse Center 的协调人, 是这个位于纽约的程序员深造机构的作者。她在北美的 PyCon 做过关于 Python 内部机制的演讲,并且她喜欢研究奇怪的 bug。她的博客地址是 [akaptur.com](http://akaptur.com)。_
|
||||||
|
|
||||||
## Introduction
|
### 介绍
|
||||||
|
|
||||||
Byterun是一个用Python实现的Python解释器。随着我在Byterun上的工作,我惊讶并很高兴地的发现,这个Python解释器的基础结构可以满足500行的限制。在这一章我们会搞清楚这个解释器的结构,给你足够的知识探索下去。我们的目标不是向你展示解释器的每个细节---像编程和计算机科学其他有趣的领域一样,你可能会投入几年的时间去搞清楚这个主题。
|
Byterun 是一个用 Python 实现的 Python 解释器。随着我对 Byterun 的开发,我惊喜地的发现,这个 Python 解释器的基础结构用 500 行代码就能实现。在这一章我们会搞清楚这个解释器的结构,给你足够探索下去的背景知识。我们的目标不是向你展示解释器的每个细节---像编程和计算机科学其他有趣的领域一样,你可能会投入几年的时间去深入了解这个主题。
|
||||||
|
|
||||||
Byterun是Ned Batchelder和我完成的,建立在Paul Swartz的工作之上。它的结构和主要的Python实现(CPython)差不多,所以理解Byterun会帮助你理解大多数解释器特别是CPython解释器。(如果你不知道你用的是什么Python,那么很可能它就是CPython)。尽管Byterun很小,但它能执行大多数简单的Python程序。
|
Byterun 是 Ned Batchelder 和我完成的,建立在 Paul Swartz 的工作之上。它的结构和主要的 Python 实现(CPython)差不多,所以理解 Byterun 会帮助你理解大多数解释器,特别是 CPython 解释器。(如果你不知道你用的是什么 Python,那么很可能它就是 CPython)。尽管 Byterun 很小,但它能执行大多数简单的 Python 程序(这一章是基于 Python 3.5 及其之前版本生成的字节码的,在 Python 3.6 中生成的字节码有一些改变)。
|
||||||
|
|
||||||
### A Python Interpreter
|
#### Python 解释器
|
||||||
|
|
||||||
在开始之前,让我们缩小一下“Pyhton解释器”的意思。在讨论Python的时候,“解释器”这个词可以用在很多不同的地方。有的时候解释器指的是REPL,当你在命令行下敲下`python`时所得到的交互式环境。有时候人们会相互替代的使用Python解释器和Python来说明执行Python代码的这一过程。在本章,“解释器”有一个更精确的意思:执行Python程序过程中的最后一步。
|
在开始之前,让我们限定一下“Pyhton 解释器”的意思。在讨论 Python 的时候,“解释器”这个词可以用在很多不同的地方。有的时候解释器指的是 Python REPL,即当你在命令行下敲下 `python` 时所得到的交互式环境。有时候人们会或多或少的互换使用 “Python 解释器”和“Python”来说明从头到尾执行 Python 代码的这一过程。在本章中,“解释器”有一个更精确的意思:Python 程序的执行过程中的最后一步。
|
||||||
|
|
||||||
在解释器接手之前,Python会执行其他3个步骤:词法分析,语法解析和编译。这三步合起来把源代码转换成_code object_,它包含着解释器可以理解的指令。而解释器的工作就是解释code object中的指令。
|
在解释器接手之前,Python 会执行其他 3 个步骤:词法分析,语法解析和编译。这三步合起来把源代码转换成代码对象(code object),它包含着解释器可以理解的指令。而解释器的工作就是解释代码对象中的指令。
|
||||||
|
|
||||||
你可能很奇怪执行Python代码会有编译这一步。Python通常被称为解释型语言,就像Ruby,Perl一样,它们和编译型语言相对,比如C,Rust。然而,这里的术语并不是它看起来的那样精确。大多数解释型语言包括Python,确实会有编译这一步。而Python被称为解释型的原因是相对于编译型语言,它在编译这一步的工作相对较少(解释器做相对多的工作)。在这章后面你会看到,Python的编译器比C语言编译器需要更少的关于程序行为的信息。
|
你可能很奇怪执行 Python 代码会有编译这一步。Python 通常被称为解释型语言,就像 Ruby,Perl 一样,它们和像 C,Rust 这样的编译型语言相对。然而,这个术语并不是它看起来的那样精确。大多数解释型语言包括 Python 在内,确实会有编译这一步。而 Python 被称为解释型的原因是相对于编译型语言,它在编译这一步的工作相对较少(解释器做相对多的工作)。在这章后面你会看到,Python 的编译器比 C 语言编译器需要更少的关于程序行为的信息。
|
||||||
|
|
||||||
### A Python Python Interpreter
|
#### Python 的 Python 解释器
|
||||||
|
|
||||||
Byterun是一个用Python写的Python解释器,这点可能让你感到奇怪,但没有比用C语言写C语言编译器更奇怪。(事实上,广泛使用的gcc编译器就是用C语言本身写的)你可以用几乎的任何语言写一个Python解释器。
|
Byterun 是一个用 Python 写的 Python 解释器,这点可能让你感到奇怪,但没有比用 C 语言写 C 语言编译器更奇怪的了。(事实上,广泛使用的 gcc 编译器就是用 C 语言本身写的)你可以用几乎任何语言写一个 Python 解释器。
|
||||||
|
|
||||||
用Python写Python既有优点又有缺点。最大的缺点就是速度:用Byterun执行代码要比用CPython执行慢的多,CPython解释器是用C语言实现的并做了优化。然而Byterun是为了学习而设计的,所以速度对我们不重要。使用Python最大优点是我们可以*仅仅*实现解释器,而不用担心Python运行时的部分,特别是对象系统。比如当Byterun需要创建一个类时,它就会回退到“真正”的Python。另外一个优势是Byterun很容易理解,部分原因是它是用高级语言写的(Python!)(另外我们不会对解释器做优化 --- 再一次,清晰和简单比速度更重要)
|
用 Python 写 Python 既有优点又有缺点。最大的缺点就是速度:用 Byterun 执行代码要比用 CPython 执行慢的多,CPython 解释器是用 C 语言实现的,并做了认真优化。然而 Byterun 是为了学习而设计的,所以速度对我们不重要。使用 Python 最大优势是我们可以*仅仅*实现解释器,而不用担心 Python 运行时部分,特别是对象系统。比如当 Byterun 需要创建一个类时,它就会回退到“真正”的 Python。另外一个优势是 Byterun 很容易理解,部分原因是它是用人们很容易理解的高级语言写的(Python !)(另外我们不会对解释器做优化 --- 再一次,清晰和简单比速度更重要)
|
||||||
|
|
||||||
## Building an Interpreter
|
### 构建一个解释器
|
||||||
|
|
||||||
在我们考察Byterun代码之前,我们需要一些对解释器结构的高层次视角。Python解释器是如何工作的?
|
在我们考察 Byterun 代码之前,我们需要从高层次对解释器结构有一些了解。Python 解释器是如何工作的?
|
||||||
|
|
||||||
Python解释器是一个_虚拟机_,模拟真实计算机的软件。我们这个虚拟机是栈机器,它用几个栈来完成操作(与之相对的是寄存器机器,它从特定的内存地址读写数据)。
|
Python 解释器是一个虚拟机(virtual machine),是一个模拟真实计算机的软件。我们这个虚拟机是栈机器(stack machine),它用几个栈来完成操作(与之相对的是寄存器机器(register machine),它从特定的内存地址读写数据)。
|
||||||
|
|
||||||
Python解释器是一个_字节码解释器_:它的输入是一些命令集合称作_字节码_。当你写Python代码时,词法分析器,语法解析器和编译器生成code object让解释器去操作。每个code object都包含一个要被执行的指令集合 --- 它就是字节码 --- 另外还有一些解释器需要的信息。字节码是Python代码的一个_中间层表示_:它以一种解释器可以理解的方式来表示源代码。这和汇编语言作为C语言和机器语言的中间表示很类似。
|
Python 解释器是一个字节码解释器(bytecode interpreter):它的输入是一些称作字节码(bytecode)的指令集。当你写 Python 代码时,词法分析器、语法解析器和编译器会生成代码对象(code object)让解释器去操作。每个代码对象都包含一个要被执行的指令集 —— 它就是字节码 —— 以及还有一些解释器需要的信息。字节码是 Python 代码的一个中间层表示( intermediate representation):它以一种解释器可以理解的方式来表示源代码。这和汇编语言作为 C 语言和机器语言的中间表示很类似。
|
||||||
|
|
||||||
### A Tiny Interpreter
|
#### 微型解释器
|
||||||
|
|
||||||
为了让说明更具体,让我们从一个非常小的解释器开始。它只能计算两个数的和,只能理解三个指令。它执行的所有代码只是这三个指令的不同组合。下面就是这三个指令:
|
为了让说明更具体,让我们从一个非常小的解释器开始。它只能计算两个数的和,只能理解三个指令。它执行的所有代码只是这三个指令的不同组合。下面就是这三个指令:
|
||||||
|
|
||||||
@ -39,7 +39,7 @@ Python解释器是一个_字节码解释器_:它的输入是一些命令集合
|
|||||||
- `ADD_TWO_VALUES`
|
- `ADD_TWO_VALUES`
|
||||||
- `PRINT_ANSWER`
|
- `PRINT_ANSWER`
|
||||||
|
|
||||||
我们不关心词法,语法和编译,所以我们也不在乎这些指令是如何产生的。你可以想象,你写下`7 + 5`,然后一个编译器为你生成那三个指令的组合。如果你有一个合适的编译器,你甚至可以用Lisp的语法来写,只要它能生成相同的指令。
|
我们不关心词法、语法和编译,所以我们也不在乎这些指令集是如何产生的。你可以想象,当你写下 `7 + 5`,然后一个编译器为你生成那三个指令的组合。如果你有一个合适的编译器,你甚至可以用 Lisp 的语法来写,只要它能生成相同的指令。
|
||||||
|
|
||||||
假设
|
假设
|
||||||
|
|
||||||
@ -58,13 +58,13 @@ what_to_execute = {
|
|||||||
"numbers": [7, 5] }
|
"numbers": [7, 5] }
|
||||||
```
|
```
|
||||||
|
|
||||||
Python解释器是一个_栈机器_,所以它必须通过操作栈来完成这个加法。(\aosafigref{500l.interpreter.stackmachine}.)解释器先执行第一条指令,`LOAD_VALUE`,把第一个数压到栈中。接着它把第二个数也压到栈中。然后,第三条指令,`ADD_TWO_VALUES`,先把两个数从栈中弹出,加起来,再把结果压入栈中。最后一步,把结果弹出并输出。
|
Python 解释器是一个栈机器(stack machine),所以它必须通过操作栈来完成这个加法(见下图)。解释器先执行第一条指令,`LOAD_VALUE`,把第一个数压到栈中。接着它把第二个数也压到栈中。然后,第三条指令,`ADD_TWO_VALUES`,先把两个数从栈中弹出,加起来,再把结果压入栈中。最后一步,把结果弹出并输出。
|
||||||
|
|
||||||
\aosafigure[240pt]{interpreter-images/interpreter-stack.png}{A stack machine}{500l.interpreter.stackmachine}
|
![栈机器](http://aosabook.org/en/500L/interpreter-images/interpreter-stack.png)
|
||||||
|
|
||||||
`LOAD_VALUE`这条指令告诉解释器把一个数压入栈中,但指令本身并没有指明这个数是多少。指令需要一个额外的信息告诉解释器去哪里找到这个数。所以我们的指令集有两个部分:指令本身和一个常量列表。(在Python中,字节码就是我们称为的“指令”,而解释器执行的是_code object_。)
|
`LOAD_VALUE`这条指令告诉解释器把一个数压入栈中,但指令本身并没有指明这个数是多少。指令需要一个额外的信息告诉解释器去哪里找到这个数。所以我们的指令集有两个部分:指令本身和一个常量列表。(在 Python 中,字节码就是我们所称的“指令”,而解释器“执行”的是代码对象。)
|
||||||
|
|
||||||
为什么不把数字直接嵌入指令之中?想象一下,如果我们加的不是数字,而是字符串。我们可不想把字符串这样的东西加到指令中,因为它可以有任意的长度。另外,我们这种设计也意味着我们只需要对象的一份拷贝,比如这个加法 `7 + 7`, 现在常量表 `"numbers"`只需包含一个`7`。
|
为什么不把数字直接嵌入指令之中?想象一下,如果我们加的不是数字,而是字符串。我们可不想把字符串这样的东西加到指令中,因为它可以有任意的长度。另外,我们这种设计也意味着我们只需要对象的一份拷贝,比如这个加法 `7 + 7`, 现在常量表 `"numbers"`只需包含一个`[7]`。
|
||||||
|
|
||||||
你可能会想为什么会需要除了`ADD_TWO_VALUES`之外的指令。的确,对于我们两个数加法,这个例子是有点人为制作的意思。然而,这个指令却是建造更复杂程序的轮子。比如,就我们目前定义的三个指令,只要给出正确的指令组合,我们可以做三个数的加法,或者任意个数的加法。同时,栈提供了一个清晰的方法去跟踪解释器的状态,这为我们增长的复杂性提供了支持。
|
你可能会想为什么会需要除了`ADD_TWO_VALUES`之外的指令。的确,对于我们两个数加法,这个例子是有点人为制作的意思。然而,这个指令却是建造更复杂程序的轮子。比如,就我们目前定义的三个指令,只要给出正确的指令组合,我们可以做三个数的加法,或者任意个数的加法。同时,栈提供了一个清晰的方法去跟踪解释器的状态,这为我们增长的复杂性提供了支持。
|
||||||
|
|
||||||
@ -89,7 +89,7 @@ class Interpreter:
|
|||||||
self.stack.append(total)
|
self.stack.append(total)
|
||||||
```
|
```
|
||||||
|
|
||||||
这三个方法完成了解释器所理解的三条指令。但解释器还需要一样东西:一个能把所有东西结合在一起并执行的方法。这个方法就叫做`run_code`, 它把我们前面定义的字典结构`what-to-execute`作为参数,循环执行里面的每条指令,如何指令有参数,处理参数,然后调用解释器对象中相应的方法。
|
这三个方法完成了解释器所理解的三条指令。但解释器还需要一样东西:一个能把所有东西结合在一起并执行的方法。这个方法就叫做 `run_code`,它把我们前面定义的字典结构 `what-to-execute` 作为参数,循环执行里面的每条指令,如果指令有参数就处理参数,然后调用解释器对象中相应的方法。
|
||||||
|
|
||||||
```python
|
```python
|
||||||
def run_code(self, what_to_execute):
|
def run_code(self, what_to_execute):
|
||||||
@ -106,20 +106,20 @@ class Interpreter:
|
|||||||
self.PRINT_ANSWER()
|
self.PRINT_ANSWER()
|
||||||
```
|
```
|
||||||
|
|
||||||
为了测试,我们创建一个解释器对象,然后用前面定义的 7 + 5 的指令集来调用`run_code`。
|
为了测试,我们创建一个解释器对象,然后用前面定义的 7 + 5 的指令集来调用 `run_code`。
|
||||||
|
|
||||||
```python
|
```python
|
||||||
interpreter = Interpreter()
|
interpreter = Interpreter()
|
||||||
interpreter.run_code(what_to_execute)
|
interpreter.run_code(what_to_execute)
|
||||||
```
|
```
|
||||||
|
|
||||||
显然,它会输出12
|
显然,它会输出12。
|
||||||
|
|
||||||
尽管我们的解释器功能受限,但这个加法过程几乎和真正的Python解释器是一样的。这里,我们还有几点要注意。
|
尽管我们的解释器功能十分受限,但这个过程几乎和真正的 Python 解释器处理加法是一样的。这里,我们还有几点要注意。
|
||||||
|
|
||||||
首先,一些指令需要参数。在真正的Python bytecode中,大概有一半的指令有参数。像我们的例子一样,参数和指令打包在一起。注意_指令_的参数和传递给对应方法的参数是不同的。
|
首先,一些指令需要参数。在真正的 Python 字节码当中,大概有一半的指令有参数。像我们的例子一样,参数和指令打包在一起。注意指令的参数和传递给对应方法的参数是不同的。
|
||||||
|
|
||||||
第二,指令`ADD_TWO_VALUES`不需要任何参数,它从解释器栈中弹出所需的值。这正是以栈为基础的解释器的特点。
|
第二,指令`ADD_TWO_VALUES`不需要任何参数,它从解释器栈中弹出所需的值。这正是以基于栈的解释器的特点。
|
||||||
|
|
||||||
记得我们说过只要给出合适的指令集,不需要对解释器做任何改变,我们就能做多个数的加法。考虑下面的指令集,你觉得会发生什么?如果你有一个合适的编译器,什么代码才能编译出下面的指令集?
|
记得我们说过只要给出合适的指令集,不需要对解释器做任何改变,我们就能做多个数的加法。考虑下面的指令集,你觉得会发生什么?如果你有一个合适的编译器,什么代码才能编译出下面的指令集?
|
||||||
|
|
||||||
@ -134,11 +134,11 @@ class Interpreter:
|
|||||||
"numbers": [7, 5, 8] }
|
"numbers": [7, 5, 8] }
|
||||||
```
|
```
|
||||||
|
|
||||||
从这点出发,我们开始看到这种结构的可扩展性:我们可以通过向解释器对象增加方法来描述更多的操作(只要有一个编译器能为我们生成组织良好的指令集)。
|
从这点出发,我们开始看到这种结构的可扩展性:我们可以通过向解释器对象增加方法来描述更多的操作(只要有一个编译器能为我们生成组织良好的指令集就行)。
|
||||||
|
|
||||||
#### Variables
|
##### 变量
|
||||||
|
|
||||||
接下来给我们的解释器增加变量的支持。我们需要一个保存变量值的指令,`STORE_NAME`;一个取变量值的指令`LOAD_NAME`;和一个变量到值的映射关系。目前,我们会忽略命名空间和作用域,所以我们可以把变量和值的映射直接存储在解释器对象中。最后,我们要保证`what_to_execute`除了一个常量列表,还要有个变量名字的列表。
|
接下来给我们的解释器增加变量的支持。我们需要一个保存变量值的指令 `STORE_NAME`;一个取变量值的指令`LOAD_NAME`;和一个变量到值的映射关系。目前,我们会忽略命名空间和作用域,所以我们可以把变量和值的映射直接存储在解释器对象中。最后,我们要保证`what_to_execute`除了一个常量列表,还要有个变量名字的列表。
|
||||||
|
|
||||||
```python
|
```python
|
||||||
>>> def s():
|
>>> def s():
|
||||||
@ -159,9 +159,9 @@ class Interpreter:
|
|||||||
"names": ["a", "b"] }
|
"names": ["a", "b"] }
|
||||||
```
|
```
|
||||||
|
|
||||||
我们的新的的实现在下面。为了跟踪哪名字绑定到那个值,我们在`__init__`方法中增加一个`environment`字典。我们也增加了`STORE_NAME`和`LOAD_NAME`方法,它们获得变量名,然后从`environment`字典中设置或取出这个变量值。
|
我们的新的实现在下面。为了跟踪哪个名字绑定到哪个值,我们在`__init__`方法中增加一个`environment`字典。我们也增加了`STORE_NAME`和`LOAD_NAME`方法,它们获得变量名,然后从`environment`字典中设置或取出这个变量值。
|
||||||
|
|
||||||
现在指令参数就有两个不同的意思,它可能是`numbers`列表的索引,也可能是`names`列表的索引。解释器通过检查所执行的指令就能知道是那种参数。而我们打破这种逻辑 ,把指令和它所用何种参数的映射关系放在另一个单独的方法中。
|
现在指令的参数就有两个不同的意思,它可能是`numbers`列表的索引,也可能是`names`列表的索引。解释器通过检查所执行的指令就能知道是那种参数。而我们打破这种逻辑 ,把指令和它所用何种参数的映射关系放在另一个单独的方法中。
|
||||||
|
|
||||||
```python
|
```python
|
||||||
class Interpreter:
|
class Interpreter:
|
||||||
@ -207,7 +207,7 @@ class Interpreter:
|
|||||||
self.LOAD_NAME(argument)
|
self.LOAD_NAME(argument)
|
||||||
```
|
```
|
||||||
|
|
||||||
仅仅五个指令,`run_code`这个方法已经开始变得冗长了。如果保持这种结构,那么每条指令都需要一个`if`分支。这里,我们要利用Python的动态方法查找。我们总会给一个称为`FOO`的指令定义一个名为`FOO`的方法,这样我们就可用Python的`getattr`函数在运行时动态查找方法,而不用这个大大的分支结构。`run_code`方法现在是这样:
|
仅仅五个指令,`run_code`这个方法已经开始变得冗长了。如果保持这种结构,那么每条指令都需要一个`if`分支。这里,我们要利用 Python 的动态方法查找。我们总会给一个称为`FOO`的指令定义一个名为`FOO`的方法,这样我们就可用 Python 的`getattr`函数在运行时动态查找方法,而不用这个大大的分支结构。`run_code`方法现在是这样:
|
||||||
|
|
||||||
```python
|
```python
|
||||||
def execute(self, what_to_execute):
|
def execute(self, what_to_execute):
|
||||||
@ -222,9 +222,9 @@ class Interpreter:
|
|||||||
bytecode_method(argument)
|
bytecode_method(argument)
|
||||||
```
|
```
|
||||||
|
|
||||||
## Real Python Bytecode
|
### 真实的 Python 字节码
|
||||||
|
|
||||||
现在,放弃我们的小指令集,去看看真正的Python字节码。字节码的结构和我们的小解释器的指令集差不多,除了字节码用一个字节而不是一个名字来指示这条指令。为了理解它的结构,我们将考察一个函数的字节码。考虑下面这个例子:
|
现在,放弃我们的小指令集,去看看真正的 Python 字节码。字节码的结构和我们的小解释器的指令集差不多,除了字节码用一个字节而不是一个名字来代表这条指令。为了理解它的结构,我们将考察一个函数的字节码。考虑下面这个例子:
|
||||||
|
|
||||||
```python
|
```python
|
||||||
>>> def cond():
|
>>> def cond():
|
||||||
@ -236,7 +236,7 @@ class Interpreter:
|
|||||||
...
|
...
|
||||||
```
|
```
|
||||||
|
|
||||||
Python在运行时会暴露一大批内部信息,并且我们可以通过REPL直接访问这些信息。对于函数对象`cond`,`cond.__code__`是与其关联的code object,而`cond.__code__.co_code`就是它的字节码。当你写Python代码时,你永远也不会想直接使用这些属性,但是这可以让我们做出各种恶作剧,同时也可以看看内部机制。
|
Python 在运行时会暴露一大批内部信息,并且我们可以通过 REPL 直接访问这些信息。对于函数对象`cond`,`cond.__code__`是与其关联的代码对象,而`cond.__code__.co_code`就是它的字节码。当你写 Python 代码时,你永远也不会想直接使用这些属性,但是这可以让我们做出各种恶作剧,同时也可以看看内部机制。
|
||||||
|
|
||||||
```python
|
```python
|
||||||
>>> cond.__code__.co_code # the bytecode as raw bytes
|
>>> cond.__code__.co_code # the bytecode as raw bytes
|
||||||
@ -247,9 +247,9 @@ b'd\x01\x00}\x00\x00|\x00\x00d\x02\x00k\x00\x00r\x16\x00d\x03\x00Sd\x04\x00Sd\x0
|
|||||||
100, 4, 0, 83, 100, 0, 0, 83]
|
100, 4, 0, 83, 100, 0, 0, 83]
|
||||||
```
|
```
|
||||||
|
|
||||||
当我们直接输出这个字节码,它看起来完全无法理解 --- 唯一我们了解的是它是一串字节。很幸运,我们有一个很强大的工具可以用:Python标准库中的`dis`模块。
|
当我们直接输出这个字节码,它看起来完全无法理解 —— 唯一我们了解的是它是一串字节。很幸运,我们有一个很强大的工具可以用:Python 标准库中的`dis`模块。
|
||||||
|
|
||||||
`dis`是一个字节码反汇编器。反汇编器以为机器而写的底层代码作为输入,比如汇编代码和字节码,然后以人类可读的方式输出。当我们运行`dis.dis`, 它输出每个字节码的解释。
|
`dis`是一个字节码反汇编器。反汇编器以为机器而写的底层代码作为输入,比如汇编代码和字节码,然后以人类可读的方式输出。当我们运行`dis.dis`,它输出每个字节码的解释。
|
||||||
|
|
||||||
```python
|
```python
|
||||||
>>> dis.dis(cond)
|
>>> dis.dis(cond)
|
||||||
@ -270,9 +270,9 @@ b'd\x01\x00}\x00\x00|\x00\x00d\x02\x00k\x00\x00r\x16\x00d\x03\x00Sd\x04\x00Sd\x0
|
|||||||
29 RETURN_VALUE
|
29 RETURN_VALUE
|
||||||
```
|
```
|
||||||
|
|
||||||
这些都是什么意思?让我们以第一条指令`LOAD_CONST`为例子。第一列的数字(`2`)表示对应源代码的行数。第二列的数字是字节码的索引,告诉我们指令`LOAD_CONST`在0位置。第三列是指令本身对应的人类可读的名字。如果第四列存在,它表示指令的参数。如果第5列存在,它是一个关于参数是什么的提示。
|
这些都是什么意思?让我们以第一条指令`LOAD_CONST`为例子。第一列的数字(`2`)表示对应源代码的行数。第二列的数字是字节码的索引,告诉我们指令`LOAD_CONST`在位置 0 。第三列是指令本身对应的人类可读的名字。如果第四列存在,它表示指令的参数。如果第五列存在,它是一个关于参数是什么的提示。
|
||||||
|
|
||||||
考虑这个字节码的前几个字节:[100, 1, 0, 125, 0, 0]。这6个字节表示两条带参数的指令。我们可以使用`dis.opname`,一个字节到可读字符串的映射,来找到指令100和指令125代表是什么:
|
考虑这个字节码的前几个字节:[100, 1, 0, 125, 0, 0]。这 6 个字节表示两条带参数的指令。我们可以使用`dis.opname`,一个字节到可读字符串的映射,来找到指令 100 和指令 125 代表的是什么:
|
||||||
|
|
||||||
```python
|
```python
|
||||||
>>> dis.opname[100]
|
>>> dis.opname[100]
|
||||||
@ -281,11 +281,11 @@ b'd\x01\x00}\x00\x00|\x00\x00d\x02\x00k\x00\x00r\x16\x00d\x03\x00Sd\x04\x00Sd\x0
|
|||||||
'STORE_FAST'
|
'STORE_FAST'
|
||||||
```
|
```
|
||||||
|
|
||||||
第二和第三个字节 --- 1 ,0 ---是`LOAD_CONST`的参数,第五和第六个字节 --- 0,0 --- 是`STORE_FAST`的参数。就像我们前面的小例子,`LOAD_CONST`需要知道的到哪去找常量,`STORE_FAST`需要找到名字。(Python的`LOAD_CONST`和我们小例子中的`LOAD_VALUE`一样,`LOAD_FAST`和`LOAD_NAME`一样)。所以这六个字节代表第一行源代码`x = 3`.(为什么用两个字节表示指令的参数?如果Python使用一个字节,每个code object你只能有256个常量/名字,而用两个字节,就增加到了256的平方,65536个)。
|
第二和第三个字节 —— 1 、0 ——是`LOAD_CONST`的参数,第五和第六个字节 —— 0、0 —— 是`STORE_FAST`的参数。就像我们前面的小例子,`LOAD_CONST`需要知道的到哪去找常量,`STORE_FAST`需要知道要存储的名字。(Python 的`LOAD_CONST`和我们小例子中的`LOAD_VALUE`一样,`LOAD_FAST`和`LOAD_NAME`一样)。所以这六个字节代表第一行源代码`x = 3` (为什么用两个字节表示指令的参数?如果 Python 使用一个字节,每个代码对象你只能有 256 个常量/名字,而用两个字节,就增加到了 256 的平方,65536个)。
|
||||||
|
|
||||||
### Conditionals and Loops
|
#### 条件语句与循环语句
|
||||||
|
|
||||||
到目前为止,我们的解释器只能一条接着一条的执行指令。这有个问题,我们经常会想多次执行某个指令,或者在特定的条件下跳过它们。为了可以写循环和分支结构,解释器必须能够在指令中跳转。在某种程度上,Python在字节码中使用`GOTO`语句来处理循环和分支!让我们再看一个`cond`函数的反汇编结果:
|
到目前为止,我们的解释器只能一条接着一条的执行指令。这有个问题,我们经常会想多次执行某个指令,或者在特定的条件下跳过它们。为了可以写循环和分支结构,解释器必须能够在指令中跳转。在某种程度上,Python 在字节码中使用`GOTO`语句来处理循环和分支!让我们再看一个`cond`函数的反汇编结果:
|
||||||
|
|
||||||
```python
|
```python
|
||||||
>>> dis.dis(cond)
|
>>> dis.dis(cond)
|
||||||
@ -306,11 +306,11 @@ b'd\x01\x00}\x00\x00|\x00\x00d\x02\x00k\x00\x00r\x16\x00d\x03\x00Sd\x04\x00Sd\x0
|
|||||||
29 RETURN_VALUE
|
29 RETURN_VALUE
|
||||||
```
|
```
|
||||||
|
|
||||||
第三行的条件表达式`if x < 5`被编译成四条指令:`LOAD_FAST`, `LOAD_CONST`, `COMPARE_OP`和 `POP_JUMP_IF_FALSE`。`x < 5`对应加载`x`,加载5,比较这两个值。指令`POP_JUMP_IF_FALSE`完成`if`语句。这条指令把栈顶的值弹出,如果值为真,什么都不发生。如果值为假,解释器会跳转到另一条指令。
|
第三行的条件表达式`if x < 5`被编译成四条指令:`LOAD_FAST`、 `LOAD_CONST`、 `COMPARE_OP`和 `POP_JUMP_IF_FALSE`。`x < 5`对应加载`x`、加载 5、比较这两个值。指令`POP_JUMP_IF_FALSE`完成这个`if`语句。这条指令把栈顶的值弹出,如果值为真,什么都不发生。如果值为假,解释器会跳转到另一条指令。
|
||||||
|
|
||||||
这条将被加载的指令称为跳转目标,它作为指令`POP_JUMP`的参数。这里,跳转目标是22,索引为22的指令是`LOAD_CONST`,对应源码的第6行。(`dis`用`>>`标记跳转目标。)如果`X < 5`为假,解释器会忽略第四行(`return yes`),直接跳转到第6行(`return "no"`)。因此解释器通过跳转指令选择性的执行指令。
|
这条将被加载的指令称为跳转目标,它作为指令`POP_JUMP`的参数。这里,跳转目标是 22,索引为 22 的指令是`LOAD_CONST`,对应源码的第 6 行。(`dis`用`>>`标记跳转目标。)如果`X < 5`为假,解释器会忽略第四行(`return yes`),直接跳转到第6行(`return "no"`)。因此解释器通过跳转指令选择性的执行指令。
|
||||||
|
|
||||||
Python的循环也依赖于跳转。在下面的字节码中,`while x < 5`这一行产生了和`if x < 10`几乎一样的字节码。在这两种情况下,解释器都是先执行比较,然后执行`POP_JUMP_IF_FALSE`来控制下一条执行哪个指令。第四行的最后一条字节码`JUMP_ABSOLUT`(循环体结束的地方),让解释器返回到循环开始的第9条指令处。当 `x < 10`变为假,`POP_JUMP_IF_FALSE`会让解释器跳到循环的终止处,第34条指令。
|
Python 的循环也依赖于跳转。在下面的字节码中,`while x < 5`这一行产生了和`if x < 10`几乎一样的字节码。在这两种情况下,解释器都是先执行比较,然后执行`POP_JUMP_IF_FALSE`来控制下一条执行哪个指令。第四行的最后一条字节码`JUMP_ABSOLUT`(循环体结束的地方),让解释器返回到循环开始的第 9 条指令处。当 `x < 10`变为假,`POP_JUMP_IF_FALSE`会让解释器跳到循环的终止处,第 34 条指令。
|
||||||
|
|
||||||
```python
|
```python
|
||||||
>>> def loop():
|
>>> def loop():
|
||||||
@ -340,23 +340,23 @@ Python的循环也依赖于跳转。在下面的字节码中,`while x < 5`这
|
|||||||
38 RETURN_VALUE
|
38 RETURN_VALUE
|
||||||
```
|
```
|
||||||
|
|
||||||
### Explore Bytecode
|
#### 探索字节码
|
||||||
|
|
||||||
我希望你用`dis.dis`来试试你自己写的函数。一些有趣的问题值得探索:
|
我希望你用`dis.dis`来试试你自己写的函数。一些有趣的问题值得探索:
|
||||||
|
|
||||||
- 对解释器而言for循环和while循环有什么不同?
|
- 对解释器而言 for 循环和 while 循环有什么不同?
|
||||||
- 能不能写出两个不同函数,却能产生相同的字节码?
|
- 能不能写出两个不同函数,却能产生相同的字节码?
|
||||||
- `elif`是怎么工作的?列表推导呢?
|
- `elif`是怎么工作的?列表推导呢?
|
||||||
|
|
||||||
## Frames
|
### 帧
|
||||||
|
|
||||||
到目前为止,我们已经知道了Python虚拟机是一个栈机器。它能顺序执行指令,在指令间跳转,压入或弹出栈值。但是这和我们期望的解释器还有一定距离。在前面的那个例子中,最后一条指令是`RETURN_VALUE`,它和`return`语句相对应。但是它返回到哪里去呢?
|
到目前为止,我们已经知道了 Python 虚拟机是一个栈机器。它能顺序执行指令,在指令间跳转,压入或弹出栈值。但是这和我们期望的解释器还有一定距离。在前面的那个例子中,最后一条指令是`RETURN_VALUE`,它和`return`语句相对应。但是它返回到哪里去呢?
|
||||||
|
|
||||||
为了回答这个问题,我们必须再增加一层复杂性:frame。一个frame是一些信息的集合和代码的执行上下文。frames在Python代码执行时动态的创建和销毁。每个frame对应函数的一次调用。--- 所以每个frame只有一个code object与之关联,而一个code object可以有多个frame。比如你有一个函数递归的调用自己10次,这会产生11个frame,每次调用对应一个,再加上启动模块对应的一个frame。总的来说,Python程序的每个作用域有一个frame,比如,模块,函数,类。
|
为了回答这个问题,我们必须再增加一层复杂性:帧(frame)。一个帧是一些信息的集合和代码的执行上下文。帧在 Python 代码执行时动态地创建和销毁。每个帧对应函数的一次调用 —— 所以每个帧只有一个代码对象与之关联,而一个代码对象可以有多个帧。比如你有一个函数递归的调用自己 10 次,这会产生 11 个帧,每次调用对应一个,再加上启动模块对应的一个帧。总的来说,Python 程序的每个作用域都有一个帧,比如,模块、函数、类定义。
|
||||||
|
|
||||||
Frame存在于_调用栈_中,一个和我们之前讨论的完全不同的栈。(你最熟悉的栈就是调用栈,就是你经常看到的异常回溯,每个以"File 'program.py'"开始的回溯对应一个frame。)解释器在执行字节码时操作的栈,我们叫它_数据栈_。其实还有第三个栈,叫做_块栈_,用于特定的控制流块,比如循环和异常处理。调用栈中的每个frame都有它自己的数据栈和块栈。
|
帧存在于调用栈(call stack)中,一个和我们之前讨论的完全不同的栈。(你最熟悉的栈就是调用栈,就是你经常看到的异常回溯,每个以"File 'program.py'"开始的回溯对应一个帧。)解释器在执行字节码时操作的栈,我们叫它数据栈(data stack)。其实还有第三个栈,叫做块栈(block stack),用于特定的控制流块,比如循环和异常处理。调用栈中的每个帧都有它自己的数据栈和块栈。
|
||||||
|
|
||||||
让我们用一个具体的例子来说明。假设Python解释器执行到标记为3的地方。解释器正在`foo`函数的调用中,它接着调用`bar`。下面是frame调用栈,块栈和数据栈的示意图。我们感兴趣的是解释器先从最底下的`foo()`开始,接着执行`foo`的函数体,然后到达`bar`。
|
让我们用一个具体的例子来说明一下。假设 Python 解释器执行到下面标记为 3 的地方。解释器正处于`foo`函数的调用中,它接着调用`bar`。下面是帧调用栈、块栈和数据栈的示意图。我们感兴趣的是解释器先从最底下的`foo()`开始,接着执行`foo`的函数体,然后到达`bar`。
|
||||||
|
|
||||||
```python
|
```python
|
||||||
>>> def bar(y):
|
>>> def bar(y):
|
||||||
@ -372,32 +372,30 @@ Frame存在于_调用栈_中,一个和我们之前讨论的完全不同的栈
|
|||||||
3
|
3
|
||||||
```
|
```
|
||||||
|
|
||||||
\aosafigure[240pt]{interpreter-images/interpreter-callstack.png}{The call stack}{500l.interpreter.callstack}
|
![调用栈](http://aosabook.org/en/500L/interpreter-images/interpreter-callstack.png)
|
||||||
|
|
||||||
现在,解释器在`bar`函数的调用中。调用栈中有3个frame:一个对应于模块层,一个对应函数`foo`,别一个对应函数`bar`。(\aosafigref{500l.interpreter.callstack}.)一旦`bar`返回,与它对应的frame就会从调用栈中弹出并丢弃。
|
现在,解释器处于`bar`函数的调用中。调用栈中有 3 个帧:一个对应于模块层,一个对应函数`foo`,另一个对应函数`bar`。(见上图)一旦`bar`返回,与它对应的帧就会从调用栈中弹出并丢弃。
|
||||||
|
|
||||||
字节码指令`RETURN_VALUE`告诉解释器在frame间传递一个值。首先,它把位于调用栈栈顶的frame中的数据栈的栈顶值弹出。然后把整个frame弹出丢弃。最后把这个值压到下一个frame的数据栈中。
|
字节码指令`RETURN_VALUE`告诉解释器在帧之间传递一个值。首先,它把位于调用栈栈顶的帧中的数据栈的栈顶值弹出。然后把整个帧弹出丢弃。最后把这个值压到下一个帧的数据栈中。
|
||||||
|
|
||||||
当Ned Batchelder和我在写Byterun时,很长一段时间我们的实现中一直有个重大的错误。我们整个虚拟机中只有一个数据栈,而不是每个frame都有一个。我们写了很多测试代码,同时在Byterun和真正的Python上运行,希望得到一致结果。我们几乎通过了所有测试,只有一样东西不能通过,那就是生成器。最后,通过仔细的阅读CPython的源码,我们发现了错误所在[^thanks]。把数据栈移到每个frame就解决了这个问题。
|
当 Ned Batchelder 和我在写 Byterun 时,很长一段时间我们的实现中一直有个重大的错误。我们整个虚拟机中只有一个数据栈,而不是每个帧都有一个。我们写了很多测试代码,同时在 Byterun 和真正的 Python 上运行,希望得到一致结果。我们几乎通过了所有测试,只有一样东西不能通过,那就是生成器(generators)。最后,通过仔细的阅读 CPython 的源码,我们发现了错误所在(感谢 Michael Arntzenius 对这个 bug 的洞悉)。把数据栈移到每个帧就解决了这个问题。
|
||||||
|
|
||||||
[^thanks]: 感谢 Michael Arntzenius 对这个bug的洞悉。
|
回头在看看这个 bug,我惊讶的发现 Python 真的很少依赖于每个帧有一个数据栈这个特性。在 Python 中几乎所有的操作都会清空数据栈,所以所有的帧公用一个数据栈是没问题的。在上面的例子中,当`bar`执行完后,它的数据栈为空。即使`foo`公用这一个栈,它的值也不会受影响。然而,对应生成器,它的一个关键的特点是它能暂停一个帧的执行,返回到其他的帧,一段时间后它能返回到原来的帧,并以它离开时的相同状态继续执行。
|
||||||
|
|
||||||
回头在看看这个bug,我惊讶的发现Python真的很少依赖于每个frame有一个数据栈这个特性。在Python中几乎所有的操作都会清空数据栈,所以所有的frame公用一个数据栈是没问题的。在上面的例子中,当`bar`执行完后,它的数据栈为空。即使`foo`公用这一个栈,它的值也不会受影响。然而,对应生成器,一个关键的特点是它能暂停一个frame的执行,返回到其他的frame,一段时间后它能返回到原来的frame,并以它离开时的相同状态继续执行。
|
### Byterun
|
||||||
|
|
||||||
## Byterun
|
现在我们有足够的 Python 解释器的知识背景去考察 Byterun。
|
||||||
|
|
||||||
现在我们有足够的Python解释器的知识背景去考察Byterun。
|
Byterun 中有四种对象。
|
||||||
|
|
||||||
Byterun中有四种对象。
|
- `VirtualMachine`类,它管理高层结构,尤其是帧调用栈,并包含了指令到操作的映射。这是一个比前面`Inteprter`对象更复杂的版本。
|
||||||
|
- `Frame`类,每个`Frame`类都有一个代码对象,并且管理着其他一些必要的状态位,尤其是全局和局部命名空间、指向调用它的整的指针和最后执行的字节码指令。
|
||||||
|
- `Function`类,它被用来代替真正的 Python 函数。回想一下,调用函数时会创建一个新的帧。我们自己实现了`Function`,以便我们控制新的`Frame`的创建。
|
||||||
|
- `Block`类,它只是包装了块的 3 个属性。(块的细节不是解释器的核心,我们不会花时间在它身上,把它列在这里,是因为 Byterun 需要它。)
|
||||||
|
|
||||||
- `VirtualMachine`类,它管理高层结构,frame调用栈,指令到操作的映射。这是一个比前面`Inteprter`对象更复杂的版本。
|
#### `VirtualMachine` 类
|
||||||
- `Frame`类,每个`Frame`类都有一个code object,并且管理者其他一些必要的状态信息,全局和局部命名空间,指向调用它的frame的指针和最后执行的字节码指令。
|
|
||||||
- `Function`类,它被用来代替真正的Python函数。回想一下,调用函数时会创建一个新的frame。我们自己实现`Function`,所以我们控制新frame的创建。
|
|
||||||
- `Block`类,它只是包装了block的3个属性。(block的细节不是解释器的核心,我们不会花时间在它身上,把它列在这里,是因为Byterun需要它。)
|
|
||||||
|
|
||||||
### The `VirtualMachine` Class
|
每次程序运行时只会创建一个`VirtualMachine`实例,因为我们只有一个 Python 解释器。`VirtualMachine` 保存调用栈、异常状态、在帧之间传递的返回值。它的入口点是`run_code`方法,它以编译后的代码对象为参数,以创建一个帧为开始,然后运行这个帧。这个帧可能再创建出新的帧;调用栈随着程序的运行而增长和缩短。当第一个帧返回时,执行结束。
|
||||||
|
|
||||||
程序运行时只有一个`VirtualMachine`被创建,因为我们只有一个解释器。`VirtualMachine`保存调用栈,异常状态,在frame中传递的返回值。它的入口点是`run_code`方法,它以编译后的code object为参数,以创建一个frame为开始,然后运行这个frame。这个frame可能再创建出新的frame;调用栈随着程序的运行增长缩短。当第一个frame返回时,执行结束。
|
|
||||||
|
|
||||||
```python
|
```python
|
||||||
class VirtualMachineError(Exception):
|
class VirtualMachineError(Exception):
|
||||||
@ -418,9 +416,9 @@ class VirtualMachine(object):
|
|||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### The `Frame` Class
|
#### `Frame` 类
|
||||||
|
|
||||||
接下来,我们来写`Frame`对象。frame是一个属性的集合,它没有任何方法。前面提到过,这些属性包括由编译器生成的code object;局部,全局和内置命名空间;前一个frame的引用;一个数据栈;一个块栈;最后执行的指令指针。(对于内置命名空间我们需要多做一点工作,Python在不同模块中对这个命名空间有不同的处理;但这个细节对我们的虚拟机不重要。)
|
接下来,我们来写`Frame`对象。帧是一个属性的集合,它没有任何方法。前面提到过,这些属性包括由编译器生成的代码对象;局部、全局和内置命名空间;前一个帧的引用;一个数据栈;一个块栈;最后执行的指令指针。(对于内置命名空间我们需要多做一点工作,Python 在不同模块中对这个命名空间有不同的处理;但这个细节对我们的虚拟机不重要。)
|
||||||
|
|
||||||
```python
|
```python
|
||||||
class Frame(object):
|
class Frame(object):
|
||||||
@ -441,7 +439,7 @@ class Frame(object):
|
|||||||
self.block_stack = []
|
self.block_stack = []
|
||||||
```
|
```
|
||||||
|
|
||||||
接着,我们在虚拟机中增加对frame的操作。这有3个帮助函数:一个创建新的frame的方法,和压栈和出栈的方法。第四个函数,`run_frame`,完成执行frame的主要工作,待会我们再讨论这个方法。
|
接着,我们在虚拟机中增加对帧的操作。这有 3 个帮助函数:一个创建新的帧的方法(它负责为新的帧找到名字空间),和压栈和出栈的方法。第四个函数,`run_frame`,完成执行帧的主要工作,待会我们再讨论这个方法。
|
||||||
|
|
||||||
```python
|
```python
|
||||||
class VirtualMachine(object):
|
class VirtualMachine(object):
|
||||||
@ -481,9 +479,9 @@ class VirtualMachine(object):
|
|||||||
# we'll come back to this shortly
|
# we'll come back to this shortly
|
||||||
```
|
```
|
||||||
|
|
||||||
### The `Function` Class
|
#### `Function` 类
|
||||||
|
|
||||||
`Function`的实现有点扭曲,但是大部分的细节对理解解释器不重要。重要的是当调用函数时 --- `__call__`方法被调用 --- 它创建一个新的`Frame`并运行它。
|
`Function`的实现有点曲折,但是大部分的细节对理解解释器不重要。重要的是当调用函数时 —— 即调用 `__call__`方法 —— 它创建一个新的`Frame`并运行它。
|
||||||
|
|
||||||
```python
|
```python
|
||||||
class Function(object):
|
class Function(object):
|
||||||
@ -534,7 +532,7 @@ def make_cell(value):
|
|||||||
return fn.__closure__[0]
|
return fn.__closure__[0]
|
||||||
```
|
```
|
||||||
|
|
||||||
接着,回到`VirtualMachine`对象,我们对数据栈的操作也增加一些帮助方法。字节码操作的栈总是在当前frame的数据栈。这些帮助函数让我们能实现`POP_TOP`,`LOAD_FAST`字节码,并且让其他操作栈的指令可读性更高。
|
接着,回到`VirtualMachine`对象,我们对数据栈的操作也增加一些帮助方法。字节码操作的栈总是在当前帧的数据栈。这些帮助函数让我们的`POP_TOP`、`LOAD_FAST`以及其他操作栈的指令的实现可读性更高。
|
||||||
|
|
||||||
```python
|
```python
|
||||||
class VirtualMachine(object):
|
class VirtualMachine(object):
|
||||||
@ -562,11 +560,11 @@ class VirtualMachine(object):
|
|||||||
return []
|
return []
|
||||||
```
|
```
|
||||||
|
|
||||||
在我们运行frame之前,我们还需两个方法。
|
在我们运行帧之前,我们还需两个方法。
|
||||||
|
|
||||||
第一个方法,`parse_byte_and_args`,以一个字节码为输入,先检查它是否有参数,如果有,就解析它的参数。这个方法同时也更新frame的`last_instruction`属性,它指向最后执行的指令。一条没有参数的指令只有一个字节长度,而有参数的字节有3个字节长。参数的意义依赖于指令是什么。比如,前面说过,指令`POP_JUMP_IF_FALSE`,它的参数指的是跳转目标。`BUILD_LIST`, 它的参数是列表的个数。`LOAD_CONST`,它的参数是常量的索引。
|
第一个方法,`parse_byte_and_args` 以一个字节码为输入,先检查它是否有参数,如果有,就解析它的参数。这个方法同时也更新帧的`last_instruction`属性,它指向最后执行的指令。一条没有参数的指令只有一个字节长度,而有参数的字节有3个字节长。参数的意义依赖于指令是什么。比如,前面说过,指令`POP_JUMP_IF_FALSE`,它的参数指的是跳转目标。`BUILD_LIST`,它的参数是列表的个数。`LOAD_CONST`,它的参数是常量的索引。
|
||||||
|
|
||||||
一些指令用简单的数字作为参数。对于另一些,虚拟机需要一点努力去发现它含意。标准库中的`dis`模块中有一个备忘单,它解释什么参数有什么意思,这让我们的代码更加简洁。比如,列表`dis.hasname`告诉我们`LOAD_NAME`, `IMPORT_NAME`,`LOAD_GLOBAL`,以及另外的9个指令都有同样的意思:名字列表的索引。
|
一些指令用简单的数字作为参数。对于另一些,虚拟机需要一点努力去发现它含意。标准库中的`dis`模块中有一个备忘单,它解释什么参数有什么意思,这让我们的代码更加简洁。比如,列表`dis.hasname`告诉我们`LOAD_NAME`、 `IMPORT_NAME`、`LOAD_GLOBAL`,以及另外的 9 个指令的参数都有同样的意义:对于这些指令,它们的参数代表了代码对象中的名字列表的索引。
|
||||||
|
|
||||||
```python
|
```python
|
||||||
class VirtualMachine(object):
|
class VirtualMachine(object):
|
||||||
@ -600,8 +598,7 @@ class VirtualMachine(object):
|
|||||||
return byte_name, argument
|
return byte_name, argument
|
||||||
```
|
```
|
||||||
|
|
||||||
下一个方法是`dispatch`,它查看给定的指令并执行相应的操作。在CPython中,这个分派函数用一个巨大的switch语句实现,有超过1500行的代码。幸运的是,我们用的是Python,我们的代码会简洁的多。我们会为每一个字节码名字定义一个方法,然后用`getattr`来查找。就像我们前面的小解释器一样,如果一条指令叫做`FOO_BAR`,那么它对应的方法就是`byte_FOO_BAR`。现在,我们先把这些方法当做一个黑盒子。每个指令方法都会返回`None`或者一个字符串`why`,有些情况下虚拟机需要这个额外`why`信息。这些指令方法的返回值,仅作为解释器状态的内部指示,千万不要和执行frame的返回值相混淆。
|
下一个方法是`dispatch`,它查找给定的指令并执行相应的操作。在 CPython 中,这个分派函数用一个巨大的 switch 语句实现,有超过 1500 行的代码。幸运的是,我们用的是 Python,我们的代码会简洁的多。我们会为每一个字节码名字定义一个方法,然后用`getattr`来查找。就像我们前面的小解释器一样,如果一条指令叫做`FOO_BAR`,那么它对应的方法就是`byte_FOO_BAR`。现在,我们先把这些方法当做一个黑盒子。每个指令方法都会返回`None`或者一个字符串`why`,有些情况下虚拟机需要这个额外`why`信息。这些指令方法的返回值,仅作为解释器状态的内部指示,千万不要和执行帧的返回值相混淆。
|
||||||
|
|
||||||
|
|
||||||
```python
|
```python
|
||||||
class VirtualMachine(object):
|
class VirtualMachine(object):
|
||||||
@ -662,13 +659,13 @@ class VirtualMachine(object):
|
|||||||
return self.return_value
|
return self.return_value
|
||||||
```
|
```
|
||||||
|
|
||||||
### The `Block` Class
|
#### `Block` 类
|
||||||
|
|
||||||
在我们完成每个字节码方法前,我们简单的讨论一下块。一个块被用于某种控制流,特别是异常处理和循环。它负责保证当操作完成后数据栈处于正确的状态。比如,在一个循环中,一个特殊的迭代器会存在栈中,当循环完成时它从栈中弹出。解释器需要检查循环仍在继续还是已经停止。
|
在我们完成每个字节码方法前,我们简单的讨论一下块。一个块被用于某种控制流,特别是异常处理和循环。它负责保证当操作完成后数据栈处于正确的状态。比如,在一个循环中,一个特殊的迭代器会存在栈中,当循环完成时它从栈中弹出。解释器需要检查循环仍在继续还是已经停止。
|
||||||
|
|
||||||
为了跟踪这些额外的信息,解释器设置了一个标志来指示它的状态。我们用一个变量`why`实现这个标志,它可以是`None`或者是下面几个字符串这一,`"continue"`, `"break"`,`"excption"`,`return`。他们指示对块栈和数据栈进行什么操作。回到我们迭代器的例子,如果块栈的栈顶是一个`loop`块,`why`是`continue`,迭代器就因该保存在数据栈上,不是如果`why`是`break`,迭代器就会被弹出。
|
为了跟踪这些额外的信息,解释器设置了一个标志来指示它的状态。我们用一个变量`why`实现这个标志,它可以是`None`或者是下面几个字符串之一:`"continue"`、`"break"`、`"excption"`、`return`。它们指示对块栈和数据栈进行什么操作。回到我们迭代器的例子,如果块栈的栈顶是一个`loop`块,`why`的代码是`continue`,迭代器就应该保存在数据栈上,而如果`why`是`break`,迭代器就会被弹出。
|
||||||
|
|
||||||
块操作的细节比较精细,我们不会花时间在这上面,但是有兴趣的读者值得仔细的看看。
|
块操作的细节比这个还要繁琐,我们不会花时间在这上面,但是有兴趣的读者值得仔细的看看。
|
||||||
|
|
||||||
```python
|
```python
|
||||||
Block = collections.namedtuple("Block", "type, handler, stack_height")
|
Block = collections.namedtuple("Block", "type, handler, stack_height")
|
||||||
@ -737,9 +734,9 @@ class VirtualMachine(object):
|
|||||||
return why
|
return why
|
||||||
```
|
```
|
||||||
|
|
||||||
## The Instructions
|
### 指令
|
||||||
|
|
||||||
剩下了的就是完成那些指令方法了:`byte_LOAD_FAST`,`byte_BINARY_MODULO`等等。而这些指令的实现并不是很有趣,这里我们只展示了一小部分,完整的实现在这儿https://github.com/nedbat/byterun。(足够执行我们前面所述的所有代码了。)
|
剩下了的就是完成那些指令方法了:`byte_LOAD_FAST`、`byte_BINARY_MODULO`等等。而这些指令的实现并不是很有趣,这里我们只展示了一小部分,完整的实现[在 GitHub 上](https://github.com/nedbat/byterun)。(这里包括的指令足够执行我们前面所述的所有代码了。)
|
||||||
|
|
||||||
```python
|
```python
|
||||||
class VirtualMachine(object):
|
class VirtualMachine(object):
|
||||||
@ -926,11 +923,11 @@ class VirtualMachine(object):
|
|||||||
return "return"
|
return "return"
|
||||||
```
|
```
|
||||||
|
|
||||||
## Dynamic Typing: What the Compiler Doesn't Know
|
### 动态类型:编译器不知道它是什么
|
||||||
|
|
||||||
你可能听过Python是一种动态语言 --- 是它是动态类型的。在我们建造解释器的过程中,已经透露出这样的信息。
|
你可能听过 Python 是一种动态语言 —— 它是动态类型的。在我们建造解释器的过程中,已经透露出这样的信息。
|
||||||
|
|
||||||
动态的一个意思是很多工作在运行时完成。前面我们看到Python的编译器没有很多关于代码真正做什么的信息。举个例子,考虑下面这个简单的函数`mod`。它取两个参数,返回它们的模运算值。从它的字节码中,我们看到变量`a`和`b`首先被加载,然后字节码`BINAY_MODULO`完成这个模运算。
|
动态的一个意思是很多工作是在运行时完成的。前面我们看到 Python 的编译器没有很多关于代码真正做什么的信息。举个例子,考虑下面这个简单的函数`mod`。它取两个参数,返回它们的模运算值。从它的字节码中,我们看到变量`a`和`b`首先被加载,然后字节码`BINAY_MODULO`完成这个模运算。
|
||||||
|
|
||||||
```python
|
```python
|
||||||
>>> def mod(a, b):
|
>>> def mod(a, b):
|
||||||
@ -944,25 +941,25 @@ class VirtualMachine(object):
|
|||||||
4
|
4
|
||||||
```
|
```
|
||||||
|
|
||||||
计算19 % 5得4,--- 一点也不奇怪。如果我们用不同类的参数呢?
|
计算 19 % 5 得4,—— 一点也不奇怪。如果我们用不同类的参数呢?
|
||||||
|
|
||||||
```python
|
```python
|
||||||
>>> mod("by%sde", "teco")
|
>>> mod("by%sde", "teco")
|
||||||
'bytecode'
|
'bytecode'
|
||||||
```
|
```
|
||||||
|
|
||||||
刚才发生了什么?你可能见过这样的语法,格式化字符串。
|
刚才发生了什么?你可能在其它地方见过这样的语法,格式化字符串。
|
||||||
|
|
||||||
```
|
```
|
||||||
>>> print("by%sde" % "teco")
|
>>> print("by%sde" % "teco")
|
||||||
bytecode
|
bytecode
|
||||||
```
|
```
|
||||||
|
|
||||||
用符号`%`去格式化字符串会调用字节码`BUNARY_MODULO`.它取栈顶的两个值求模,不管这两个值是字符串,数字或是你自己定义的类的实例。字节码在函数编译时生成(或者说,函数定义时)相同的字节码会用于不同类的参数。
|
用符号`%`去格式化字符串会调用字节码`BUNARY_MODULO`。它取栈顶的两个值求模,不管这两个值是字符串、数字或是你自己定义的类的实例。字节码在函数编译时生成(或者说,函数定义时)相同的字节码会用于不同类的参数。
|
||||||
|
|
||||||
Python的编译器关于字节码的功能知道的很少。而取决于解释器来决定`BINAYR_MODULO`应用于什么类型的对象并完成正确的操作。这就是为什么Python被描述为_动态类型_:直到运行前你不必知道这个函数参数的类型。相反,在一个静态类型语言中,程序员需要告诉编译器参数的类型是什么(或者编译器自己推断出参数的类型。)
|
Python 的编译器关于字节码的功能知道的很少,而取决于解释器来决定`BINAYR_MODULO`应用于什么类型的对象并完成正确的操作。这就是为什么 Python 被描述为动态类型(dynamically typed):直到运行前你不必知道这个函数参数的类型。相反,在一个静态类型语言中,程序员需要告诉编译器参数的类型是什么(或者编译器自己推断出参数的类型。)
|
||||||
|
|
||||||
编译器的无知是优化Python的一个挑战 --- 只看字节码,而不真正运行它,你就不知道每条字节码在干什么!你可以定义一个类,实现`__mod__`方法,当你对这个类的实例使用`%`时,Python就会自动调用这个方法。所以,`BINARY_MODULO`其实可以运行任何代码。
|
编译器的无知是优化 Python 的一个挑战 —— 只看字节码,而不真正运行它,你就不知道每条字节码在干什么!你可以定义一个类,实现`__mod__`方法,当你对这个类的实例使用`%`时,Python 就会自动调用这个方法。所以,`BINARY_MODULO`其实可以运行任何代码。
|
||||||
|
|
||||||
看看下面的代码,第一个`a % b`看起来没有用。
|
看看下面的代码,第一个`a % b`看起来没有用。
|
||||||
|
|
||||||
@ -972,25 +969,23 @@ def mod(a,b):
|
|||||||
return a %b
|
return a %b
|
||||||
```
|
```
|
||||||
|
|
||||||
不幸的是,对这段代码进行静态分析 --- 不运行它 --- 不能确定第一个`a % b`没有做任何事。用 `%`调用`__mod__`可能会写一个文件,或是和程序的其他部分交互,或者其他任何可以在Python中完成的事。很难优化一个你不知道它会做什么的函数。在Russell Power和Alex Rubinsteyn的优秀论文中写道,“我们可以用多快的速度解释Python?”,他们说,“在普遍缺乏类型信息下,每条指令必须被看作一个`INVOKE_ARBITRARY_METHOD`。”
|
不幸的是,对这段代码进行静态分析 —— 不运行它 —— 不能确定第一个`a % b`没有做任何事。用 `%`调用`__mod__`可能会写一个文件,或是和程序的其他部分交互,或者其他任何可以在 Python 中完成的事。很难优化一个你不知道它会做什么的函数。在 Russell Power 和 Alex Rubinsteyn 的优秀论文中写道,“我们可以用多快的速度解释 Python?”,他们说,“在普遍缺乏类型信息下,每条指令必须被看作一个`INVOKE_ARBITRARY_METHOD`。”
|
||||||
|
|
||||||
## Conclusion
|
### 总结
|
||||||
|
|
||||||
Byterun是一个比CPython容易理解的简洁的Python解释器。Byterun复制了CPython的主要结构:一个基于栈的指令集称为字节码,它们顺序执行或在指令间跳转,向栈中压入和从中弹出数据。解释器随着函数和生成器的调用和返回,动态的创建,销毁frame,并在frame间跳转。Byterun也有着和真正解释器一样的限制:因为Python使用动态类型,解释器必须在运行时决定指令的正确行为。
|
Byterun 是一个比 CPython 容易理解的简洁的 Python 解释器。Byterun 复制了 CPython 的主要结构:一个基于栈的解释器对称之为字节码的指令集进行操作,它们顺序执行或在指令间跳转,向栈中压入和从中弹出数据。解释器随着函数和生成器的调用和返回,动态的创建、销毁帧,并在帧之间跳转。Byterun 也有着和真正解释器一样的限制:因为 Python 使用动态类型,解释器必须在运行时决定指令的正确行为。
|
||||||
|
|
||||||
我鼓励你去反汇编你的程序,然后用Byterun来运行。你很快会发现这个缩短版的Byterun所没有实现的指令。完整的实现在https://github.com/nedbat/byterun,或者仔细阅读真正的CPython解释器`ceval.c`,你也可以实现自己的解释器!
|
我鼓励你去反汇编你的程序,然后用 Byterun 来运行。你很快会发现这个缩短版的 Byterun 所没有实现的指令。完整的实现在 https://github.com/nedbat/byterun ,或者你可以仔细阅读真正的 CPython 解释器`ceval.c`,你也可以实现自己的解释器!
|
||||||
|
|
||||||
## Acknowledgements
|
### 致谢
|
||||||
|
|
||||||
Thanks to Ned Batchelder for originating this project and guiding my contributions, Michael Arntzenius for his help debugging the code and editing the prose, Leta Montopoli for her edits, and the entire Recurse Center community for their support and interest. Any errors are my own.
|
感谢 Ned Batchelder 发起这个项目并引导我的贡献,感谢 Michael Arntzenius 帮助调试代码和这篇文章的修订,感谢 Leta Montopoli 的修订,以及感谢整个 Recurse Center 社区的支持和鼓励。所有的不足全是我自己没搞好。
|
||||||
|
|
||||||
--------------------------------------
|
--------------------------------------
|
||||||
via:http://aosabook.org/en/500L/a-python-interpreter-written-in-python.html
|
via: http://aosabook.org/en/500L/a-python-interpreter-written-in-python.html
|
||||||
|
|
||||||
作者: Allison Kaptur Allison
|
|
||||||
|
|
||||||
|
作者: Allison Kaptur
|
||||||
译者:[qingyunha](https://github.com/qingyunha)
|
译者:[qingyunha](https://github.com/qingyunha)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
校对:[校对者ID](https://github.com/校对者ID)
|
|
||||||
|
|
||||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创翻译,[Linux中国](http://linux.cn/) 荣誉推出
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创翻译,[Linux中国](http://linux.cn/) 荣誉推出
|
@ -0,0 +1,236 @@
|
|||||||
|
使用 Python 和 Asyncio 编写在线多用人游戏(二)
|
||||||
|
==================================================================
|
||||||
|
|
||||||
|
![](https://7webpages.com/media/cache/fd/d1/fdd1f8f8bbbf4166de5f715e6ed0ac00.gif)
|
||||||
|
|
||||||
|
> 你在 Python 中用过异步编程吗?本文中我会告诉你怎样做,而且用一个[能工作的例子][1]来展示它:这是一个流行的贪吃蛇游戏,而且是为多人游戏而设计的。
|
||||||
|
|
||||||
|
介绍和理论部分参见“[第一部分 异步化][2]”。
|
||||||
|
|
||||||
|
- [游戏入口在此,点此体验][1]。
|
||||||
|
|
||||||
|
### 3、编写游戏循环主体
|
||||||
|
|
||||||
|
游戏循环是每一个游戏的核心。它持续地运行以读取玩家的输入、更新游戏的状态,并且在屏幕上渲染游戏结果。在在线游戏中,游戏循环分为客户端和服务端两部分,所以一般有两个循环通过网络通信。通常客户端的角色是获取玩家输入,比如按键或者鼠标移动,将数据传输给服务端,然后接收需要渲染的数据。服务端处理来自玩家的所有数据,更新游戏的状态,执行渲染下一帧的必要计算,然后将结果传回客户端,例如游戏中对象的新位置。如果没有可靠的理由,不混淆客户端和服务端的角色是一件很重要的事。如果你在客户端执行游戏逻辑的计算,很容易就会和其它客户端失去同步,其实你的游戏也可以通过简单地传递客户端的数据来创建。
|
||||||
|
|
||||||
|
> 游戏循环的一次迭代称为一个嘀嗒(tick)。嘀嗒是一个事件,表示当前游戏循环的迭代已经结束,下一帧(或者多帧)的数据已经就绪。
|
||||||
|
|
||||||
|
在后面的例子中,我们使用相同的客户端,它使用 WebSocket 从一个网页上连接到服务端。它执行一个简单的循环,将按键码发送给服务端,并显示来自服务端的所有信息。[客户端代码戳这里][4]。
|
||||||
|
|
||||||
|
### 例子 3.1:基本游戏循环
|
||||||
|
|
||||||
|
> [例子 3.1 源码][5]。
|
||||||
|
|
||||||
|
我们使用 [aiohttp][6] 库来创建游戏服务器。它可以通过 asyncio 创建网页服务器和客户端。这个库的一个优势是它同时支持普通 http 请求和 websocket。所以我们不用其他网页服务器来渲染游戏的 html 页面。
|
||||||
|
|
||||||
|
下面是启动服务器的方法:
|
||||||
|
|
||||||
|
```
|
||||||
|
app = web.Application()
|
||||||
|
app["sockets"] = []
|
||||||
|
|
||||||
|
asyncio.ensure_future(game_loop(app))
|
||||||
|
|
||||||
|
app.router.add_route('GET', '/connect', wshandler)
|
||||||
|
app.router.add_route('GET', '/', handle)
|
||||||
|
|
||||||
|
web.run_app(app)
|
||||||
|
```
|
||||||
|
|
||||||
|
`web.run_app` 是创建服务主任务的快捷方法,通过它的 `run_forever()` 方法来执行 `asyncio` 事件循环。建议你查看这个方法的源码,弄清楚服务器到底是如何创建和结束的。
|
||||||
|
|
||||||
|
`app` 变量就是一个类似于字典的对象,它用于在所连接的客户端之间共享数据。我们使用它来存储连接的套接字的列表。随后会用这个列表来给所有连接的客户端发送消息。`asyncio.ensure_future()` 调用会启动主游戏循环的任务,每隔2 秒向客户端发送嘀嗒消息。这个任务会在同样的 asyncio 事件循环中和网页服务器并行执行。
|
||||||
|
|
||||||
|
有两个网页请求处理器:`handle` 是提供 html 页面的处理器;`wshandler` 是主要的 websocket 服务器任务,处理和客户端之间的交互。在事件循环中,每一个连接的客户端都会创建一个新的 `wshandler` 任务。这个任务会添加客户端的套接字到列表中,以便 `game_loop` 任务可以给所有的客户端发送消息。然后它将随同消息回显客户端的每个击键。
|
||||||
|
|
||||||
|
在启动的任务中,我们在 `asyncio` 的主事件循环中启动 worker 循环。任务之间的切换发生在它们之间任何一个使用 `await`语句来等待某个协程结束时。例如 `asyncio.sleep` 仅仅是将程序执行权交给调度器一段指定的时间;`ws.receive` 等待 websocket 的消息,此时调度器可能切换到其它任务。
|
||||||
|
|
||||||
|
在浏览器中打开主页,连接上服务器后,试试随便按下键。它们的键值会从服务端返回,每隔 2 秒这个数字会被游戏循环中发给所有客户端的嘀嗒消息所覆盖。
|
||||||
|
|
||||||
|
我们刚刚创建了一个处理客户端按键的服务器,主游戏循环在后台做一些处理,周期性地同时更新所有的客户端。
|
||||||
|
|
||||||
|
### 例子 3.2: 根据请求启动游戏
|
||||||
|
|
||||||
|
> [例子 3.2 的源码][7]
|
||||||
|
|
||||||
|
在前一个例子中,在服务器的生命周期内,游戏循环一直运行着。但是现实中,如果没有一个人连接服务器,空运行游戏循环通常是不合理的。而且,同一个服务器上可能有不同的“游戏房间”。在这种假设下,每一个玩家“创建”一个游戏会话(比如说,多人游戏中的一个比赛或者大型多人游戏中的副本),这样其他用户可以加入其中。当游戏会话开始时,游戏循环才开始执行。
|
||||||
|
|
||||||
|
在这个例子中,我们使用一个全局标记来检测游戏循环是否在执行。当第一个用户发起连接时,启动它。最开始,游戏循环没有执行,标记设置为 `False`。游戏循环是通过客户端的处理方法启动的。
|
||||||
|
|
||||||
|
```
|
||||||
|
if app["game_is_running"] == False:
|
||||||
|
asyncio.ensure_future(game_loop(app))
|
||||||
|
```
|
||||||
|
|
||||||
|
当 `game_loop()` 运行时,这个标记设置为 `True`;当所有客户端都断开连接时,其又被设置为 `False`。
|
||||||
|
|
||||||
|
### 例子 3.3:管理任务
|
||||||
|
|
||||||
|
> [例子3.3源码][8]
|
||||||
|
|
||||||
|
这个例子用来解释如何和任务对象协同工作。我们把游戏循环的任务直接存储在游戏循环的全局字典中,代替标记的使用。在像这样的一个简单例子中并不一定是最优的,但是有时候你可能需要控制所有已经启动的任务。
|
||||||
|
|
||||||
|
```
|
||||||
|
if app["game_loop"] is None or \
|
||||||
|
app["game_loop"].cancelled():
|
||||||
|
app["game_loop"] = asyncio.ensure_future(game_loop(app))
|
||||||
|
```
|
||||||
|
|
||||||
|
这里 `ensure_future()` 返回我们存放在全局字典中的任务对象,当所有用户都断开连接时,我们使用下面方式取消任务:
|
||||||
|
|
||||||
|
```
|
||||||
|
app["game_loop"].cancel()
|
||||||
|
```
|
||||||
|
|
||||||
|
这个 `cancel()` 调用将通知调度器不要向这个协程传递执行权,而且将它的状态设置为已取消:`cancelled`,之后可以通过 `cancelled()` 方法来检查是否已取消。这里有一个值得一提的小注意点:当你持有一个任务对象的外部引用时,而这个任务执行中发生了异常,这个异常不会抛出。取而代之的是为这个任务设置一个异常状态,可以通过 `exception()` 方法来检查是否出现了异常。这种悄无声息地失败在调试时不是很有用。所以,你可能想用抛出所有异常来取代这种做法。你可以对所有未完成的任务显式地调用 `result()` 来实现。可以通过如下的回调来实现:
|
||||||
|
|
||||||
|
```
|
||||||
|
app["game_loop"].add_done_callback(lambda t: t.result())
|
||||||
|
```
|
||||||
|
|
||||||
|
如果我们打算在我们代码中取消这个任务,但是又不想产生 `CancelError` 异常,有一个检查 `cancelled` 状态的点:
|
||||||
|
|
||||||
|
```
|
||||||
|
app["game_loop"].add_done_callback(lambda t: t.result()
|
||||||
|
if not t.cancelled() else None)
|
||||||
|
```
|
||||||
|
|
||||||
|
注意仅当你持有任务对象的引用时才需要这么做。在前一个例子,所有的异常都是没有额外的回调,直接抛出所有异常。
|
||||||
|
|
||||||
|
### 例子 3.4:等待多个事件
|
||||||
|
|
||||||
|
> [例子 3.4 源码][9]
|
||||||
|
|
||||||
|
在许多场景下,在客户端的处理方法中你需要等待多个事件的发生。除了来自客户端的消息,你可能需要等待不同类型事件的发生。比如,如果你的游戏时间有限制,那么你可能需要等一个来自定时器的信号。或者你需要使用管道来等待来自其它进程的消息。亦或者是使用分布式消息系统的网络中其它服务器的信息。
|
||||||
|
|
||||||
|
为了简单起见,这个例子是基于例子 3.1。但是这个例子中我们使用 `Condition` 对象来与已连接的客户端保持游戏循环的同步。我们不保存套接字的全局列表,因为只在该处理方法中使用套接字。当游戏循环停止迭代时,我们使用 `Condition.notify_all()` 方法来通知所有的客户端。这个方法允许在 `asyncio` 的事件循环中使用发布/订阅的模式。
|
||||||
|
|
||||||
|
为了等待这两个事件,首先我们使用 `ensure_future()` 来封装任务中这个可等待对象。
|
||||||
|
|
||||||
|
```
|
||||||
|
if not recv_task:
|
||||||
|
recv_task = asyncio.ensure_future(ws.receive())
|
||||||
|
if not tick_task:
|
||||||
|
await tick.acquire()
|
||||||
|
tick_task = asyncio.ensure_future(tick.wait())
|
||||||
|
```
|
||||||
|
|
||||||
|
在我们调用 `Condition.wait()` 之前,我们需要在它后面获取一把锁。这就是我们为什么先调用 `tick.acquire()` 的原因。在调用 `tick.wait()` 之后,锁会被释放,这样其他的协程也可以使用它。但是当我们收到通知时,会重新获取锁,所以在收到通知后需要调用 `tick.release()` 来释放它。
|
||||||
|
|
||||||
|
我们使用 `asyncio.wait()` 协程来等待两个任务。
|
||||||
|
|
||||||
|
```
|
||||||
|
done, pending = await asyncio.wait(
|
||||||
|
[recv_task,
|
||||||
|
tick_task],
|
||||||
|
return_when=asyncio.FIRST_COMPLETED)
|
||||||
|
```
|
||||||
|
|
||||||
|
程序会阻塞,直到列表中的任意一个任务完成。然后它返回两个列表:执行完成的任务列表和仍然在执行的任务列表。如果任务执行完成了,其对应变量赋值为 `None`,所以在下一个迭代时,它可能会被再次创建。
|
||||||
|
|
||||||
|
### 例子 3.5: 结合多个线程
|
||||||
|
|
||||||
|
> [例子 3.5 源码][10]
|
||||||
|
|
||||||
|
在这个例子中,我们结合 `asyncio` 循环和线程,在一个单独的线程中执行主游戏循环。我之前提到过,由于 `GIL` 的存在,Python 代码的真正并行执行是不可能的。所以使用其它线程来执行复杂计算并不是一个好主意。然而,在使用 `asyncio` 时结合线程有原因的:当我们使用的其它库不支持 `asyncio` 时就需要。在主线程中调用这些库会阻塞循环的执行,所以异步使用他们的唯一方法是在不同的线程中使用他们。
|
||||||
|
|
||||||
|
我们使用 `asyncio` 循环的`run_in_executor()` 方法和 `ThreadPoolExecutor` 来执行游戏循环。注意 `game_loop()` 已经不再是一个协程了。它是一个由其它线程执行的函数。然而我们需要和主线程交互,在游戏事件到来时通知客户端。`asyncio` 本身不是线程安全的,它提供了可以在其它线程中执行你的代码的方法。普通函数有 `call_soon_threadsafe()`,协程有 `run_coroutine_threadsafe()`。我们在 `notify()` 协程中增加了通知客户端游戏的嘀嗒的代码,然后通过另外一个线程执行主事件循环。
|
||||||
|
|
||||||
|
```
|
||||||
|
def game_loop(asyncio_loop):
|
||||||
|
print("Game loop thread id {}".format(threading.get_ident()))
|
||||||
|
async def notify():
|
||||||
|
print("Notify thread id {}".format(threading.get_ident()))
|
||||||
|
await tick.acquire()
|
||||||
|
tick.notify_all()
|
||||||
|
tick.release()
|
||||||
|
|
||||||
|
while 1:
|
||||||
|
task = asyncio.run_coroutine_threadsafe(notify(), asyncio_loop)
|
||||||
|
# blocking the thread
|
||||||
|
sleep(1)
|
||||||
|
# make sure the task has finished
|
||||||
|
task.result()
|
||||||
|
```
|
||||||
|
|
||||||
|
当你执行这个例子时,你会看到 “Notify thread id” 和 “Main thread id” 相等,因为 `notify()` 协程在主线程中执行。与此同时 `sleep(1)` 在另外一个线程中执行,因此它不会阻塞主事件循环。
|
||||||
|
|
||||||
|
### 例子 3.6:多进程和扩展
|
||||||
|
|
||||||
|
> [例子 3.6 源码][11]
|
||||||
|
|
||||||
|
单线程的服务器可能运行得很好,但是它只能使用一个 CPU 核。为了将服务扩展到多核,我们需要执行多个进程,每个进程执行各自的事件循环。这样我们需要在进程间交互信息或者共享游戏的数据。而且在一个游戏中经常需要进行复杂的计算,例如路径查找之类。这些任务有时候在一个游戏嘀嗒中没法快速完成。在协程中不推荐进行费时的计算,因为它会阻塞事件的处理。在这种情况下,将这个复杂任务交给其它并行执行的进程可能更合理。
|
||||||
|
|
||||||
|
最简单的使用多个核的方法是启动多个使用单核的服务器,就像之前的例子中一样,每个服务器占用不同的端口。你可以使用 `supervisord` 或者其它进程控制的系统。这个时候你需要一个像 `HAProxy` 这样的负载均衡器,使得连接的客户端分布在多个进程间。已经有一些可以连接 asyncio 和一些流行的消息及存储系统的适配系统。例如:
|
||||||
|
|
||||||
|
- [aiomcache][12] 用于 memcached 客户端
|
||||||
|
- [aiozmq][13] 用于 zeroMQ
|
||||||
|
- [aioredis][14] 用于 Redis 存储,支持发布/订阅
|
||||||
|
|
||||||
|
你可以在 github 或者 pypi 上找到其它的软件包,大部分以 `aio` 开头。
|
||||||
|
|
||||||
|
使用网络服务在存储持久状态和交换某些信息时可能比较有效。但是如果你需要进行进程间通信的实时处理,它的性能可能不足。此时,使用标准的 unix 管道可能更合适。`asyncio` 支持管道,在`aiohttp`仓库有个 [使用管道的服务器的非常底层的例子][15]。
|
||||||
|
|
||||||
|
在当前的例子中,我们使用 Python 的高层类库 [multiprocessing][16] 来在不同的核上启动复杂的计算,使用 `multiprocessing.Queue` 来进行进程间的消息交互。不幸的是,当前的 `multiprocessing` 实现与 `asyncio` 不兼容。所以每一个阻塞方法的调用都会阻塞事件循环。但是此时线程正好可以起到帮助作用,因为如果在不同线程里面执行 `multiprocessing` 的代码,它就不会阻塞主线程。所有我们需要做的就是把所有进程间的通信放到另外一个线程中去。这个例子会解释如何使用这个方法。和上面的多线程例子非常类似,但是我们从线程中创建的是一个新的进程。
|
||||||
|
|
||||||
|
```
|
||||||
|
def game_loop(asyncio_loop):
|
||||||
|
# coroutine to run in main thread
|
||||||
|
async def notify():
|
||||||
|
await tick.acquire()
|
||||||
|
tick.notify_all()
|
||||||
|
tick.release()
|
||||||
|
|
||||||
|
queue = Queue()
|
||||||
|
|
||||||
|
# function to run in a different process
|
||||||
|
def worker():
|
||||||
|
while 1:
|
||||||
|
print("doing heavy calculation in process {}".format(os.getpid()))
|
||||||
|
sleep(1)
|
||||||
|
queue.put("calculation result")
|
||||||
|
|
||||||
|
Process(target=worker).start()
|
||||||
|
|
||||||
|
while 1:
|
||||||
|
# blocks this thread but not main thread with event loop
|
||||||
|
result = queue.get()
|
||||||
|
print("getting {} in process {}".format(result, os.getpid()))
|
||||||
|
task = asyncio.run_coroutine_threadsafe(notify(), asyncio_loop)
|
||||||
|
task.result()
|
||||||
|
```
|
||||||
|
|
||||||
|
这里我们在另外一个进程中运行 `worker()` 函数。它包括一个执行复杂计算并把计算结果放到 `queue` 中的循环,这个 `queue` 是 `multiprocessing.Queue` 的实例。然后我们就可以在另外一个线程的主事件循环中获取结果并通知客户端,就和例子 3.5 一样。这个例子已经非常简化了,它没有合理的结束进程。而且在真实的游戏中,我们可能需要另外一个队列来将数据传递给 `worker`。
|
||||||
|
|
||||||
|
有一个项目叫 [aioprocessing][17],它封装了 `multiprocessing`,使得它可以和 `asyncio` 兼容。但是实际上它只是和上面例子使用了完全一样的方法:从线程中创建进程。它并没有给你带来任何方便,除了它使用了简单的接口隐藏了后面的这些技巧。希望在 Python 的下一个版本中,我们能有一个基于协程且支持 `asyncio` 的 `multiprocessing` 库。
|
||||||
|
|
||||||
|
> 注意!如果你从主线程或者主进程中创建了一个不同的线程或者子进程来运行另外一个 `asyncio` 事件循环,你需要显式地使用 `asyncio.new_event_loop()` 来创建循环,不然的话可能程序不会正常工作。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://7webpages.com/blog/writing-online-multiplayer-game-with-python-and-asyncio-writing-game-loop/
|
||||||
|
|
||||||
|
作者:[Kyrylo Subbotin][a]
|
||||||
|
译者:[chunyang-wen](https://github.com/chunyang-wen)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://7webpages.com/blog/writing-online-multiplayer-game-with-python-and-asyncio-writing-game-loop/
|
||||||
|
[1]: http://snakepit-game.com/
|
||||||
|
[2]: https://linux.cn/article-7767-1.html
|
||||||
|
[3]: http://snakepit-game.com/
|
||||||
|
[4]: https://github.com/7WebPages/snakepit-game/blob/master/simple/index.html
|
||||||
|
[5]: https://github.com/7WebPages/snakepit-game/blob/master/simple/game_loop_basic.py
|
||||||
|
[6]: http://aiohttp.readthedocs.org/
|
||||||
|
[7]: https://github.com/7WebPages/snakepit-game/blob/master/simple/game_loop_handler.py
|
||||||
|
[8]: https://github.com/7WebPages/snakepit-game/blob/master/simple/game_loop_global.py
|
||||||
|
[9]: https://github.com/7WebPages/snakepit-game/blob/master/simple/game_loop_wait.py
|
||||||
|
[10]: https://github.com/7WebPages/snakepit-game/blob/master/simple/game_loop_thread.py
|
||||||
|
[11]: https://github.com/7WebPages/snakepit-game/blob/master/simple/game_loop_process.py
|
||||||
|
[12]: https://github.com/aio-libs/aiomcache
|
||||||
|
[13]: https://github.com/aio-libs/aiozmq
|
||||||
|
[14]: https://github.com/aio-libs/aioredis
|
||||||
|
[15]: https://github.com/KeepSafe/aiohttp/blob/master/examples/mpsrv.py
|
||||||
|
[16]: https://docs.python.org/3.5/library/multiprocessing.html
|
||||||
|
[17]: https://github.com/dano/aioprocessing
|
@ -0,0 +1,76 @@
|
|||||||
|
Instagram 基于 Python 语言的 Web Service 效率提升之道
|
||||||
|
===============================================
|
||||||
|
|
||||||
|
Instagram 目前部署了世界上最大规模的 Django Web 框架(该框架完全使用 Python 编写)。我们最初选用 Python 是因为它久负盛名的简洁性与实用性,这非常符合我们的哲学思想——“先做简单的事情”。但简洁性也会带来效率方面的折衷。Instagram 的规模在过去两年中已经翻番,并且最近已突破 5 亿用户,所以急需最大程度地提升 web 服务效率以便我们的平台能够继续顺利地扩大。在过去的一年,我们已经将效率计划(efficiency program)提上日程,并在过去的六个月,我们已经能够做到无需向我们的 Django 层(Django tiers)添加新的容量来维持我们的用户增长。我们将在本文分享一些由我们构建的工具以及如何使用它们来优化我们的日常部署流程。
|
||||||
|
|
||||||
|
### 为何需要提升效率?
|
||||||
|
|
||||||
|
Instagram,正如所有的软件,受限于像服务器和数据中心能源这样的物理限制。鉴于这些限制,在我们的效率计划中有两个我们希望实现的主要目标:
|
||||||
|
|
||||||
|
1. Instagram 应当能够利用持续代码发布正常地提供通信服务,防止因为自然灾害、区域性网络问题等造成某一个数据中心区丢失。
|
||||||
|
2. Instagram 应当能够自由地滚动发布新产品和新功能,不必因容量而受阻。
|
||||||
|
|
||||||
|
想要实现这些目标,我们意识到我们需要持续不断地监控我们的系统并与回归(regressions)进行战斗。
|
||||||
|
|
||||||
|
### 定义效率
|
||||||
|
|
||||||
|
Web services 的瓶颈通常在于每台服务器上可用的 CPU 时间。在这种环境下,效率就意味着利用相同的 CPU 资源完成更多的任务,也就是说,每秒处理更多的用户请求(requests per second, RPS)。当我们寻找优化方法时,我们面临的第一个最大的挑战就是尝试量化我们当前的效率。到目前为止,我们一直在使用“每次请求的平均 CPU 时间”来评估效率,但使用这种指标也有其固有限制:
|
||||||
|
|
||||||
|
1. **设备多样性**。使用 CPU 时间来测量 CPU 资源并非理想方案,因为它同时受到 CPU 型号与 CPU 负载的影响。
|
||||||
|
2. **请求影响数据**。测量每次请求的 CPU 资源并非理想方案,因为在使用每次请求测量(per-request measurement)方案时,添加或移除轻量级或重量级的请求也会影响到效率指标。
|
||||||
|
|
||||||
|
相对于 CPU 时间来说,CPU 指令是一种更好的指标,因为对于相同的请求,它会报告相同的数字,不管 CPU 型号和 CPU 负载情况如何。我们选择使用了一种叫做“每个活动用户(per active user)”的指标,而不是将我们所有的数据关联到每个用户请求上。我们最终采用“每个活动用户在高峰期间的 CPU 指令(CPU instruction per active user during peak minute)”来测量效率。我们建立好新的度量标准后,下一步就是通过对 Django 的分析来更多的了解一下我们的回归。
|
||||||
|
|
||||||
|
### Django web services 分析
|
||||||
|
|
||||||
|
通过分析我们的 Django web services,我们希望回答两个主要问题:
|
||||||
|
|
||||||
|
1. CPU 回归会发生吗?
|
||||||
|
2. 是什么导致了 CPU 回归发生以及我们该怎样修复它?
|
||||||
|
|
||||||
|
想要回答第一个问题,我们需要追踪”每个活动用户的 CPU 指令(CPU-instruction-per-active-user)“指标。如果该指标增加,我们就知道已经发生了一次 CPU 回归。
|
||||||
|
|
||||||
|
我们为此构建的工具叫做 Dynostats。Dynostats 利用 Django 中间件以一定的速率采样用户请求,记录关键的效率以及性能指标,例如 CPU 总指令数、端到端请求时延、花费在访问内存缓存(memcache)和数据库服务的时间等。另一方面,每个请求都有很多可用于聚合的元数据(metadata),例如端点名称、HTTP 请求返回码、服务该请求的服务器名称以及请求中最新提交的哈希值(hash)。对于单个请求记录来说,有两个方面非常强大,因为我们可以在不同的维度上进行切割,那将帮助我们减少任何导致 CPU 回归的原因。例如,我们可以根据它们的端点名称聚合所有请求,正如下面的时间序列图所示,从图中可以清晰地看出在特定端点上是否发生了回归。
|
||||||
|
|
||||||
|
![](https://d262ilb51hltx0.cloudfront.net/max/800/1*3iouYiAchYBwzF-v0bALMw.png)
|
||||||
|
|
||||||
|
CPU 指令对测量效率很重要——当然,它们也很难获得。Python 并没有支持直接访问 CPU 硬件计数器(CPU 硬件计数器是指可编程 CPU 寄存器,用于测量性能指标,例如 CPU 指令)的公共库。另一方面,Linux 内核提供了 `perf_event_open` 系统调用。通过 Python `ctypes` 桥接技术能够让我们调用标准 C 库的系统调用函数 `syscall`,它也为我们提供了兼容 C 的数据类型,从而可以编程硬件计数器并从它们读取数据。
|
||||||
|
|
||||||
|
使用 Dynostats,我们已经可以找出 CPU 回归,并探究 CPU 回归发生的原因,例如哪个端点受到的影响最多,谁提交了真正会导致 CPU 回归的变更等。然而,当开发者收到他们的变更已经导致一次 CPU 回归发生的通知时,他们通常难以找出问题所在。如果问题很明显,那么回归可能就不会一开始就被提交!
|
||||||
|
|
||||||
|
这就是为何我们需要一个 Python 分析器,(一旦 Dynostats 发现了它)从而使开发者能够使用它找出回归发生的根本原因。不同于白手起家,我们决定对一个现成的 Python 分析器 cProfile 做适当的修改。cProfile 模块通常会提供一个统计集合来描述程序不同的部分执行时间和执行频率。我们将 cProfile 的定时器(timer)替换成了一个从硬件计数器读取的 CPU 指令计数器,以此取代对时间的测量。我们在采样请求后产生数据并把数据发送到数据流水线。我们也会发送一些我们在 Dynostats 所拥有的类似元数据,例如服务器名称、集群、区域、端点名称等。
|
||||||
|
|
||||||
|
在数据流水线的另一边,我们创建了一个消费数据的尾随者(tailer)。尾随者的主要功能是解析 cProfile 的统计数据并创建能够表示 Python 函数级别的 CPU 指令的实体。如此,我们能够通过 Python 函数来聚合 CPU 指令,从而更加方便地找出是什么函数导致了 CPU 回归。
|
||||||
|
|
||||||
|
### 监控与警报机制
|
||||||
|
|
||||||
|
在 Instagram,我们[每天部署 30-50 次后端服务][1]。这些部署中的任何一个都能发生 CPU 回归的问题。因为每次发生通常都包含至少一个差异(diff),所以找出任何回归是很容易的。我们的效率监控机制包括在每次发布前后都会在 Dynostats 中扫描 CPU 指令,并且当变更超出某个阈值时发出警告。对于长期会发生 CPU 回归的情况,我们也有一个探测器为负载最繁重的端点提供日常和每周的变更扫描。
|
||||||
|
|
||||||
|
部署新的变更并非触发一次 CPU 回归的唯一情况。在许多情况下,新的功能和新的代码路径都由全局环境变量(global environment variables,GEV)控制。 在一个计划好的时间表上,给一部分用户发布新功能是很常见事情。我们在 Dynostats 和 cProfile 统计数据中为每个请求添加了这个信息作为额外的元数据字段。按这些字段将请求分组可以找出由全局环境变量(GEV)改变导致的可能的 CPU 回归问题。这让我们能够在它们对性能造成影响前就捕获到 CPU 回归。
|
||||||
|
|
||||||
|
### 接下来是什么?
|
||||||
|
|
||||||
|
Dynostats 和我们定制的 cProfile,以及我们建立的支持它们的监控和警报机制能够有效地找出大多数导致 CPU 回归的元凶。这些进展已经帮助我们恢复了超过 50% 的不必要的 CPU 回归,否则我们就根本不会知道。
|
||||||
|
|
||||||
|
我们仍然还有一些可以提升的方面,并很容易将它们地加入到 Instagram 的日常部署流程中:
|
||||||
|
|
||||||
|
1. CPU 指令指标应该要比其它指标如 CPU 时间更加稳定,但我们仍然观察了让我们头疼的差异。保持“信噪比(signal:noise ratio)”合理地低是非常重要的,这样开发者们就可以集中于真实的回归上。这可以通过引入置信区间(confidence intervals)的概念来提升,并在信噪比过高时发出警报。针对不同的端点,变化的阈值也可以设置为不同值。
|
||||||
|
2. 通过更改 GEV 来探测 CPU 回归的一个限制就是我们要在 Dynostats 中手动启用这些比较的日志输出。当 GEV 的数量逐渐增加,开发了越来越多的功能,这就不便于扩展了。相反,我们能够利用一个自动化框架来调度这些比较的日志输出,并对所有的 GEV 进行遍历,然后当检查到回归时就发出警告。
|
||||||
|
3. cProfile 需要一些增强以便更好地处理封装函数以及它们的子函数。
|
||||||
|
|
||||||
|
鉴于我们在为 Instagram 的 web service 构建效率框架中所投入的工作,所以我们对于将来使用 Python 继续扩展我们的服务很有信心。我们也开始向 Python 语言本身投入更多,并且开始探索从 Python 2 转移 Python 3 之道。我们将会继续探索并做更多的实验以继续提升基础设施与开发者效率,我们期待着很快能够分享更多的经验。
|
||||||
|
|
||||||
|
本文作者 Min Ni 是 Instagram 的软件工程师。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://engineering.instagram.com/web-service-efficiency-at-instagram-with-python-4976d078e366#.tiakuoi4p
|
||||||
|
|
||||||
|
作者:[Min Ni][a]
|
||||||
|
译者:[ChrisLeeGit](https://github.com/chrisleegit)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://engineering.instagram.com/@InstagramEng?source=post_header_lockup
|
||||||
|
[1]: https://engineering.instagram.com/continuous-deployment-at-instagram-1e18548f01d1#.p5adp7kcz
|
@ -0,0 +1,134 @@
|
|||||||
|
使用 Python 和 Asyncio 来编写在线多人游戏(三)
|
||||||
|
=================================================================
|
||||||
|
|
||||||
|
![](https://7webpages.com/media/cache/17/81/178135a6db5074c72a1394d31774c658.gif)
|
||||||
|
|
||||||
|
> 在这个系列中,我们基于多人游戏 [贪吃蛇][1] 来制作一个异步的 Python 程序。上一篇文章聚焦于[编写游戏循环][2]上,而本系列第 1 部分则涵盖了如何[异步化][3]。
|
||||||
|
|
||||||
|
- 代码戳[这里][4]
|
||||||
|
|
||||||
|
### 4、制作一个完整的游戏
|
||||||
|
|
||||||
|
![](https://7webpages.com/static/img/14chs7.gif)
|
||||||
|
|
||||||
|
#### 4.1 工程概览
|
||||||
|
|
||||||
|
在此部分,我们将回顾一个完整在线游戏的设计。这是一个经典的贪吃蛇游戏,增加了多玩家支持。你可以自己在 (<http://snakepit-game.com>) 亲自试玩。源码在 GitHub 的这个[仓库][5]。游戏包括下列文件:
|
||||||
|
|
||||||
|
- [server.py][6] - 处理主游戏循环和连接。
|
||||||
|
- [game.py][7] - 主要的 `Game` 类。实现游戏的逻辑和游戏的大部分通信协议。
|
||||||
|
- [player.py][8] - `Player` 类,包括每一个独立玩家的数据和蛇的展现。这个类负责获取玩家的输入并相应地移动蛇。
|
||||||
|
- [datatypes.py][9] - 基本数据结构。
|
||||||
|
- [settings.py][10] - 游戏设置,在注释中有相关的说明。
|
||||||
|
- [index.html][11] - 客户端所有的 html 和 javascript代码都放在一个文件中。
|
||||||
|
|
||||||
|
#### 4.2 游戏循环内窥
|
||||||
|
|
||||||
|
多人的贪吃蛇游戏是个用于学习十分好的例子,因为它简单。所有的蛇在每个帧中移动到一个位置,而且帧以非常低的频率进行变化,这样就可以让你就观察到游戏引擎到底是如何工作的。因为速度慢,对于玩家的按键不会立马响应。按键先是记录下来,然后在一个游戏循环迭代的最后计算下一帧时使用。
|
||||||
|
|
||||||
|
> 现代的动作游戏帧频率更高,而且通常服务端和客户端的帧频率是不相等的。客户端的帧频率通常依赖于客户端的硬件性能,而服务端的帧频率则是固定的。一个客户端可能根据一个游戏“嘀嗒”的数据渲染多个帧。这样就可以创建平滑的动画,这个受限于客户端的性能。在这个例子中,服务端不仅传输物体的当前位置,也要传输它们的移动方向、速度和加速度。客户端的帧频率称之为 FPS(每秒帧数:frames per second),服务端的帧频率称之为 TPS(每秒滴答数:ticks per second)。在这个贪吃蛇游戏的例子中,二者的值是相等的,在客户端显示的一帧是在服务端的一个“嘀嗒”内计算出来的。
|
||||||
|
|
||||||
|
我们使用类似文本模式的游戏区域,事实上是 html 表格中的一个字符宽的小格。游戏中的所有对象都是通过表格中的不同颜色字符来表示。大部分时候,客户端将按键的码发送至服务端,然后每个“滴答”更新游戏区域。服务端一次更新包括需要更新字符的坐标和颜色。所以我们将所有游戏逻辑放置在服务端,只将需要渲染的数据发送给客户端。此外,我们通过替换通过网络发送的数据来减少游戏被破解的概率。
|
||||||
|
|
||||||
|
#### 4.3 它是如何运行的?
|
||||||
|
|
||||||
|
这个游戏中的服务端出于简化的目的,它和例子 3.2 类似。但是我们用一个所有服务端都可访问的 `Game` 对象来代替之前保存了所有已连接 websocket 的全局列表。一个 `Game` 实例包括一个表示连接到此游戏的玩家的 `Player` 对象的列表(在 `self._players` 属性里面),以及他们的个人数据和 websocket 对象。将所有游戏相关的数据存储在一个 `Game` 对象中,会方便我们增加多个游戏房间这个功能——如果我们要增加这个功能的话。这样,我们维护多个 `Game` 对象,每个游戏开始时创建一个。
|
||||||
|
|
||||||
|
客户端和服务端的所有交互都是通过编码成 json 的消息来完成。来自客户端的消息仅包含玩家所按下键码对应的编号。其它来自客户端消息使用如下格式:
|
||||||
|
|
||||||
|
```
|
||||||
|
[command, arg1, arg2, ... argN ]
|
||||||
|
```
|
||||||
|
|
||||||
|
来自服务端的消息以列表的形式发送,因为通常一次要发送多个消息 (大多数情况下是渲染的数据):
|
||||||
|
|
||||||
|
```
|
||||||
|
[[command, arg1, arg2, ... argN ], ... ]
|
||||||
|
```
|
||||||
|
|
||||||
|
在每次游戏循环迭代的最后会计算下一帧,并且将数据发送给所有的客户端。当然,每次不是发送完整的帧,而是发送两帧之间的变化列表。
|
||||||
|
|
||||||
|
注意玩家连接上服务端后不是立马加入游戏。连接开始时是观望者(spectator)模式,玩家可以观察其它玩家如何玩游戏。如果游戏已经开始或者上一个游戏会话已经在屏幕上显示 “game over” (游戏结束),用户此时可以按下 “Join”(参与),来加入一个已经存在的游戏,或者如果游戏没有运行(没有其它玩家)则创建一个新的游戏。后一种情况下,游戏区域在开始前会被先清空。
|
||||||
|
|
||||||
|
游戏区域存储在 `Game._field` 这个属性中,它是由嵌套列表组成的二维数组,用于内部存储游戏区域的状态。数组中的每一个元素表示区域中的一个小格,最终小格会被渲染成 html 表格的格子。它有一个 `Char` 的类型,是一个 `namedtuple` ,包括一个字符和颜色。在所有连接的客户端之间保证游戏区域的同步很重要,所以所有游戏区域的更新都必须依据发送到客户端的相应的信息。这是通过 `Game.apply_render()` 来实现的。它接受一个 `Draw` 对象的列表,其用于内部更新游戏区域和发送渲染消息给客户端。
|
||||||
|
|
||||||
|
> 我们使用 `namedtuple` 不仅因为它表示简单数据结构很方便,也因为用它生成 json 格式的消息时相对于 `dict` 更省空间。如果你在一个真实的游戏循环中需要发送复杂的数据结构,建议先将它们序列化成一个简单的、更短的格式,甚至打包成二进制格式(例如 bson,而不是 json),以减少网络传输。
|
||||||
|
|
||||||
|
`Player` 对象包括用 `deque` 对象表示的蛇。这种数据类型和 `list` 相似,但是在两端增加和删除元素时效率更高,用它来表示蛇很理想。它的主要方法是 `Player.render_move()`,它返回移动玩家的蛇至下一个位置的渲染数据。一般来说它在新的位置渲染蛇的头部,移除上一帧中表示蛇的尾巴的元素。如果蛇吃了一个数字变长了,在相应的多个帧中尾巴是不需要移动的。蛇的渲染数据在主类的 `Game.next_frame()` 中使用,该方法中实现所有的游戏逻辑。这个方法渲染所有蛇的移动,检查每一个蛇前面的障碍物,而且生成数字和“石头”。每一个“嘀嗒”,`game_loop()` 都会直接调用它来生成下一帧。
|
||||||
|
|
||||||
|
如果蛇头前面有障碍物,在 `Game.next_frame()` 中会调用 `Game.game_over()`。它后通知所有的客户端那个蛇死掉了 (会调用 `player.render_game_over()` 方法将其变成石头),然后更新表中的分数排行榜。`Player` 对象的 `alive` 标记被置为 `False`,当渲染下一帧时,这个玩家会被跳过,除非他重新加入游戏。当没有蛇存活时,游戏区域会显示 “game over” (游戏结束)。而且,主游戏循环会停止,设置 `game.running` 标记为 `False`。当某个玩家下次按下 “Join” (加入)时,游戏区域会被清空。
|
||||||
|
|
||||||
|
在渲染游戏的每个下一帧时也会产生数字和石头,它们是由随机值决定的。产生数字或者石头的概率可以在 `settings.py` 中修改成其它值。注意数字的产生是针对游戏区域每一个活的蛇的,所以蛇越多,产生的数字就越多,这样它们都有足够的食物来吃掉。
|
||||||
|
|
||||||
|
#### 4.4 网络协议
|
||||||
|
|
||||||
|
从客户端发送消息的列表:
|
||||||
|
|
||||||
|
命令 | 参数 |描述
|
||||||
|
:-- |:-- |:--
|
||||||
|
new_player | [name] |设置玩家的昵称
|
||||||
|
join | |玩家加入游戏
|
||||||
|
|
||||||
|
|
||||||
|
从服务端发送消息的列表:
|
||||||
|
|
||||||
|
命令 | 参数 |描述
|
||||||
|
:-- |:-- |:--
|
||||||
|
handshake |[id] |给一个玩家指定 ID
|
||||||
|
world |[[(char, color), ...], ...] |初始化游戏区域(世界地图)
|
||||||
|
reset_world | |清除实际地图,替换所有字符为空格
|
||||||
|
render |[x, y, char, color] |在某个位置显示字符
|
||||||
|
p_joined |[id, name, color, score] |新玩家加入游戏
|
||||||
|
p_gameover |[id] |某个玩家游戏结束
|
||||||
|
p_score |[id, score] |给某个玩家计分
|
||||||
|
top_scores |[[name, score, color], ...] |更新排行榜
|
||||||
|
|
||||||
|
典型的消息交换顺序:
|
||||||
|
|
||||||
|
客户端 -> 服务端 |服务端 -> 客户端 |服务端 -> 所有客户端 |备注
|
||||||
|
:-- |:-- |:-- |:--
|
||||||
|
new_player | | |名字传递给服务端
|
||||||
|
|handshake | |指定 ID
|
||||||
|
|world | |初始化传递的世界地图
|
||||||
|
|top_scores | |收到传递的排行榜
|
||||||
|
join | | |玩家按下“Join”,游戏循环开始
|
||||||
|
| |reset_world |命令客户端清除游戏区域
|
||||||
|
| |render, render, ... |第一个游戏“滴答”,渲染第一帧
|
||||||
|
(key code) | | |玩家按下一个键
|
||||||
|
| |render, render, ... |渲染第二帧
|
||||||
|
| |p_score |蛇吃掉了一个数字
|
||||||
|
| |render, render, ... |渲染第三帧
|
||||||
|
| | |... 重复若干帧 ...
|
||||||
|
| |p_gameover |试着吃掉障碍物时蛇死掉了
|
||||||
|
| |top_scores |更新排行榜(如果需要更新的话)
|
||||||
|
|
||||||
|
### 5. 总结
|
||||||
|
|
||||||
|
说实话,我十分享受 Python 最新的异步特性。新的语法做了改善,所以异步代码很容易阅读。可以明显看出哪些调用是非阻塞的,什么时候发生 greenthread 的切换。所以现在我可以宣称 Python 是异步编程的好工具。
|
||||||
|
|
||||||
|
SnakePit 在 7WebPages 团队中非常受欢迎。如果你在公司想休息一下,不要忘记给我们在 [Twitter][12] 或者 [Facebook][13] 留下反馈。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://7webpages.com/blog/writing-online-multiplayer-game-with-python-and-asyncio-part-3/
|
||||||
|
|
||||||
|
作者:[Kyrylo Subbotin][a]
|
||||||
|
译者:[chunyang-wen](https://github.com/chunyang-wen)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://7webpages.com/blog/writing-online-multiplayer-game-with-python-and-asyncio-part-3/
|
||||||
|
[1]: http://snakepit-game.com/
|
||||||
|
[2]: https://linux.cn/article-7784-1.html
|
||||||
|
[3]: https://linux.cn/article-7767-1.html
|
||||||
|
[4]: https://github.com/7WebPages/snakepit-game
|
||||||
|
[5]: https://github.com/7WebPages/snakepit-game
|
||||||
|
[6]: https://github.com/7WebPages/snakepit-game/blob/master/server.py
|
||||||
|
[7]: https://github.com/7WebPages/snakepit-game/blob/master/game.py
|
||||||
|
[8]: https://github.com/7WebPages/snakepit-game/blob/master/player.py
|
||||||
|
[9]: https://github.com/7WebPages/snakepit-game/blob/master/datatypes.py
|
||||||
|
[10]: https://github.com/7WebPages/snakepit-game/blob/master/settings.py
|
||||||
|
[11]: https://github.com/7WebPages/snakepit-game/blob/master/index.html
|
||||||
|
[12]: https://twitter.com/7WebPages
|
||||||
|
[13]: https://www.facebook.com/7WebPages/
|
@ -0,0 +1,464 @@
|
|||||||
|
从零构建一个简单的 Python 框架
|
||||||
|
===================================
|
||||||
|
|
||||||
|
为什么你想要自己构建一个 web 框架呢?我想,原因有以下几点:
|
||||||
|
|
||||||
|
- 你有一个新奇的想法,觉得将会取代其他的框架
|
||||||
|
- 你想要获得一些名气
|
||||||
|
- 你遇到的问题很独特,以至于现有的框架不太合适
|
||||||
|
- 你对 web 框架是如何工作的很感兴趣,因为你想要成为一位更好的 web 开发者。
|
||||||
|
|
||||||
|
接下来的笔墨将着重于最后一点。这篇文章旨在通过对设计和实现过程一步一步的阐述告诉读者,我在完成一个小型的服务器和框架之后学到了什么。你可以在这个[代码仓库][1]中找到这个项目的完整代码。
|
||||||
|
|
||||||
|
我希望这篇文章可以鼓励更多的人来尝试,因为这确实很有趣。它让我知道了 web 应用是如何工作的,而且这比我想的要容易的多!
|
||||||
|
|
||||||
|
### 范围
|
||||||
|
|
||||||
|
框架可以处理请求-响应周期、身份认证、数据库访问、模板生成等部分工作。Web 开发者使用框架是因为,大多数的 web 应用拥有大量相同的功能,而对每个项目都重新实现同样的功能意义不大。
|
||||||
|
|
||||||
|
比较大的的框架如 Rails 和 Django 实现了高层次的抽象,或者说“自备电池”(“batteries-included”,这是 Python 的口号之一,意即所有功能都自足。)。而实现所有的这些功能可能要花费数千小时,因此在这个项目上,我们重点完成其中的一小部分。在开始写代码前,我先列举一下所需的功能以及限制。
|
||||||
|
|
||||||
|
功能:
|
||||||
|
|
||||||
|
- 处理 HTTP 的 GET 和 POST 请求。你可以在[这篇 wiki][2] 中对 HTTP 有个大致的了解。
|
||||||
|
- 实现异步操作(我*喜欢* Python 3 的 asyncio 模块)。
|
||||||
|
- 简单的路由逻辑以及参数撷取。
|
||||||
|
- 像其他微型框架一样,提供一个简单的用户级 API 。
|
||||||
|
- 支持身份认证,因为学会这个很酷啊(微笑)。
|
||||||
|
|
||||||
|
限制:
|
||||||
|
|
||||||
|
- 将只支持 HTTP 1.1 的一个小子集,不支持传输编码(transfer-encoding)、HTTP 认证(http-auth)、内容编码(content-encoding,如 gzip)以及[持久化连接][3]等功能。
|
||||||
|
- 不支持对响应内容的 MIME 判断 - 用户需要手动指定。
|
||||||
|
- 不支持 WSGI - 仅能处理简单的 TCP 连接。
|
||||||
|
- 不支持数据库。
|
||||||
|
|
||||||
|
我觉得一个小的用例可以让上述内容更加具体,也可以用来演示这个框架的 API:
|
||||||
|
|
||||||
|
```
|
||||||
|
from diy_framework import App, Router
|
||||||
|
from diy_framework.http_utils import Response
|
||||||
|
|
||||||
|
|
||||||
|
# GET simple route
|
||||||
|
async def home(r):
|
||||||
|
rsp = Response()
|
||||||
|
rsp.set_header('Content-Type', 'text/html')
|
||||||
|
rsp.body = '<html><body><b>test</b></body></html>'
|
||||||
|
return rsp
|
||||||
|
|
||||||
|
|
||||||
|
# GET route + params
|
||||||
|
async def welcome(r, name):
|
||||||
|
return "Welcome {}".format(name)
|
||||||
|
|
||||||
|
# POST route + body param
|
||||||
|
async def parse_form(r):
|
||||||
|
if r.method == 'GET':
|
||||||
|
return 'form'
|
||||||
|
else:
|
||||||
|
name = r.body.get('name', '')[0]
|
||||||
|
password = r.body.get('password', '')[0]
|
||||||
|
|
||||||
|
return "{0}:{1}".format(name, password)
|
||||||
|
|
||||||
|
# application = router + http server
|
||||||
|
router = Router()
|
||||||
|
router.add_routes({
|
||||||
|
r'/welcome/{name}': welcome,
|
||||||
|
r'/': home,
|
||||||
|
r'/login': parse_form,})
|
||||||
|
|
||||||
|
app = App(router)
|
||||||
|
app.start_server()
|
||||||
|
```
|
||||||
|
'
|
||||||
|
用户需要定义一些能够返回字符串或 `Response` 对象的异步函数,然后将这些函数与表示路由的字符串配对,最后通过一个函数调用(`start_server`)开始处理请求。
|
||||||
|
|
||||||
|
完成设计之后,我将它抽象为几个我需要编码的部分:
|
||||||
|
|
||||||
|
- 接受 TCP 连接以及调度一个异步函数来处理这些连接的部分
|
||||||
|
- 将原始文本解析成某种抽象容器的部分
|
||||||
|
- 对于每个请求,用来决定调用哪个函数的部分
|
||||||
|
- 将上述部分集中到一起,并为开发者提供一个简单接口的部分
|
||||||
|
|
||||||
|
我先编写一些测试,这些测试被用来描述每个部分的功能。几次重构后,整个设计被分成若干部分,每个部分之间是相对解耦的。这样就非常好,因为每个部分可以被独立地研究学习。以下是我上文列出的抽象的具体体现:
|
||||||
|
|
||||||
|
- 一个 HTTPServer 对象,需要一个 Router 对象和一个 http_parser 模块,并使用它们来初始化。
|
||||||
|
- HTTPConnection 对象,每一个对象表示一个单独的客户端 HTTP 连接,并且处理其请求-响应周期:使用 http_parser 模块将收到的字节流解析为一个 Request 对象;使用一个 Router 实例寻找并调用正确的函数来生成一个响应;最后将这个响应发送回客户端。
|
||||||
|
- 一对 Request 和 Response 对象为用户提供了一种友好的方式,来处理实质上是字节流的字符串。用户不需要知道正确的消息格式和分隔符是怎样的。
|
||||||
|
- 一个包含“路由:函数”对应关系的 Router 对象。它提供一个添加配对的方法,可以根据 URL 路径查找到相应的函数。
|
||||||
|
- 最后,一个 App 对象。它包含配置信息,并使用它们实例化一个 HTTPServer 实例。
|
||||||
|
|
||||||
|
让我们从 `HTTPConnection` 开始来讲解各个部分。
|
||||||
|
|
||||||
|
### 模拟异步连接
|
||||||
|
|
||||||
|
为了满足上述约束条件,每一个 HTTP 请求都是一个单独的 TCP 连接。这使得处理请求的速度变慢了,因为建立多个 TCP 连接需要相对高的花销(DNS 查询,TCP 三次握手,[慢启动][4]等等的花销),不过这样更加容易模拟。对于这一任务,我选择相对高级的 [asyncio-stream][5] 模块,它建立在 [asyncio 的传输和协议][6]的基础之上。我强烈推荐你读一读标准库中的相应代码,很有意思!
|
||||||
|
|
||||||
|
一个 `HTTPConnection` 的实例能够处理多个任务。首先,它使用 `asyncio.StreamReader` 对象以增量的方式从 TCP 连接中读取数据,并存储在缓存中。每一个读取操作完成后,它会尝试解析缓存中的数据,并生成一个 `Request` 对象。一旦收到了这个完整的请求,它就生成一个回复,并通过 `asyncio.StreamWriter` 对象发送回客户端。当然,它还有两个任务:超时连接以及错误处理。
|
||||||
|
|
||||||
|
你可以在[这里][7]浏览这个类的完整代码。我将分别介绍代码的每一部分。为了简单起见,我移除了代码文档。
|
||||||
|
|
||||||
|
```
|
||||||
|
class HTTPConnection(object):
|
||||||
|
def init(self, http_server, reader, writer):
|
||||||
|
self.router = http_server.router
|
||||||
|
self.http_parser = http_server.http_parser
|
||||||
|
self.loop = http_server.loop
|
||||||
|
|
||||||
|
self._reader = reader
|
||||||
|
self._writer = writer
|
||||||
|
self._buffer = bytearray()
|
||||||
|
self._conn_timeout = None
|
||||||
|
self.request = Request()
|
||||||
|
```
|
||||||
|
|
||||||
|
这个 `init` 方法没啥意思,它仅仅是收集了一些对象以供后面使用。它存储了一个 `router` 对象、一个 `http_parser` 对象以及 `loop` 对象,分别用来生成响应、解析请求以及在事件循环中调度任务。
|
||||||
|
|
||||||
|
然后,它存储了代表一个 TCP 连接的读写对,和一个充当原始字节缓冲区的空[字节数组][8]。`_conn_timeout` 存储了一个 [asyncio.Handle][9] 的实例,用来管理超时逻辑。最后,它还存储了 `Request` 对象的一个单一实例。
|
||||||
|
|
||||||
|
下面的代码是用来接受和发送数据的核心功能:
|
||||||
|
|
||||||
|
```
|
||||||
|
async def handle_request(self):
|
||||||
|
try:
|
||||||
|
while not self.request.finished and not self._reader.at_eof():
|
||||||
|
data = await self._reader.read(1024)
|
||||||
|
if data:
|
||||||
|
self._reset_conn_timeout()
|
||||||
|
await self.process_data(data)
|
||||||
|
if self.request.finished:
|
||||||
|
await self.reply()
|
||||||
|
elif self._reader.at_eof():
|
||||||
|
raise BadRequestException()
|
||||||
|
except (NotFoundException,
|
||||||
|
BadRequestException) as e:
|
||||||
|
self.error_reply(e.code, body=Response.reason_phrases[e.code])
|
||||||
|
except Exception as e:
|
||||||
|
self.error_reply(500, body=Response.reason_phrases[500])
|
||||||
|
|
||||||
|
self.close_connection()
|
||||||
|
```
|
||||||
|
|
||||||
|
所有内容被包含在 `try-except` 代码块中,这样在解析请求或响应期间抛出的异常可以被捕获到,然后一个错误响应会发送回客户端。
|
||||||
|
|
||||||
|
在 `while` 循环中不断读取请求,直到解析器将 `self.request.finished` 设置为 True ,或者客户端关闭连接所触发的信号使得 `self._reader_at_eof()` 函数返回值为 True 为止。这段代码尝试在每次循环迭代中从 `StreamReader` 中读取数据,并通过调用 `self.process_data(data)` 函数以增量方式生成 `self.request`。每次循环读取数据时,连接超时计数器被重置。
|
||||||
|
|
||||||
|
这儿有个错误,你发现了吗?稍后我们会再讨论这个。需要注意的是,这个循环可能会耗尽 CPU 资源,因为如果没有读取到东西 `self._reader.read()` 函数将会返回一个空的字节对象 `b''`。这就意味着循环将会不断运行,却什么也不做。一个可能的解决方法是,用非阻塞的方式等待一小段时间:`await asyncio.sleep(0.1)`。我们暂且不对它做优化。
|
||||||
|
|
||||||
|
还记得上一段我提到的那个错误吗?只有从 `StreamReader` 读取数据时,`self._reset_conn_timeout()` 函数才会被调用。这就意味着,**直到第一个字节到达时**,`timeout` 才被初始化。如果有一个客户端建立了与服务器的连接却不发送任何数据,那就永远不会超时。这可能被用来消耗系统资源,从而导致拒绝服务式攻击(DoS)。修复方法就是在 `init` 函数中调用 `self._reset_conn_timeout()` 函数。
|
||||||
|
|
||||||
|
当请求接受完成或连接中断时,程序将运行到 `if-else` 代码块。这部分代码会判断解析器收到完整的数据后是否完成了解析。如果是,好,生成一个回复并发送回客户端。如果不是,那么请求信息可能有错误,抛出一个异常!最后,我们调用 `self.close_connection` 执行清理工作。
|
||||||
|
|
||||||
|
解析请求的部分在 `self.process_data` 方法中。这个方法非常简短,也易于测试:
|
||||||
|
|
||||||
|
```
|
||||||
|
async def process_data(self, data):
|
||||||
|
self._buffer.extend(data)
|
||||||
|
|
||||||
|
self._buffer = self.http_parser.parse_into(
|
||||||
|
self.request, self._buffer)
|
||||||
|
```
|
||||||
|
|
||||||
|
每一次调用都将数据累积到 `self._buffer` 中,然后试着用 `self.http_parser` 来解析已经收集的数据。这里需要指出的是,这段代码展示了一种称为[依赖注入(Dependency Injection)][10]的模式。如果你还记得 `init` 函数的话,应该知道我们传入了一个包含 `http_parser` 对象的 `http_server` 对象。在这个例子里,`http_parser` 对象是 `diy_framework` 包中的一个模块。不过它也可以是任何含有 `parse_into` 函数的类,这个 `parse_into` 函数接受一个 `Request` 对象以及字节数组作为参数。这很有用,原因有二:一是,这意味着这段代码更易扩展。如果有人想通过一个不同的解析器来使用 `HTTPConnection`,没问题,只需将它作为参数传入即可。二是,这使得测试更加容易,因为 `http_parser` 不是硬编码的,所以使用虚假数据或者 [mock][11] 对象来替代是很容易的。
|
||||||
|
|
||||||
|
下一段有趣的部分就是 `reply` 方法了:
|
||||||
|
|
||||||
|
```
|
||||||
|
async def reply(self):
|
||||||
|
request = self.request
|
||||||
|
handler = self.router.get_handler(request.path)
|
||||||
|
|
||||||
|
response = await handler.handle(request)
|
||||||
|
|
||||||
|
if not isinstance(response, Response):
|
||||||
|
response = Response(code=200, body=response)
|
||||||
|
|
||||||
|
self._writer.write(response.to_bytes())
|
||||||
|
await self._writer.drain()
|
||||||
|
```
|
||||||
|
|
||||||
|
这里,一个 `HTTPConnection` 的实例使用了 `HTTPServer` 中的 `router` 对象来得到一个生成响应的对象。一个路由可以是任何一个拥有 `get_handler` 方法的对象,这个方法接收一个字符串作为参数,返回一个可调用的对象或者抛出 `NotFoundException` 异常。而这个可调用的对象被用来处理请求以及生成响应。处理程序由框架的使用者编写,如上文所说的那样,应该返回字符串或者 `Response` 对象。`Response` 对象提供了一个友好的接口,因此这个简单的 if 语句保证了无论处理程序返回什么,代码最终都得到一个统一的 `Response` 对象。
|
||||||
|
|
||||||
|
接下来,被赋值给 `self._writer` 的 `StreamWriter` 实例被调用,将字节字符串发送回客户端。函数返回前,程序在 `await self._writer.drain()` 处等待,以确保所有的数据被发送给客户端。只要缓存中还有未发送的数据,`self._writer.close()` 方法就不会执行。
|
||||||
|
|
||||||
|
`HTTPConnection` 类还有两个更加有趣的部分:一个用于关闭连接的方法,以及一组用来处理超时机制的方法。首先,关闭一条连接由下面这个小函数完成:
|
||||||
|
|
||||||
|
```
|
||||||
|
def close_connection(self):
|
||||||
|
self._cancel_conn_timeout()
|
||||||
|
self._writer.close()
|
||||||
|
```
|
||||||
|
|
||||||
|
每当一条连接将被关闭时,这段代码首先取消超时,然后把连接从事件循环中清除。
|
||||||
|
|
||||||
|
超时机制由三个相关的函数组成:第一个函数在超时后给客户端发送错误消息并关闭连接;第二个函数用于取消当前的超时;第三个函数调度超时功能。前两个函数比较简单,我将详细解释第三个函数 `_reset_cpmm_timeout()` 。
|
||||||
|
|
||||||
|
```
|
||||||
|
def _conn_timeout_close(self):
|
||||||
|
self.error_reply(500, 'timeout')
|
||||||
|
self.close_connection()
|
||||||
|
|
||||||
|
def _cancel_conn_timeout(self):
|
||||||
|
if self._conn_timeout:
|
||||||
|
self._conn_timeout.cancel()
|
||||||
|
|
||||||
|
def _reset_conn_timeout(self, timeout=TIMEOUT):
|
||||||
|
self._cancel_conn_timeout()
|
||||||
|
self._conn_timeout = self.loop.call_later(
|
||||||
|
timeout, self._conn_timeout_close)
|
||||||
|
```
|
||||||
|
|
||||||
|
每当 `_reset_conn_timeout` 函数被调用时,它会先取消之前所有赋值给 `self._conn_timeout` 的 `asyncio.Handle` 对象。然后,使用 [BaseEventLoop.call_later][12] 函数让 `_conn_timeout_close` 函数在超时数秒(`timeout`)后执行。如果你还记得 `handle_request` 函数的内容,就知道每当接收到数据时,这个函数就会被调用。这就取消了当前的超时并且重新安排 `_conn_timeout_close` 函数在超时数秒(`timeout`)后执行。只要接收到数据,这个循环就会不断地重置超时回调。如果在超时时间内没有接收到数据,最后函数 `_conn_timeout_close` 就会被调用。
|
||||||
|
|
||||||
|
### 创建连接
|
||||||
|
|
||||||
|
我们需要创建 `HTTPConnection` 对象,并且正确地使用它们。这一任务由 `HTTPServer` 类完成。`HTTPServer` 类是一个简单的容器,可以存储着一些配置信息(解析器,路由和事件循环实例),并使用这些配置来创建 `HTTPConnection` 实例:
|
||||||
|
|
||||||
|
```
|
||||||
|
class HTTPServer(object):
|
||||||
|
def init(self, router, http_parser, loop):
|
||||||
|
self.router = router
|
||||||
|
self.http_parser = http_parser
|
||||||
|
self.loop = loop
|
||||||
|
|
||||||
|
async def handle_connection(self, reader, writer):
|
||||||
|
connection = HTTPConnection(self, reader, writer)
|
||||||
|
asyncio.ensure_future(connection.handle_request(), loop=self.loop)
|
||||||
|
```
|
||||||
|
|
||||||
|
`HTTPServer` 的每一个实例能够监听一个端口。它有一个 `handle_connection` 的异步方法来创建 `HTTPConnection` 的实例,并安排它们在事件循环中运行。这个方法被传递给 [asyncio.start_server][13] 作为一个回调函数。也就是说,每当一个 TCP 连接初始化时(以 `StreamReader` 和 `StreamWriter` 为参数),它就会被调用。
|
||||||
|
|
||||||
|
```
|
||||||
|
self._server = HTTPServer(self.router, self.http_parser, self.loop)
|
||||||
|
self._connection_handler = asyncio.start_server(
|
||||||
|
self._server.handle_connection,
|
||||||
|
host=self.host,
|
||||||
|
port=self.port,
|
||||||
|
reuse_address=True,
|
||||||
|
reuse_port=True,
|
||||||
|
loop=self.loop)
|
||||||
|
```
|
||||||
|
|
||||||
|
这就是构成整个应用程序工作原理的核心:`asyncio.start_server` 接受 TCP 连接,然后在一个预配置的 `HTTPServer` 对象上调用一个方法。这个方法将处理一条 TCP 连接的所有逻辑:读取、解析、生成响应并发送回客户端、以及关闭连接。它的重点是 IO 逻辑、解析和生成响应。
|
||||||
|
|
||||||
|
讲解了核心的 IO 部分,让我们继续。
|
||||||
|
|
||||||
|
### 解析请求
|
||||||
|
|
||||||
|
这个微型框架的使用者被宠坏了,不愿意和字节打交道。它们想要一个更高层次的抽象 —— 一种更加简单的方法来处理请求。这个微型框架就包含了一个简单的 HTTP 解析器,能够将字节流转化为 Request 对象。
|
||||||
|
|
||||||
|
这些 Request 对象是像这样的容器:
|
||||||
|
|
||||||
|
```
|
||||||
|
class Request(object):
|
||||||
|
def init(self):
|
||||||
|
self.method = None
|
||||||
|
self.path = None
|
||||||
|
self.query_params = {}
|
||||||
|
self.path_params = {}
|
||||||
|
self.headers = {}
|
||||||
|
self.body = None
|
||||||
|
self.body_raw = None
|
||||||
|
self.finished = False
|
||||||
|
```
|
||||||
|
|
||||||
|
它包含了所有需要的数据,可以用一种容易理解的方法从客户端接受数据。哦,不包括 cookie ,它对身份认证是非常重要的,我会将它留在第二部分。
|
||||||
|
|
||||||
|
每一个 HTTP 请求都包含了一些必需的内容,如请求路径和请求方法。它们也包含了一些可选的内容,如请求体、请求头,或是 URL 参数。随着 REST 的流行,除了 URL 参数,URL 本身会包含一些信息。比如,"/user/1/edit" 包含了用户的 id 。
|
||||||
|
|
||||||
|
一个请求的每个部分都必须被识别、解析,并正确地赋值给 Request 对象的对应属性。HTTP/1.1 是一个文本协议,事实上这简化了很多东西。(HTTP/2 是一个二进制协议,这又是另一种乐趣了)
|
||||||
|
|
||||||
|
解析器不需要跟踪状态,因此 `http_parser` 模块其实就是一组函数。调用函数需要用到 `Request` 对象,并将它连同一个包含原始请求信息的字节数组传递给 `parse_into` 函数。然后解析器会修改 `Request` 对象以及充当缓存的字节数组。字节数组的信息被逐渐地解析到 request 对象中。
|
||||||
|
|
||||||
|
`http_parser` 模块的核心功能就是下面这个 `parse_into` 函数:
|
||||||
|
|
||||||
|
```
|
||||||
|
def parse_into(request, buffer):
|
||||||
|
_buffer = buffer[:]
|
||||||
|
if not request.method and can_parse_request_line(_buffer):
|
||||||
|
(request.method, request.path,
|
||||||
|
request.query_params) = parse_request_line(_buffer)
|
||||||
|
remove_request_line(_buffer)
|
||||||
|
|
||||||
|
if not request.headers and can_parse_headers(_buffer):
|
||||||
|
request.headers = parse_headers(_buffer)
|
||||||
|
if not has_body(request.headers):
|
||||||
|
request.finished = True
|
||||||
|
|
||||||
|
remove_intro(_buffer)
|
||||||
|
|
||||||
|
if not request.finished and can_parse_body(request.headers, _buffer):
|
||||||
|
request.body_raw, request.body = parse_body(request.headers, _buffer)
|
||||||
|
clear_buffer(_buffer)
|
||||||
|
request.finished = True
|
||||||
|
return _buffer
|
||||||
|
```
|
||||||
|
|
||||||
|
从上面的代码中可以看到,我把解析的过程分为三个部分:解析请求行(这行像这样:GET /resource HTTP/1.1),解析请求头以及解析请求体。
|
||||||
|
|
||||||
|
请求行包含了 HTTP 请求方法以及 URL 地址。而 URL 地址则包含了更多的信息:路径、url 参数和开发者自定义的 url 参数。解析请求方法和 URL 还是很容易的 - 合适地分割字符串就好了。函数 `urlparse.parse` 可以用来解析 URL 参数。开发者自定义的 URL 参数可以通过正则表达式来解析。
|
||||||
|
|
||||||
|
接下来是 HTTP 头部。它们是一行行由键值对组成的简单文本。问题在于,可能有多个 HTTP 头有相同的名字,却有不同的值。一个值得关注的 HTTP 头部是 `Content-Length`,它描述了请求体的字节长度(不是整个请求,仅仅是请求体)。这对于决定是否解析请求体有很重要的作用。
|
||||||
|
|
||||||
|
最后,解析器根据 HTTP 方法和头部来决定是否解析请求体。
|
||||||
|
|
||||||
|
### 路由!
|
||||||
|
|
||||||
|
在某种意义上,路由就像是连接框架和用户的桥梁,用户用合适的方法创建 `Router` 对象并为其设置路径/函数对,然后将它赋值给 App 对象。而 App 对象依次调用 `get_handler` 函数生成相应的回调函数。简单来说,路由就负责两件事,一是存储路径/函数对,二是返回需要的路径/函数对
|
||||||
|
|
||||||
|
`Router` 类中有两个允许最终开发者添加路由的方法,分别是 `add_routes` 和 `add_route`。因为 `add_routes` 就是 `add_route` 函数的一层封装,我们将主要讲解 `add_route` 函数:
|
||||||
|
|
||||||
|
```
|
||||||
|
def add_route(self, path, handler):
|
||||||
|
compiled_route = self.class.build_route_regexp(path)
|
||||||
|
if compiled_route not in self.routes:
|
||||||
|
self.routes[compiled_route] = handler
|
||||||
|
else:
|
||||||
|
raise DuplicateRoute
|
||||||
|
```
|
||||||
|
|
||||||
|
首先,这个函数使用 `Router.build_router_regexp` 的类方法,将一条路由规则(如 '/cars/{id}' 这样的字符串),“编译”到一个已编译的正则表达式对象。这些已编译的正则表达式用来匹配请求路径,以及解析开发者自定义的 URL 参数。如果已经存在一个相同的路由,程序就会抛出一个异常。最后,这个路由/处理程序对被添加到一个简单的字典`self.routes`中。
|
||||||
|
|
||||||
|
下面展示 Router 是如何“编译”路由的:
|
||||||
|
|
||||||
|
```
|
||||||
|
@classmethod
|
||||||
|
def build_route_regexp(cls, regexp_str):
|
||||||
|
"""
|
||||||
|
Turns a string into a compiled regular expression. Parses '{}' into
|
||||||
|
named groups ie. '/path/{variable}' is turned into
|
||||||
|
'/path/(?P<variable>[a-zA-Z0-9_-]+)'.
|
||||||
|
|
||||||
|
:param regexp_str: a string representing a URL path.
|
||||||
|
:return: a compiled regular expression.
|
||||||
|
"""
|
||||||
|
def named_groups(matchobj):
|
||||||
|
return '(?P<{0}>[a-zA-Z0-9_-]+)'.format(matchobj.group(1))
|
||||||
|
|
||||||
|
re_str = re.sub(r'{([a-zA-Z0-9_-]+)}', named_groups, regexp_str)
|
||||||
|
re_str = ''.join(('^', re_str, '$',))
|
||||||
|
return re.compile(re_str)
|
||||||
|
```
|
||||||
|
|
||||||
|
这个方法使用正则表达式将所有出现的 `{variable}` 替换为 `(?P<variable>)`。然后在字符串头尾分别添加 `^` 和 `$` 标记,最后编译正则表达式对象。
|
||||||
|
|
||||||
|
完成了路由存储仅成功了一半,下面是如何得到路由对应的函数:
|
||||||
|
|
||||||
|
```
|
||||||
|
def get_handler(self, path):
|
||||||
|
logger.debug('Getting handler for: {0}'.format(path))
|
||||||
|
for route, handler in self.routes.items():
|
||||||
|
path_params = self.class.match_path(route, path)
|
||||||
|
if path_params is not None:
|
||||||
|
logger.debug('Got handler for: {0}'.format(path))
|
||||||
|
wrapped_handler = HandlerWrapper(handler, path_params)
|
||||||
|
return wrapped_handler
|
||||||
|
|
||||||
|
raise NotFoundException()
|
||||||
|
```
|
||||||
|
|
||||||
|
一旦 `App` 对象获得一个 `Request` 对象,也就获得了 URL 的路径部分(如 /users/15/edit)。然后,我们需要匹配函数来生成一个响应或者 404 错误。`get_handler` 函数将路径作为参数,循环遍历路由,对每条路由调用 `Router.match_path` 类方法检查是否有已编译的正则对象与这个请求路径匹配。如果存在,我们就调用 `HandleWrapper` 来包装路由对应的函数。`path_params` 字典包含了路径变量(如 '/users/15/edit' 中的 '15'),若路由没有指定变量,字典就为空。最后,我们将包装好的函数返回给 `App` 对象。
|
||||||
|
|
||||||
|
如果遍历了所有的路由都找不到与路径匹配的,函数就会抛出 `NotFoundException` 异常。
|
||||||
|
|
||||||
|
这个 `Route.match` 类方法挺简单:
|
||||||
|
|
||||||
|
```
|
||||||
|
def match_path(cls, route, path):
|
||||||
|
match = route.match(path)
|
||||||
|
try:
|
||||||
|
return match.groupdict()
|
||||||
|
except AttributeError:
|
||||||
|
return None
|
||||||
|
```
|
||||||
|
|
||||||
|
它使用正则对象的 [match 方法][14]来检查路由是否与路径匹配。若果不匹配,则返回 None 。
|
||||||
|
|
||||||
|
最后,我们有 `HandleWraapper` 类。它的唯一任务就是封装一个异步函数,存储 `path_params` 字典,并通过 `handle` 方法对外提供一个统一的接口。
|
||||||
|
|
||||||
|
```
|
||||||
|
class HandlerWrapper(object):
|
||||||
|
def init(self, handler, path_params):
|
||||||
|
self.handler = handler
|
||||||
|
self.path_params = path_params
|
||||||
|
self.request = None
|
||||||
|
|
||||||
|
async def handle(self, request):
|
||||||
|
return await self.handler(request, **self.path_params)
|
||||||
|
```
|
||||||
|
|
||||||
|
### 组合到一起
|
||||||
|
|
||||||
|
框架的最后部分就是用 `App` 类把所有的部分联系起来。
|
||||||
|
|
||||||
|
`App` 类用于集中所有的配置细节。一个 `App` 对象通过其 `start_server` 方法,使用一些配置数据创建一个 `HTTPServer` 的实例,然后将它传递给 [asyncio.start_server 函数][15]。`asyncio.start_server` 函数会对每一个 TCP 连接调用 `HTTPServer` 对象的 `handle_connection` 方法。
|
||||||
|
|
||||||
|
```
|
||||||
|
def start_server(self):
|
||||||
|
if not self._server:
|
||||||
|
self.loop = asyncio.get_event_loop()
|
||||||
|
self._server = HTTPServer(self.router, self.http_parser, self.loop)
|
||||||
|
self._connection_handler = asyncio.start_server(
|
||||||
|
self._server.handle_connection,
|
||||||
|
host=self.host,
|
||||||
|
port=self.port,
|
||||||
|
reuse_address=True,
|
||||||
|
reuse_port=True,
|
||||||
|
loop=self.loop)
|
||||||
|
|
||||||
|
logger.info('Starting server on {0}:{1}'.format(
|
||||||
|
self.host, self.port))
|
||||||
|
self.loop.run_until_complete(self._connection_handler)
|
||||||
|
|
||||||
|
try:
|
||||||
|
self.loop.run_forever()
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
logger.info('Got signal, killing server')
|
||||||
|
except DiyFrameworkException as e:
|
||||||
|
logger.error('Critical framework failure:')
|
||||||
|
logger.error(e.traceback)
|
||||||
|
finally:
|
||||||
|
self.loop.close()
|
||||||
|
else:
|
||||||
|
logger.info('Server already started - {0}'.format(self))
|
||||||
|
```
|
||||||
|
|
||||||
|
### 总结
|
||||||
|
|
||||||
|
如果你查看源码,就会发现所有的代码仅 320 余行(包括测试代码的话共 540 余行)。这么少的代码实现了这么多的功能,让我有点惊讶。这个框架没有提供模板、身份认证以及数据库访问等功能(这些内容也很有趣哦)。这也让我知道,像 Django 和 Tornado 这样的框架是如何工作的,而且我能够快速地调试它们了。
|
||||||
|
|
||||||
|
这也是我按照测试驱动开发完成的第一个项目,整个过程有趣而有意义。先编写测试用例迫使我思考设计和架构,而不仅仅是把代码放到一起,让它们可以运行。不要误解我的意思,有很多时候,后者的方式更好。不过如果你想给确保这些不怎么维护的代码在之后的几周甚至几个月依然工作,那么测试驱动开发正是你需要的。
|
||||||
|
|
||||||
|
我研究了下[整洁架构][16]以及依赖注入模式,这些充分体现在 `Router` 类是如何作为一个更高层次的抽象的(实体?)。`Router` 类是比较接近核心的,像 `http_parser` 和 `App` 的内容比较边缘化,因为它们只是完成了极小的字符串和字节流、或是中层 IO 的工作。测试驱动开发(TDD)迫使我独立思考每个小部分,这使我问自己这样的问题:方法调用的组合是否易于理解?类名是否准确地反映了我正在解决的问题?我的代码中是否很容易区分出不同的抽象层?
|
||||||
|
|
||||||
|
来吧,写个小框架,真的很有趣:)
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://mattscodecave.com/posts/simple-python-framework-from-scratch.html
|
||||||
|
|
||||||
|
作者:[Matt][a]
|
||||||
|
译者:[Cathon](https://github.com/Cathon)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://mattscodecave.com/hire-me.html
|
||||||
|
[1]: https://github.com/sirMackk/diy_framework
|
||||||
|
[2]:https://en.wikipedia.org/wiki/Hypertext_Transfer_Protocol
|
||||||
|
[3]: https://en.wikipedia.org/wiki/HTTP_persistent_connection
|
||||||
|
[4]: https://en.wikipedia.org/wiki/TCP_congestion-avoidance_algorithm#Slow_start
|
||||||
|
[5]: https://docs.python.org/3/library/asyncio-stream.html
|
||||||
|
[6]: https://docs.python.org/3/library/asyncio-protocol.html
|
||||||
|
[7]: https://github.com/sirMackk/diy_framework/blob/88968e6b30e59504251c0c7cd80abe88f51adb79/diy_framework/http_server.py#L46
|
||||||
|
[8]: https://docs.python.org/3/library/functions.html#bytearray
|
||||||
|
[9]: https://docs.python.org/3/library/asyncio-eventloop.html#asyncio.Handle
|
||||||
|
[10]: https://en.wikipedia.org/wiki/Dependency_injection
|
||||||
|
[11]: https://docs.python.org/3/library/unittest.mock.html
|
||||||
|
[12]: https://docs.python.org/3/library/asyncio-eventloop.html#asyncio.BaseEventLoop.call_later
|
||||||
|
[13]: https://docs.python.org/3/library/asyncio-stream.html#asyncio.start_server
|
||||||
|
[14]: https://docs.python.org/3/library/re.html#re.match
|
||||||
|
[15]: https://docs.python.org/3/library/asyncio-stream.html?highlight=start_server#asyncio.start_server
|
||||||
|
[16]: https://blog.8thlight.com/uncle-bob/2012/08/13/the-clean-architecture.html
|
@ -1,14 +1,13 @@
|
|||||||
|
科学音频处理(一):怎样使用 Octave 对音频文件进行读写操作
|
||||||
科学音频处理--怎样在 Ubuntu 上使用 Octave 对音频文件进行读写操作
|
|
||||||
================
|
================
|
||||||
|
|
||||||
Octave 是一个类似于 Linux 上的 Matlab 的软件,它有数量众多的函数和命令进行声音采集,记录,回放以及娱乐应用程序,研究,医学方面的音频信号的处理,这也包括任何其他的科学领域。在本教程中,我们会在 Ubuntu 上使用 Octave 的 4.0.0 版本读取音频文件,然后通过生成信号并且播放来模仿在各种情况下对音频信号的使用。
|
Octave 是一个类似于 Linux 上的 Matlab 的软件,它拥有数量众多的函数和命令,支持声音采集、记录、回放以及音频信号的数字化处理,用于娱乐应用、研究、医学以及其它科学领域。在本教程中,我们会在 Ubuntu 上使用 Octave 的 4.0.0 版本读取音频文件,然后通过生成信号并且播放来模仿在各种情况下对音频信号的使用。
|
||||||
|
|
||||||
本教程中关注的不是安装和学习使用已有的音频处理软件,而是从设计和音频工程的角度理解它是如何工作的。
|
本教程中关注的不是安装和学习使用安装好的音频处理软件,而是从设计和音频工程的角度理解它是如何工作的。
|
||||||
|
|
||||||
### 环境准备
|
### 环境准备
|
||||||
|
|
||||||
首先是安装 octave,在 Ubuntu 终端运行下面的命令添加 Octave PPA 然后安装 Octave 。
|
首先是安装 octave,在 Ubuntu 终端运行下面的命令添加 Octave PPA,然后安装 Octave 。
|
||||||
|
|
||||||
```
|
```
|
||||||
sudo apt-add-repository ppa:octave/stable
|
sudo apt-add-repository ppa:octave/stable
|
||||||
@ -16,7 +15,6 @@ sudo apt-get update
|
|||||||
sudo apt-get install octave
|
sudo apt-get install octave
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
### 步骤1:打开 Octave
|
### 步骤1:打开 Octave
|
||||||
|
|
||||||
在这一步中我们单击软件图标打开 Octave,可以通过单击下拉式按钮选择工作路径。
|
在这一步中我们单击软件图标打开 Octave,可以通过单击下拉式按钮选择工作路径。
|
||||||
@ -25,8 +23,8 @@ sudo apt-get install octave
|
|||||||
|
|
||||||
### 步骤2:音频信息
|
### 步骤2:音频信息
|
||||||
|
|
||||||
|
使用`audioinfo`命令查看要处理的音频文件的相关信息。
|
||||||
|
|
||||||
使用“audioinfo”命令查看要处理的音频文件的相关信息。
|
|
||||||
```
|
```
|
||||||
>> info = audioinfo ('testing.ogg')
|
>> info = audioinfo ('testing.ogg')
|
||||||
```
|
```
|
||||||
@ -36,8 +34,8 @@ sudo apt-get install octave
|
|||||||
|
|
||||||
### 步骤3:读取音频文件
|
### 步骤3:读取音频文件
|
||||||
|
|
||||||
|
在本教程中我会使用 ogg 文件来读取这种文件的属性,比如采样、音频类型(stereo 和 mono)、信道数量等。必须声明的一点是教程中使用的所有的命令都是在 Octave 终端窗口中执行的。首先,我们必须要把这个 ogg 文件赋给一个变量。注意:**文件必须在 Octave 的工作路径中。**
|
||||||
|
|
||||||
在本教程中我会使用 ogg 文件,因为读取这种文件的属性比较容易,比如采样,音频类型(stereo 和 mono),信道数量等。必须声明的一点是教程中使用的所有的命令都是在 Octave 终端窗口中执行的。首先,我们必须要把这个 ogg 文件赋给一个变量。注意:文件必须在 Octave 的工作路径中。
|
|
||||||
```
|
```
|
||||||
>> file='yourfile.ogg'
|
>> file='yourfile.ogg'
|
||||||
```
|
```
|
||||||
@ -46,8 +44,8 @@ sudo apt-get install octave
|
|||||||
>> [M, fs] = audioread(file)
|
>> [M, fs] = audioread(file)
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
这里的 M 是一个一列或两列的矩阵,取决于信道的数量,fs 是采样率。
|
这里的 M 是一个一列或两列的矩阵,取决于信道的数量,fs 是采样率。
|
||||||
|
|
||||||
![](https://www.howtoforge.com/images/how-to-read-and-write-audio-files-with-octave-4-in-ubuntu/reading.png)
|
![](https://www.howtoforge.com/images/how-to-read-and-write-audio-files-with-octave-4-in-ubuntu/reading.png)
|
||||||
|
|
||||||
![](https://www.howtoforge.com/images/how-to-read-and-write-audio-files-with-octave-4-in-ubuntu/matrix.png)
|
![](https://www.howtoforge.com/images/how-to-read-and-write-audio-files-with-octave-4-in-ubuntu/matrix.png)
|
||||||
@ -73,21 +71,19 @@ samples 指定开始帧和结束帧,datatype 指定返回的数据类型。可
|
|||||||
>> [y, fs] = audioread (filename, samples)
|
>> [y, fs] = audioread (filename, samples)
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
数据类型:
|
数据类型:
|
||||||
|
|
||||||
```
|
```
|
||||||
>> [y,Fs] = audioread(filename,'native')
|
>> [y,Fs] = audioread(filename,'native')
|
||||||
```
|
```
|
||||||
|
|
||||||
如果值是"native",那么它的数据类型就依数据在音频文件中的存储情况而定。
|
如果值是“native”,那么它的数据类型就依数据在音频文件中的存储情况而定。
|
||||||
|
|
||||||
### 步骤4:音频文件的写操作
|
### 步骤4:音频文件的写操作
|
||||||
|
|
||||||
新建一个 ogg 文件:
|
新建一个 ogg 文件:
|
||||||
|
|
||||||
|
我们会从一个余弦值创建一个 ogg 文件。采样率是每秒 44100 次,这个文件最少进行 10 秒的采样。余弦信号的频率是 440 Hz。
|
||||||
我们会创建一个带有余弦值的 ogg 文件。采样率是每秒 44100 次,这个文件最少进行 10 秒的采样。余弦信号的频率是 440 Hz。
|
|
||||||
|
|
||||||
```
|
```
|
||||||
>> filename='cosine.ogg';
|
>> filename='cosine.ogg';
|
||||||
@ -99,18 +95,17 @@ samples 指定开始帧和结束帧,datatype 指定返回的数据类型。可
|
|||||||
```
|
```
|
||||||
|
|
||||||
这就在工作路径中创建了一个 'cosine.ogg' 文件,这个文件中包含余弦信号。
|
这就在工作路径中创建了一个 'cosine.ogg' 文件,这个文件中包含余弦信号。
|
||||||
|
|
||||||
![](https://www.howtoforge.com/images/how-to-read-and-write-audio-files-with-octave-4-in-ubuntu/cosinefile.png)
|
![](https://www.howtoforge.com/images/how-to-read-and-write-audio-files-with-octave-4-in-ubuntu/cosinefile.png)
|
||||||
|
|
||||||
|
|
||||||
播放这个 'cosine.ogg' 文件就会产生一个 440Hz 的 音调,这个音调正好是乐理中的 'A' 调。如果需要查看保存在文件中的值就必须使用 'audioread' 函数读取文件。在后续的教程中,我们会看到怎样在两个信道中读取一个音频文件。
|
播放这个 'cosine.ogg' 文件就会产生一个 440Hz 的 音调,这个音调正好是乐理中的 'A' 调。如果需要查看保存在文件中的值就必须使用 'audioread' 函数读取文件。在后续的教程中,我们会看到怎样在两个信道中读取一个音频文件。
|
||||||
|
|
||||||
|
|
||||||
### 步骤5:播放音频文件
|
### 步骤5:播放音频文件
|
||||||
|
|
||||||
Octave 有一个默认的音频播放器,可以用这个音频播放器进行测试。使用下面的函数:
|
Octave 有一个默认的音频播放器,可以用这个音频播放器进行测试。使用下面的函数:
|
||||||
|
|
||||||
```
|
```
|
||||||
>> [y,fs]=audioread('yourfile.ogg');
|
>> [y,fs]=audioread('yourfile.ogg');
|
||||||
>> player=audioplayer(y, fs, 8)
|
>> player=audioplayer(y, fs, 8)
|
||||||
|
|
||||||
scalar structure containing the fields:
|
scalar structure containing the fields:
|
||||||
@ -129,7 +124,7 @@ Octave 有一个默认的音频播放器,可以用这个音频播放器进行
|
|||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
在教程的后续,我们会进入音频处理的高级特性部分,可能会接触到一些科学和商业应用中的实例。
|
在这个教程的续篇,我们会进入音频处理的高级特性部分,可能会接触到一些科学和商业应用中的实例。
|
||||||
|
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
@ -137,7 +132,7 @@ via: https://www.howtoforge.com/tutorial/how-to-read-and-write-audio-files-with-
|
|||||||
|
|
||||||
作者:[David Duarte][a]
|
作者:[David Duarte][a]
|
||||||
译者:[vim-kakali](https://github.com/vim-kakali)
|
译者:[vim-kakali](https://github.com/vim-kakali)
|
||||||
校对:[校对ID](https://github.com/校对ID)
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
@ -0,0 +1,114 @@
|
|||||||
|
5 个给 Linux 新手的最佳包管理器
|
||||||
|
=====================================================
|
||||||
|
|
||||||
|
一个 Linux 新用户应该知道他或她的进步源自于对 Linux 发行版的使用,而 Linux 发行版有好几种,并以不同的方式管理软件包。
|
||||||
|
|
||||||
|
在 Linux 中,包管理器非常重要,知道如何使用多种包管理器可以让你像一个高手一样活得很舒适,从在仓库下载软件、安装软件,到更新软件、处理依赖和删除软件是非常重要的,这也是Linux 系统管理的一个重要部分。
|
||||||
|
|
||||||
|
![](http://www.tecmint.com/wp-content/uploads/2016/06/Best-Linux-Package-Managers.png)
|
||||||
|
|
||||||
|
*最好的Linux包管理器*
|
||||||
|
|
||||||
|
成为一个 Linux 高手的一个标志是了解主要的 Linux 发行版如何处理包,在这篇文章中,我们应该看一些你在 Linux 上能找到的最佳的包管理器,
|
||||||
|
|
||||||
|
在这里,我们的主要重点是关于一些最佳包管理器的相关信息,但不是如何使用它们,这些留给你亲自发现。但我会提供一些有意义的链接,使用指南或更多。
|
||||||
|
|
||||||
|
### 1. DPKG - Debian 包管理系统(Debian Package Management System)
|
||||||
|
|
||||||
|
Dpkg 是 Debian Linux 家族的基础包管理系统,它用于安装、删除、存储和提供`.deb`包的信息。
|
||||||
|
|
||||||
|
这是一个低层面的工具,并且有多个前端工具可以帮助用户从远程的仓库获取包,或处理复杂的包关系的工具,包括如下:
|
||||||
|
|
||||||
|
- 参考:[15 个用于基于 Debian 的发行版的 “dpkg” 命令实例][1]
|
||||||
|
|
||||||
|
#### APT (高级打包工具(Advanced Packaging Tool))
|
||||||
|
|
||||||
|
这个是一个 dpkg 包管理系统的前端工具,它是一个非常受欢迎的、自由而强大的,有用的命令行包管理器系统。
|
||||||
|
|
||||||
|
Debian 及其衍生版,例如 Ubuntu 和 Linux Mint 的用户应该非常熟悉这个包管理工具。
|
||||||
|
|
||||||
|
想要了解它是如何工作的,你可以去看看下面这些 HOW TO 指南:
|
||||||
|
|
||||||
|
- 参考:[15 个怎样在 Ubuntu/Debian 上使用新的 APT 工具的例子][2]
|
||||||
|
- 参考:[25 个用于包管理的有用的 APT-GET 和 APT-CACHE 的基础命令][3]
|
||||||
|
|
||||||
|
#### Aptitude 包管理器
|
||||||
|
|
||||||
|
这个也是 Debian Linux 家族一个非常出名的命令行前端包管理工具,它工作方式类似 APT ,它们之间有很多可以比较的地方,不过,你应该两个都试试才知道哪个工作的更好。
|
||||||
|
|
||||||
|
它最初为 Debian 及其衍生版设计的,但是现在它的功能延伸到 RHEL 家族。你可以参考这个指南了解更多关于 APT 和 Aptitude。
|
||||||
|
|
||||||
|
- 参考:[APT 和 Aptitude 是什么?它们知道到底有什么不同?][4]
|
||||||
|
|
||||||
|
#### Synaptic 包管理器
|
||||||
|
|
||||||
|
Synaptic是一个基于GTK+的APT的可视化包管理器,对于一些不想使用命令行的用户,它非常好用。
|
||||||
|
|
||||||
|
### 2. RPM 红帽包管理器(Red Hat Package Manager)
|
||||||
|
|
||||||
|
这个是红帽创建的 Linux 基本标准(LSB)打包格式和基础包管理系统。基于这个底层系统,有多个前端包管理工具可供你使用,但我们应该只看那些最好的,那就是:
|
||||||
|
|
||||||
|
#### YUM (黄狗更新器,修改版(Yellowdog Updater, Modified))
|
||||||
|
|
||||||
|
这个是一个开源、流行的命令行包管理器,它是用户使用 RPM 的界面(之一)。你可以把它和 Debian Linux 系统中的 APT 进行对比,它和 APT 拥有相同的功能。你可以从这个 HOW TO 指南中的例子更加清晰的理解YUM:
|
||||||
|
|
||||||
|
- 参考:[20 个用于包管理的 YUM 命令][5]
|
||||||
|
|
||||||
|
#### DNF(优美的 Yum(Dandified Yum))
|
||||||
|
|
||||||
|
这个也是一个用于基于 RPM 的发行版的包管理器,Fedora 18 引入了它,它是下一代 YUM。
|
||||||
|
|
||||||
|
如果你用 Fedora 22 及更新版本,你肯定知道它是默认的包管理器。这里有一些链接,将为你提供更多关于 DNF 的信息和如何使用它。
|
||||||
|
|
||||||
|
- 参考:[DNF - 基于 RPM 的发行版的下一代通用包管理软件][6]
|
||||||
|
- 参考: [27 个管理 Fedora 软件包的 ‘DNF’ 命令例子][7]
|
||||||
|
|
||||||
|
### 3. Pacman 包管理器 – Arch Linux
|
||||||
|
|
||||||
|
这个是一个流行的、强大而易用的包管理器,它用于 Arch Linux 和其他的一些小众发行版。它提供了一些其他包管理器提供的基本功能,包括安装、自动解决依赖关系、升级、卸载和降级软件。
|
||||||
|
|
||||||
|
但是最大的用处是,它为 Arch 用户创建了一个简单易用的包管理方式。你可以阅读 [Pacman 概览][8],它会解释上面提到的一些功能。
|
||||||
|
|
||||||
|
### 4. Zypper 包管理器 – openSUSE
|
||||||
|
|
||||||
|
这个是一个使用 libzypp 库制作的用于 OpenSUSE 系统上的命令行包管理器,它的常用功能包括访问仓库、安装包、解决依赖问题和其他功能。
|
||||||
|
|
||||||
|
更重要的是,它也可以支持存储库扩展功能,如模式、补丁和产品。新的 OpenSUSE 用户可以参考下面的链接来掌控它。
|
||||||
|
|
||||||
|
- 参考:[45 个让你精通 openSUSE 包管理的 Zypper 命令][9]
|
||||||
|
|
||||||
|
### 5. Portage 包管理器 – Gentoo
|
||||||
|
|
||||||
|
这个是 Gentoo 的包管理器,当下不怎么流行的一个发行版,但是这并不阻止它成为 Linux 下最好的软件包管理器之一。
|
||||||
|
|
||||||
|
Portage 项目的主要目标是创建一个简单、无故障的包管理系统,包含向后兼容、自动化等功能。
|
||||||
|
|
||||||
|
如果希望理解的更清晰,可以看下: [Portage 项目页][10]。
|
||||||
|
|
||||||
|
### 结束语
|
||||||
|
|
||||||
|
正如我在开始时提到的,这个指南的主要意图是给 Linux 用户提供一个最佳软件包管理器的列表,但知道如何使用它们可以通过其后提供的重要的链接,并实际去试试它们。
|
||||||
|
|
||||||
|
各个发行版的用户需要学习超出他们的发行版之外的一些东西,才能更好理解上述提到的这些不同的包管理器。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.tecmint.com/linux-package-managers/
|
||||||
|
|
||||||
|
作者:[Ravi Saive][a]
|
||||||
|
译者:[Bestony](https://github.com/bestony)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创翻译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.tecmint.com/author/admin/
|
||||||
|
[1]: http://www.tecmint.com/dpkg-command-examples/
|
||||||
|
[2]: http://www.tecmint.com/apt-advanced-package-command-examples-in-ubuntu/
|
||||||
|
[3]: http://www.tecmint.com/useful-basic-commands-of-apt-get-and-apt-cache-for-package-management/
|
||||||
|
[4]: http://www.tecmint.com/difference-between-apt-and-aptitude/
|
||||||
|
[5]: http://www.tecmint.com/20-linux-yum-yellowdog-updater-modified-commands-for-package-mangement/
|
||||||
|
[6]: http://www.tecmint.com/dnf-next-generation-package-management-utility-for-linux/
|
||||||
|
[7]: http://www.tecmint.com/dnf-commands-for-fedora-rpm-package-management/
|
||||||
|
[8]: https://wiki.archlinux.org/index.php/Pacman
|
||||||
|
[9]: http://www.tecmint.com/zypper-commands-to-manage-suse-linux-package-management/
|
||||||
|
[10]: https://wiki.gentoo.org/wiki/Project:Portage
|
@ -0,0 +1,239 @@
|
|||||||
|
科学音频处理(二):如何使用 Octave 对音频文件进行基本数学信号处理
|
||||||
|
=========
|
||||||
|
|
||||||
|
在[前一篇的指导教程][1]中,我们看到了读、写以及重放音频文件的简单步骤,我们甚至看到如何从一个周期函数比如余弦函数合成一个音频文件。在这篇指导教程中,我们将会看到如何对信号进行叠加和倍乘(调整),并应用一些基本的数学函数看看它们对原始信号的影响。
|
||||||
|
|
||||||
|
### 信号叠加
|
||||||
|
|
||||||
|
两个信号 S1(t)和 S2(t)相加形成一个新的信号 R(t),这个信号在任何瞬间的值等于构成它的两个信号在那个时刻的值之和。就像下面这样:
|
||||||
|
|
||||||
|
```
|
||||||
|
R(t) = S1(t) + S2(t)
|
||||||
|
```
|
||||||
|
|
||||||
|
我们将用 Octave 重新产生两个信号的和并通过图表看达到的效果。首先,我们生成两个不同频率的信号,看一看它们的叠加信号是什么样的。
|
||||||
|
|
||||||
|
#### 第一步:产生两个不同频率的信号(oog 文件)
|
||||||
|
|
||||||
|
```
|
||||||
|
>> sig1='cos440.ogg'; %creating the audio file @440 Hz
|
||||||
|
>> sig2='cos880.ogg'; %creating the audio file @880 Hz
|
||||||
|
>> fs=44100; %generating the parameters values (Period, sampling frequency and angular frequency)
|
||||||
|
>> t=0:1/fs:0.02;
|
||||||
|
>> w1=2*pi*440*t;
|
||||||
|
>> w2=2*pi*880*t;
|
||||||
|
>> audiowrite(sig1,cos(w1),fs); %writing the function cos(w) on the files created
|
||||||
|
>> audiowrite(sig2,cos(w2),fs);
|
||||||
|
```
|
||||||
|
|
||||||
|
然后我们绘制出两个信号的图像。
|
||||||
|
|
||||||
|
**信号 1 的图像(440 赫兹)**
|
||||||
|
|
||||||
|
```
|
||||||
|
>> [y1, fs] = audioread(sig1);
|
||||||
|
>> plot(y1)
|
||||||
|
```
|
||||||
|
|
||||||
|
[![信号 1 的图像](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/plotsignal1.png)](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/big/plotsignal1.png)
|
||||||
|
|
||||||
|
**信号 2 的图像(880 赫兹)**
|
||||||
|
|
||||||
|
```
|
||||||
|
>> [y2, fs] = audioread(sig2);
|
||||||
|
>> plot(y2)
|
||||||
|
```
|
||||||
|
|
||||||
|
[![信号 2 的图像](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/plotsignal2.png)](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/big/plotsignal2.png)
|
||||||
|
|
||||||
|
#### 第二步:把两个信号叠加
|
||||||
|
|
||||||
|
现在我们展示一下前面步骤中产生的两个信号的和。
|
||||||
|
|
||||||
|
```
|
||||||
|
>> sumres=y1+y2;
|
||||||
|
>> plot(sumres)
|
||||||
|
```
|
||||||
|
|
||||||
|
叠加信号的图像
|
||||||
|
|
||||||
|
[![和信号的图像](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/plotsum.png)](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/big/plotsum.png)
|
||||||
|
|
||||||
|
**Octaver 中的效果**
|
||||||
|
|
||||||
|
在 Octaver 中,这个效果产生的声音是独特的,因为它可以仿真音乐家弹奏的低八度或者高八度音符(取决于内部程序设计),仿真音符和原始音符成对,也就是两个音符发出相同的声音。
|
||||||
|
|
||||||
|
#### 第三步:把两个真实的信号相加(比如两首音乐歌曲)
|
||||||
|
|
||||||
|
为了实现这个目的,我们使用格列高利圣咏(Gregorian Chants)中的两首歌曲(声音采样)。
|
||||||
|
|
||||||
|
**圣母颂曲(Avemaria Track)**
|
||||||
|
|
||||||
|
首先,我们看一下圣母颂曲并绘出它的图像:
|
||||||
|
|
||||||
|
```
|
||||||
|
>> [y1,fs]=audioread('avemaria_.ogg');
|
||||||
|
>> plot(y1)
|
||||||
|
```
|
||||||
|
|
||||||
|
[![圣母歌曲](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/avemaria.png)](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/big/avemaria.png)
|
||||||
|
|
||||||
|
**赞美诗曲(Hymnus Track)**
|
||||||
|
|
||||||
|
现在我们看一下赞美诗曲并绘出它的图像。
|
||||||
|
|
||||||
|
```
|
||||||
|
>> [y2,fs]=audioread('hymnus.ogg');
|
||||||
|
>> plot(y2)
|
||||||
|
```
|
||||||
|
|
||||||
|
[![赞美诗曲](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/hymnus.png)](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/big/hymnus.png)
|
||||||
|
|
||||||
|
**圣母颂曲 + 赞美诗曲**
|
||||||
|
|
||||||
|
```
|
||||||
|
>> y='avehymnus.ogg';
|
||||||
|
>> audiowrite(y, y1+y2, fs);
|
||||||
|
>> [y, fs]=audioread('avehymnus.ogg');
|
||||||
|
>> plot(y)
|
||||||
|
```
|
||||||
|
|
||||||
|
[![圣母歌曲 + 赞美诗曲](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/avehymnus.png)](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/big/avehymnus.png)
|
||||||
|
|
||||||
|
结果,从音频的角度来看,两个声音信号混合在了一起。
|
||||||
|
|
||||||
|
### 两个信号的乘积
|
||||||
|
|
||||||
|
对于求两个信号的乘积,我们可以使用类似求和的方法。我们使用之前生成的相同文件。
|
||||||
|
|
||||||
|
```
|
||||||
|
R(t) = S1(t) * S2(t)
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
>> sig1='cos440.ogg'; %creating the audio file @440 Hz
|
||||||
|
>> sig2='cos880.ogg'; %creating the audio file @880 Hz
|
||||||
|
>> product='prod.ogg'; %creating the audio file for product
|
||||||
|
>> fs=44100; %generating the parameters values (Period, sampling frequency and angular frequency)
|
||||||
|
>> t=0:1/fs:0.02;
|
||||||
|
>> w1=2*pi*440*t;
|
||||||
|
>> w2=2*pi*880*t;
|
||||||
|
>> audiowrite(sig1, cos(w1), fs); %writing the function cos(w) on the files created
|
||||||
|
>> audiowrite(sig2, cos(w2), fs);>> [y1,fs]=audioread(sig1);>> [y2,fs]=audioread(sig2);
|
||||||
|
>> audiowrite(product, y1.*y2, fs); %performing the product
|
||||||
|
>> [yprod,fs]=audioread(product);
|
||||||
|
>> plot(yprod); %plotting the product
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
注意:我们必须使用操作符 ‘.*’,因为在参数文件中,这个乘积是值与值相乘。更多信息,请参考 Octave 矩阵操作产品手册。
|
||||||
|
|
||||||
|
#### 乘积生成信号的图像
|
||||||
|
|
||||||
|
[![乘积信号的图像](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/plotprod.png)](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/big/plotprod.png)
|
||||||
|
|
||||||
|
#### 两个基本频率相差很大的信号相乘后的图表效果(调制原理)
|
||||||
|
|
||||||
|
**第一步:**
|
||||||
|
|
||||||
|
生成两个频率为 220 赫兹的声音信号。
|
||||||
|
|
||||||
|
```
|
||||||
|
>> fs=44100;
|
||||||
|
>> t=0:1/fs:0.03;
|
||||||
|
>> w=2*pi*220*t;
|
||||||
|
>> y1=cos(w);
|
||||||
|
>> plot(y1);
|
||||||
|
```
|
||||||
|
|
||||||
|
[![载波](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/carrier.png)](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/big/carrier.png)
|
||||||
|
|
||||||
|
**第二步:**
|
||||||
|
|
||||||
|
生成一个 22000 赫兹的高频调制信号。
|
||||||
|
|
||||||
|
```
|
||||||
|
>> y2=cos(100*w);
|
||||||
|
>> plot(y2);
|
||||||
|
```
|
||||||
|
|
||||||
|
[![调制中](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/modulating.png)](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/big/modulating.png)
|
||||||
|
|
||||||
|
**第三步:**
|
||||||
|
|
||||||
|
把两个信号相乘并绘出图像。
|
||||||
|
|
||||||
|
```
|
||||||
|
>> plot(y1.*y2);
|
||||||
|
```
|
||||||
|
|
||||||
|
[![调制后的信号](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/modulated.png)](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/big/modulated.png)
|
||||||
|
|
||||||
|
### 一个信号和一个标量相乘
|
||||||
|
|
||||||
|
一个函数和一个标量相乘的效果等于更改它的值域,在某些情况下,更改的是相标志。给定一个标量 K ,一个函数 F(t) 和这个标量相乘定义为:
|
||||||
|
|
||||||
|
```
|
||||||
|
R(t) = K*F(t)
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
>> [y,fs]=audioread('cos440.ogg'); %creating the work files
|
||||||
|
>> res1='coslow.ogg';
|
||||||
|
>> res2='coshigh.ogg';>> res3='cosinverted.ogg';
|
||||||
|
>> K1=0.2; %values of the scalars
|
||||||
|
>> K2=0.5;>> K3=-1;
|
||||||
|
>> audiowrite(res1, K1*y, fs); %product function-scalar
|
||||||
|
>> audiowrite(res2, K2*y, fs);
|
||||||
|
>> audiowrite(res3, K3*y, fs);
|
||||||
|
```
|
||||||
|
|
||||||
|
**原始信号的图像**
|
||||||
|
|
||||||
|
```
|
||||||
|
>> plot(y)
|
||||||
|
```
|
||||||
|
|
||||||
|
[![](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/originalsignal.png)](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/big/originalsignal.png)
|
||||||
|
|
||||||
|
**信号振幅减为原始信号振幅的 0.2 倍后的图像**
|
||||||
|
|
||||||
|
```
|
||||||
|
>> plot(res1)
|
||||||
|
```
|
||||||
|
|
||||||
|
[![低余弦](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/coslow.png)](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/big/coslow.png)
|
||||||
|
|
||||||
|
**信号振幅减为原始振幅的 0.5 倍后的图像**
|
||||||
|
|
||||||
|
```
|
||||||
|
>> plot(res2)
|
||||||
|
```
|
||||||
|
|
||||||
|
[![高余弦](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/coshigh.png)](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/big/coshigh.png)
|
||||||
|
|
||||||
|
**倒相后的信号图像**
|
||||||
|
|
||||||
|
```
|
||||||
|
>> plot(res3)
|
||||||
|
```
|
||||||
|
|
||||||
|
[![倒相余弦](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/cosinverted.png)](https://www.howtoforge.com/images/octave-audio-signal-processing-ubuntu/big/cosinverted.png)
|
||||||
|
|
||||||
|
### 结论
|
||||||
|
|
||||||
|
基本数学运算比如代数和、乘,以及函数与常量相乘是更多高级运算比如谱分析、振幅调制,角调制等的支柱和基础。在下一个教程中,我们来看一看如何进行这样的运算以及它们对声音文件产生的效果。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://www.howtoforge.com/tutorial/octave-audio-signal-processing-ubuntu/
|
||||||
|
|
||||||
|
作者:[David Duarte][a]
|
||||||
|
译者:[ucasFL](https://github.com/ucasFL)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://www.howtoforge.com/tutorial/octave-audio-signal-processing-ubuntu/
|
||||||
|
[1]: https://linux.cn/article-7755-1.html
|
||||||
|
|
@ -0,0 +1,135 @@
|
|||||||
|
你必须了解的基础的 Linux 网络命令
|
||||||
|
==================================================
|
||||||
|
|
||||||
|
![](https://itsfoss.com/wp-content/uploads/2016/06/Basic-Networking-Commands-Linux.jpg)
|
||||||
|
|
||||||
|
> 摘要:有抱负的 Linux 系统管理员和 Linux 狂热者必须知道的、最重要的、而且基础的 Linux 网络命令合集。
|
||||||
|
|
||||||
|
在 It’s FOSS 我们并非每天都谈论 Linux 的“命令行方面”。基本上,我更专注于 Linux 的桌面端。但你们读者中的一些人在内部调查(仅面向 It's FOSS newsletter 订阅者)中指出,你们也想学些命令行技巧。速查表也受大部分读者所喜欢和支持。
|
||||||
|
|
||||||
|
为此,我编辑了一个 Linux 中基础网络命令的列表。它并不是一个教你如何使用这些命令的教程,而是一个命令合集和他们的简短解释。所以,如果你已经使用过这些命令,你可以用它来快速记住命令。
|
||||||
|
|
||||||
|
你可以把这个网页添加为书签以便快速查阅,或输出一个 PDF 版本以便离线使用。
|
||||||
|
|
||||||
|
当我还是通信系统工程专业的学生的时候我就有这个 Linux 网络命令的列表了。它帮助我在计算机网络课程获得了高分。希望它也能以同样的方式帮助你。
|
||||||
|
|
||||||
|
### Linux 基础网络命令列表
|
||||||
|
|
||||||
|
我在计算机网络课程上使用 FreeBSD,不过这些 UNIX 命令应该也能在 Linux 上同样工作。
|
||||||
|
|
||||||
|
#### 连通性
|
||||||
|
|
||||||
|
- `ping <host>`:发送 ICMP echo 消息(一个包)到主机。这可能会不停地发送直到你按下 `Control-C`。Ping 的通意味着一个包从你的机器通过 ICMP 发送出去,并在 IP 层回显。Ping 告诉你另一个主机是否在运行。
|
||||||
|
- `telnet <host> [port]`:与主机在指定的端口通信。默认的 telnet 端口是 23。按 Control-] 以退出 telnet。其它一些常用的端口是:
|
||||||
|
- 7 —— echo 端口
|
||||||
|
- 25 —— SMTP,用于发送邮件
|
||||||
|
- 79 —— Finger (LCTT 译注:[维基百科 - Finger protocal](https://en.wikipedia.org/wiki/Finger_protocol),不过举例 Finger 恐怕不合时宜,倒不如试试 80?),提供该网络下其它用户的信息。
|
||||||
|
|
||||||
|
#### ARP
|
||||||
|
|
||||||
|
ARP 用于将 IP 地址转换为以太网地址。root 用户可以添加和删除 ARP 记录。当 ARP 记录被污染或者错误时,删除它们会有用。root 显式添加的 ARP 记录是永久的 —— 代理设置的也是。ARP 表保存在内核中,动态地被操作。ARP 记录会被缓存,通常在 20 分钟后失效并被删除。
|
||||||
|
|
||||||
|
- `arp -a`:打印 ARP 表。
|
||||||
|
- `arp -s <ip_address> <mac_address> [pub]`:添加一条记录到表中。
|
||||||
|
- `arp -a -d`:删除 ARP 表中的所有记录。
|
||||||
|
|
||||||
|
#### 路由
|
||||||
|
|
||||||
|
- `netstat -r`:打印路由表。路由表保存在内核中,用于 IP 层把包路由到非本地网络。
|
||||||
|
- `route add`:route 命令用于向路由表添加静态(手动指定而非动态)路由路径。所有从该 PC 到那个 IP/子网的流量都会经由指定的网关 IP。它也可以用来设置一个默认路由。例如,在 IP/子网处使用 0.0.0.0,就可以发送所有包到特定的网关。
|
||||||
|
- `routed`:控制动态路由的 BSD 守护程序。开机时启动。它运行 RIP 路由协议。只有 root 用户可用。没有 root 权限你不能运行它。
|
||||||
|
- `gated`:gated 是另一个使用 RIP 协议的路由守护进程。它同时支持 OSPF、EGP 和 RIP 协议。只有 root 用户可用。
|
||||||
|
- `traceroute`:用于跟踪 IP 包的路由。它每次发送包时都把跳数加 1,从而使得从源地址到目的地之间的所有网关都会返回消息。
|
||||||
|
- `netstat -rnf inet`:显示 IPv4 的路由表。
|
||||||
|
- `sysctl net.inet.ip.forwarding=1`:启用包转发(把主机变为路由器)。
|
||||||
|
- `route add|delete [-net|-host] <destination> <gateway>`:(如 `route add 192.168.20.0/24 192.168.30.4`)添加一条路由。
|
||||||
|
- `route flush`:删除所有路由。
|
||||||
|
- `route add -net 0.0.0.0 192.168.10.2`:添加一条默认路由。
|
||||||
|
- `routed -Pripv2 -Pno_rdisc -d [-s|-q]`:运行 routed 守护进程,使用 RIPv2 协议,不启用 ICMP 自动发现,在前台运行,供给模式或安静模式。
|
||||||
|
- `route add 224.0.0.0/4 127.0.0.1`:为本地地址定义多播路由。(LCTT 译注:原文存疑)
|
||||||
|
- `rtquery -n <host>`(LCTT 译注:增加了 host 参数):查询指定主机上的 RIP 守护进程(手动更新路由表)。
|
||||||
|
|
||||||
|
#### 其它
|
||||||
|
|
||||||
|
- `nslookup`:向 DNS 服务器查询,将 IP 转为名称,或反之。例如,`nslookup facebook.com` 会给出 facebook.com 的 IP。
|
||||||
|
- `ftp <host> [port]`(LCTT 译注:原文中 water 应是笔误):传输文件到指定主机。通常可以使用 登录名 "anonymous" , 密码 "guest" 来登录。
|
||||||
|
- `rlogin -l <host>`(LCTT 译注:添加了 host 参数):使用类似 telnet 的虚拟终端登录到主机。
|
||||||
|
|
||||||
|
#### 重要文件
|
||||||
|
|
||||||
|
- `/etc/hosts`:域名到 IP 地址的映射。
|
||||||
|
- `/etc/networks`:网络名称到 IP 地址的映射。
|
||||||
|
- `/etc/protocols`:协议名称到协议编号的映射。
|
||||||
|
- `/etc/services`:TCP/UDP 服务名称到端口号的映射。
|
||||||
|
|
||||||
|
#### 工具和网络性能分析
|
||||||
|
|
||||||
|
- `ifconfig <interface> <address> [up]`:启动接口。
|
||||||
|
- `ifconfig <interface> [down|delete]`:停止接口。
|
||||||
|
- `ethereal &`:在后台打开 `ethereal` 而非前台。
|
||||||
|
- `tcpdump -i -vvv`:抓取和分析包的工具。
|
||||||
|
- `netstat -w [seconds] -I [interface]`:显示网络设置和统计信息。
|
||||||
|
- `udpmt -p [port] -s [bytes] target_host`:发送 UDP 流量。
|
||||||
|
- `udptarget -p [port]`:接收 UDP 流量。
|
||||||
|
- `tcpmt -p [port] -s [bytes] target_host`:发送 TCP 流量。
|
||||||
|
- `tcptarget -p [port]`:接收 TCP 流量。
|
||||||
|
|
||||||
|
|
||||||
|
#### 交换机
|
||||||
|
|
||||||
|
- `ifconfig sl0 srcIP dstIP`:配置一个串行接口(在此前先执行 `slattach -l /dev/ttyd0`,此后执行 `sysctl net.inet.ip.forwarding=1`)
|
||||||
|
- `telnet 192.168.0.254`:从子网中的一台主机访问交换机。
|
||||||
|
- `sh ru` 或 `show running-configuration`:查看当前配置。
|
||||||
|
- `configure terminal`:进入配置模式。
|
||||||
|
- `exit`:退出当前模式。(LCTT 译注:原文存疑)
|
||||||
|
|
||||||
|
#### VLAN
|
||||||
|
|
||||||
|
- `vlan n`:创建一个 ID 为 n 的 VLAN。
|
||||||
|
- `no vlan N`:删除 ID 为 n 的 VLAN。
|
||||||
|
- `untagged Y`:添加端口 Y 到 VLAN n。
|
||||||
|
- `ifconfig vlan0 create`:创建 vlan0 接口。
|
||||||
|
- `ifconfig vlan0 vlan_ID vlandev em0`:把 em0 加入到 vlan0 接口(LCTT 译注:原文存疑),并设置标记为 ID。
|
||||||
|
- `ifconfig vlan0 [up]`:启用虚拟接口。
|
||||||
|
- `tagged Y`:为当前 VLAN 的端口 Y 添加标记帧支持。
|
||||||
|
|
||||||
|
#### UDP/TCP
|
||||||
|
|
||||||
|
- `socklab udp`:使用 UDP 协议运行 `socklab`。
|
||||||
|
- `sock`:创建一个 UDP 套接字,等效于输入 `sock udp` 和 `bind`。
|
||||||
|
- `sendto <Socket ID> <hostname> <port #>`:发送数据包。
|
||||||
|
- `recvfrom <Socket ID> <byte #>`:从套接字接收数据。
|
||||||
|
- `socklab tcp`:使用 TCP 协议运行 `socklab`。
|
||||||
|
- `passive`:创建一个被动模式的套接字,等效于 `socklab`,`sock tcp`,`bind`,`listen`。
|
||||||
|
- `accept`:接受进来的连接(可以在发起进来的连接之前或之后执行)。
|
||||||
|
- `connect <hostname> <port #>`:等效于 `socklab`,`sock tcp`,`bind`,`connect`。
|
||||||
|
- `close`:关闭连接。
|
||||||
|
- `read <byte #>`:从套接字中读取 n 字节。
|
||||||
|
- `write`:(例如,`write ciao`、`write #10`)向套接字写入 "ciao" 或 10 个字节。
|
||||||
|
|
||||||
|
#### NAT/防火墙
|
||||||
|
|
||||||
|
- `rm /etc/resolv.conf`:禁止地址解析,保证你的过滤和防火墙规则正确工作。
|
||||||
|
- `ipnat -f file_name`:将过滤规则写入文件。
|
||||||
|
- `ipnat -l`:显示活动的规则列表。
|
||||||
|
- `ipnat -C -F`:重新初始化规则表。
|
||||||
|
- `map em0 192.168.1.0/24 -> 195.221.227.57/32 em0`:将 IP 地址映射到接口。
|
||||||
|
- `map em0 192.168.1.0/24 -> 195.221.227.57/32 portmap tcp/udp 20000:50000`:带端口号的映射。
|
||||||
|
- `ipf -f file_name`:将过滤规则写入文件。
|
||||||
|
- `ipf -F -a`:重置规则表。
|
||||||
|
- `ipfstat -I`:当与 -s 选项合用时列出活动的状态条目(LCTT 译注:原文存疑)。
|
||||||
|
|
||||||
|
希望这份基础的 Linux 网络命令合集对你有用。欢迎各种问题和建议。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://itsfoss.com/basic-linux-networking-commands
|
||||||
|
|
||||||
|
作者:[Abhishek Prakash][a]
|
||||||
|
译者:[bianjp](https://github.com/bianjp)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://itsfoss.com/author/abhishek/
|
||||||
|
[1]: https://drive.google.com/open?id=0By49_3Av9sT1cDdaZnh4cHB4aEk
|
@ -0,0 +1,46 @@
|
|||||||
|
DAISY : 一种 Linux 上可用的服务于视力缺陷者的文本格式
|
||||||
|
=================================================================
|
||||||
|
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/images/education/osdc-lead_books.png?itok=K8wqfPT5)
|
||||||
|
|
||||||
|
*图片: 由Kate Ter Haar提供图片。 opensource.com 后期修饰。 CC BY-SA 2.0 *
|
||||||
|
|
||||||
|
如果你是盲人或像我一样有视力障碍,你可能经常需要各种软硬件才能做到视觉正常的人们视之为理所当然的事情。这其中之一就是阅读的印刷图书的专用格式:布莱叶盲文(Braille)(假设你知道怎样阅读它)或特殊的文本格式例如DAISY。
|
||||||
|
|
||||||
|
### DAISY 是什么?
|
||||||
|
|
||||||
|
DAISY 是数字化无障碍信息系统(Digital Accessible Information System)的缩写。 它是一种开放的标准,专用于帮助盲人阅读课本、杂志、报纸、小说,以及你想到的各种东西。 它由[ DAISY 联盟][1]创立于上世纪 90 年代中期,该联盟包括的组织们致力于制定出一套标准,可以让以这种方式标记的文本易于阅读、可以跳转、进行注释以及其它的文本操作,就像视觉正常的人能做的一样。
|
||||||
|
|
||||||
|
当前的 DAISY 3.0 版本发布于 2005 年中期,是一个完全重写了的标准。它创建的目的是更容易撰写遵守该规范的书籍。值得注意的是,DAISY 能够仅支持纯文本、或仅是录音(PCM wave 文件格式或者 MP3 格式)、或既有文本也有录音。特殊的软件能阅读这类书,并支持用户设置书签和目录导航,就像正常人阅读印刷书籍一样。
|
||||||
|
|
||||||
|
### DAISY 是怎样工作的呢?
|
||||||
|
|
||||||
|
DAISY,除开特殊的版本,它工作时有点像这样:你拥有自己的主向导文件(在 DAISY 2.02 中是 ncc.html),它包含书籍的元数据,比如作者姓名、版权信息、书籍页数等等。而在 DAISY 3.0 中这个文件是一个有效的 XML 文件,以及一个被强烈建议包含在每一本书中的 DTD(文档类型定义)文件。
|
||||||
|
|
||||||
|
在导航控制文件中,标记精确描述了各个位置——无论是文本导航中当前光标位置还是录音中的毫秒级定位,这让该软件可以跳到确切的位置,就像视力健康的人翻到某个章节一样。值得注意的是这种导航控制文件仅包含书中主要的、最大的书籍组成部分的位置。
|
||||||
|
|
||||||
|
更小的内容组成部分由 SMIL(同步多媒体集成语言(synchronized multimedia integration language))文件处理。导航的层次很大程度上取决于书籍的标记的怎么样。这样设想一下,如果印刷书籍没有章节标题,你就需要花很多的时间来确定自己阅读的位置。如果一本 DAISY 格式的书籍被标记的很差,你可能只能转到书本的开头或者目录。如果书籍被标记的太差了(或者完全没有标记),你的 DAISY 阅读软件很可能会直接忽略它。
|
||||||
|
|
||||||
|
### 为什么需要专门的软件?
|
||||||
|
|
||||||
|
你可能会问,如果 DAISY 仅仅是 HTML、XML、录音文件,为什么还需要使用专门的软件进行阅读和操作。单纯从技术上而言,你并不需要。专业化的软件大多数情况下是为了方便。这就像在 Linux 操作系统中,一个简单的 Web 浏览器可以被用来打开并阅读书籍。如果你在一本 DAISY 3 的书中点击 XML 文件,软件通常做的就是读取那些你赋予访问权限的书籍的名称,并建立一个列表让你点击选择要打开的书。如果书籍被标记的很差,它不会显示在这份清单中。
|
||||||
|
|
||||||
|
创建 DAISY 则完全是另一件事了,通常需要专门的软件,或需要拥有足够的专业知识来修改一个通用的软件以达到这样的目的。
|
||||||
|
|
||||||
|
### 结语
|
||||||
|
|
||||||
|
幸运的是,DAISY 是一个已确立的标准。虽然它在阅读方面表现的很棒,但是需要特殊软件来生产它使得视力缺陷者孤立于正常人眼中的世界,在那里人们可以以各种格式去阅读他们电子化书籍。这就是 DAISY 联盟在 EPUB 格式取得了 DAISY 成功的原因,它的第 3 版支持一种叫做“媒体覆盖”的规范,基本上来说是在 EPUB 电子书中可选增加声频或视频。由于 EPUB 和 DAISY 共享了很多 XML 标记,一些能够阅读 DAISY 的软件能够看到 EPUB 电子书但不能阅读它们。这也就意味着只要网站为我们换到这种开放格式的书籍,我们将会有更多可选的软件来阅读我们的书籍。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://opensource.com/life/16/5/daisy-linux-compatible-text-format-visually-impaired
|
||||||
|
|
||||||
|
作者:[Kendell Clark][a]
|
||||||
|
译者:[theArcticOcean](https://github.com/theArcticOcean)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创翻译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://opensource.com/users/kendell-clark
|
||||||
|
[1]: http://www.daisy.org
|
@ -0,0 +1,110 @@
|
|||||||
|
Ubuntu 的 Snap、Red Hat 的 Flatpak 这种通吃所有发行版的打包方式真的有用吗?
|
||||||
|
=================================================================================
|
||||||
|
|
||||||
|
![](http://www.iwillfolo.com/wordpress/wp-content/uploads/2016/06/Flatpak-and-Snap-Packages.jpg)
|
||||||
|
|
||||||
|
**对新一代的打包格式开始渗透到 Linux 生态系统中的深入观察**
|
||||||
|
|
||||||
|
最近我们听到越来越多的有关于 Ubuntu 的 Snap 包和由 Red Hat 员工 Alexander Larsson 创造的 Flatpak (曾经叫做 xdg-app)的消息。
|
||||||
|
|
||||||
|
这两种下一代打包方法在本质上拥有相同的目标和特点:即不依赖于第三方系统功能库的独立包装。
|
||||||
|
|
||||||
|
这种 Linux 新技术方向似乎自然会让人脑海中浮现这样的问题:独立包的优点/缺点是什么?这是否让我们拥有更好的 Linux 系统?其背后的动机是什么?
|
||||||
|
|
||||||
|
为了回答这些问题,让我们先深入了解一下 Snap 和 Flatpak。
|
||||||
|
|
||||||
|
### 动机
|
||||||
|
|
||||||
|
根据 [Flatpak][1] 和 [Snap][2] 的声明,背后的主要动机是使同一版本的应用程序能够运行在多个 Linux 发行版。
|
||||||
|
|
||||||
|
> “从一开始它的主要目标是允许相同的应用程序运行在各种 Linux 发行版和操作系统上。” —— Flatpak
|
||||||
|
|
||||||
|
> “……‘snap’ 通用 Linux 包格式,使简单的二进制包能够完美的、安全的运行在任何 Linux 桌面、服务器、云和设备上。” —— Snap
|
||||||
|
|
||||||
|
说得更具体一点,站在 Snap 和 Flatpak (以下称之为 S&F)背后的人认为,Linux 平台存在碎片化的问题。
|
||||||
|
|
||||||
|
这个问题导致了开发者们需要做许多不必要的工作来使他的软件能够运行在各种不同的发行版上,这影响了整个平台的前进。
|
||||||
|
|
||||||
|
所以,作为 Linux 发行版(Ubuntu 和 Red Hat)的领导者,他们希望消除这个障碍,推动平台发展。
|
||||||
|
|
||||||
|
但是,是否是更多的个人收益刺激了 S&F 的开发?
|
||||||
|
|
||||||
|
#### 个人收益?
|
||||||
|
|
||||||
|
虽然没有任何官方声明,但是试想一下,如果能够创造这种可能会被大多数发行版(即便不是全部)所采用的打包方式,那么这个项目的领导者将可能成为一个能够决定 Linux 大船航向的重要人物。
|
||||||
|
|
||||||
|
### 优势
|
||||||
|
|
||||||
|
这种独立包的好处多多,并且取决于不同的因素。
|
||||||
|
|
||||||
|
这些因素基本上可以归为两类:
|
||||||
|
|
||||||
|
#### 用户角度
|
||||||
|
|
||||||
|
**+** 从 Liunx 用户的观点来看:Snap 和 Flatpak 带来了将任何软件包(软件或应用)安装在用户使用的任何发行版上的可能性。
|
||||||
|
|
||||||
|
例如你在使用一个不是很流行的发行版,由于开发工作的缺乏,它的软件仓库只有很稀少的包。现在,通过 S&F 你就可以显著的增加包的数量,这是一个多么美好的事情。
|
||||||
|
|
||||||
|
**+** 同样,对于使用流行的发行版的用户,即使该发行版的软件仓库上有很多的包,他也可以在不改变它现有的功能库的同时安装一个新的包。
|
||||||
|
|
||||||
|
比方说, 一个 Debian 的用户想要安装一个 “测试分支” 的包,但是他又不想将他的整个系统变成测试版(来让该包运行在更新的功能库上)。现在,他就可以简单的想安装哪个版本就安装哪个版本,而不需要考虑库的问题。
|
||||||
|
|
||||||
|
对于持后者观点的人,可能基本上都是使用源文件编译他们的包的人,然而,除非你使用类似 Gentoo 这样基于源代码的发行版,否则大多数用户将从头编译视为是一个恶心到吐的事情。
|
||||||
|
|
||||||
|
**+** 高级用户,或者称之为 “拥有安全意识的用户” 可能会觉得更容易接受这种类型的包,只要它们来自可靠来源,这种包倾向于提供另一层隔离,因为它们通常是与系统包想隔离的。
|
||||||
|
|
||||||
|
\* 不论是 Snap 还是 Flatpak 都在不断努力增强它们的安全性,通常他们都使用 “沙盒化” 来隔离,以防止它们可能携带病毒感染整个系统,就像微软 Windows 系统中的 .exe 程序一样。(关于微软和 S&F 后面还会谈到)
|
||||||
|
|
||||||
|
#### 开发者角度
|
||||||
|
|
||||||
|
与普通用户相比,对于开发者来说,开发 S&F 包的优点可能更加清楚。这一点已经在上一节有所提示。
|
||||||
|
|
||||||
|
尽管如此,这些优点有:
|
||||||
|
|
||||||
|
**+** S&F 通过统一开发的过程,将多发行版的开发变得简单了起来。对于需要将他的应用运行在多个发行版的开发者来说,这大大的减少了他们的工作量。
|
||||||
|
|
||||||
|
**++** 因此,开发者能够更容易的使他的应用运行在更多的发行版上。
|
||||||
|
|
||||||
|
**+** S&F 允许开发者私自发布他的包,不需要依靠发行版维护者在每一个/每一次发行版中发布他的包。
|
||||||
|
|
||||||
|
**++** 通过上述方法,开发者可以不依赖发行版而直接获取到用户安装和卸载其软件的统计数据。
|
||||||
|
|
||||||
|
**++** 同样是通过上述方法,开发者可以更好的直接与用户互动,而不需要通过中间媒介,比如发行版这种中间媒介。
|
||||||
|
|
||||||
|
### 缺点
|
||||||
|
|
||||||
|
**–** 膨胀。就是这么简单。Flatpak 和 Snap 并不是凭空变出来它的依赖关系。相反,它是通过将依赖关系预构建在其中来代替使用系统中的依赖关系。
|
||||||
|
|
||||||
|
就像谚语说的:“山不来就我,我就去就山”。
|
||||||
|
|
||||||
|
**–** 之前提到安全意识强的用户会喜欢 S&F 提供的额外的一层隔离,只要该应用来自一个受信任的来源。但是从另外一个角度看,对这方面了解较少的用户,可能会从一个不靠谱的地方弄来一个包含恶意软件的包从而导致危害。
|
||||||
|
|
||||||
|
上面提到的观点可以说是有很有意义的,虽说今天的流行方法,像 PPA、overlay 等也可能是来自不受信任的来源。
|
||||||
|
|
||||||
|
但是,S&F 包更加增加这个风险,因为恶意软件开发者只需要开发一个版本就可以感染各种发行版。相反,如果没有 S&F,恶意软件的开发者就需要创建不同的版本以适应不同的发行版。
|
||||||
|
|
||||||
|
### 原来微软一直是正确的吗?
|
||||||
|
|
||||||
|
考虑到上面提到的,很显然,在大多数情况下,使用 S&F 包的优点超过缺点。
|
||||||
|
|
||||||
|
至少对于二进制发行版的用户,或者重点不是轻量级的发行版的用户来说是这样的。
|
||||||
|
|
||||||
|
这促使我问出这个问题,可能微软一直是正确的吗?如果是的,那么当 S&F 变成 Linux 的标准后,你还会一如既往的使用 Linux 或者类 Unix 系统吗?
|
||||||
|
|
||||||
|
很显然,时间会是这个问题的最好答案。
|
||||||
|
|
||||||
|
不过,我认为,即使不完全正确,但是微软有些地方也是值得赞扬的,并且以我的观点来看,所有这些方式在 Linux 上都立马能用也确实是一个亮点。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.iwillfolo.com/ubuntus-snap-red-hats-flatpack-and-is-one-fits-all-linux-packages-useful/
|
||||||
|
|
||||||
|
作者:[Editorials][a]
|
||||||
|
译者:[Chao-zhi](https://github.com/Chao-zhi)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创翻译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.iwillfolo.com/category/editorials/
|
||||||
|
[1]: http://flatpak.org/press/2016-06-21-flatpak-released.html
|
||||||
|
[2]: https://insights.ubuntu.com/2016/06/14/universal-snap-packages-launch-on-multiple-linux-distros
|
@ -0,0 +1,404 @@
|
|||||||
|
旅行时通过树莓派和 iPad Pro 备份图片
|
||||||
|
===================================================================
|
||||||
|
|
||||||
|
![](http://www.movingelectrons.net/images/bkup_photos_main.jpg)
|
||||||
|
|
||||||
|
*旅行中备份图片 - 组件*
|
||||||
|
|
||||||
|
### 介绍
|
||||||
|
|
||||||
|
我在很长的时间内一直在寻找一个旅行中备份图片的理想方法,把 SD 卡放进你的相机包会让你暴露在太多的风险之中:SD 卡可能丢失或者被盗,数据可能损坏或者在传输过程中失败。比较好的一个选择是复制到另外一个介质中,即使它也是个 SD 卡,并且将它放到一个比较安全的地方去,备份到远端也是一个可行的办法,但是如果去了一个没有网络的地方就不太可行了。
|
||||||
|
|
||||||
|
我理想的备份步骤需要下面的工具:
|
||||||
|
|
||||||
|
1. 用一台 iPad pro 而不是一台笔记本。我喜欢轻装旅行,我的大部分旅程都是商务相关的(而不是拍摄休闲的),我痛恨带着个人笔记本的时候还得带着商务本。而我的 iPad 却一直带着,这就是我为什么选择它的原因。
|
||||||
|
2. 用尽可能少的硬件设备。
|
||||||
|
3. 设备之间的连接需要很安全。我需要在旅馆和机场使用这套设备,所以设备之间的连接需要是封闭而加密的。
|
||||||
|
4. 整个过程应该是可靠稳定的,我还用过其他的路由器/组合设备,但是[效果不太理想][1]。
|
||||||
|
|
||||||
|
### 设备
|
||||||
|
|
||||||
|
我配置了一套满足上面条件并且在未来可以扩充的设备,它包含下面这些部件的使用:
|
||||||
|
|
||||||
|
1. [9.7 英寸的 iPad Pro][2],这是本文写作时最强大、轻薄的 iOS 设备,苹果笔不是必需的,但是作为零件之一,当我在路上可以做一些编辑工作,所有的重活由树莓派做 ,其他设备只能通过 SSH 连接就行。
|
||||||
|
2. 安装了 Raspbian 操作系统[树莓派 3][3](LCTT 译注:Raspbian 是基于 Debian 的树莓派操作系统)。
|
||||||
|
3. 树莓派的 [Mini SD卡][4] 和 [盒子/外壳][5]。
|
||||||
|
5. [128G 的优盘][6],对于我是够用了,你可以买个更大的。你也可以买个像[这样][7]的移动硬盘,但是树莓派没法通过 USB 给移动硬盘提供足够的电量,这意味你需要额外准备一个[供电的 USB hub][8] 以及电缆,这就破坏了我们让设备轻薄的初衷。
|
||||||
|
6. [SD 读卡器][9]
|
||||||
|
7. [另外的 SD 卡][10],我会使用几个 SD 卡,在用满之前就会立即换一个,这样就会让我在一次旅途当中的照片散布在不同的 SD 卡上。
|
||||||
|
|
||||||
|
下图展示了这些设备之间如何相互连接。
|
||||||
|
|
||||||
|
![](http://www.movingelectrons.net/images/bkup_photos_diag.jpg)
|
||||||
|
|
||||||
|
*旅行时照片的备份-流程图*
|
||||||
|
|
||||||
|
树莓派会作为一个安全的热点。它会创建一个自己的 WPA2 加密的 WIFI 网络,iPad Pro 会连入其中。虽然有很多在线教程教你如何创建 Ad Hoc 网络(计算机到计算机的单对单网络),还更简单一些,但是它的连接是不加密的,而且附件的设备很容易就能连接进去。因此我选择创建 WIFI 网络。
|
||||||
|
|
||||||
|
相机的 SD 卡通过 SD 读卡器插到树莓派 USB 端口之一,128G 的大容量优盘一直插在树莓派的另外一个 USB 端口上,我选择了一款[闪迪的][11],因为体积比较小。主要的思路就是通过 Python 脚本把 SD 卡的照片备份到优盘上,备份过程是增量备份,每次脚本运行时都只有变化的(比如新拍摄的照片)部分会添加到备份文件夹中,所以这个过程特别快。如果你有很多的照片或者拍摄了很多 RAW 格式的照片,在就是个巨大的优势。iPad 将用来运行 Python 脚本,而且用来浏览 SD 卡和优盘的文件。
|
||||||
|
|
||||||
|
作为额外的好处,如果给树莓派连上一根能上网的网线(比如通过以太网口),那么它就可以共享互联网连接给那些通过 WIFI 连入的设备。
|
||||||
|
|
||||||
|
### 1. 树莓派的设置
|
||||||
|
|
||||||
|
这部分需要你卷起袖子亲自动手了,我们要用到 Raspbian 的命令行模式,我会尽可能详细的介绍,方便大家进行下去。
|
||||||
|
|
||||||
|
#### 安装和配置 Raspbian
|
||||||
|
|
||||||
|
给树莓派连接鼠标、键盘和 LCD 显示器,将 SD 卡插到树莓派上,按照[树莓派官网][12]的步骤安装 Raspbian。
|
||||||
|
|
||||||
|
安装完后,打开 Raspbian 的终端,执行下面的命令:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt-get update
|
||||||
|
sudo apt-get upgrade
|
||||||
|
```
|
||||||
|
|
||||||
|
这将升级机器上所有的软件到最新,我将树莓派连接到本地网络,而且为了安全更改了默认的密码。
|
||||||
|
|
||||||
|
Raspbian 默认开启了 SSH,这样所有的设置可以在一个远程的设备上完成。我也设置了 RSA 验证,但这是可选的功能,可以在[这里][13]查看更多信息。
|
||||||
|
|
||||||
|
这是一个在 Mac 上在 [iTerm][14] 里建立 SSH 连接到树莓派上的截图[14]。(LCTT 译注:原文图丢失。)
|
||||||
|
|
||||||
|
#### 建立 WPA2 加密的 WIFI AP
|
||||||
|
|
||||||
|
安装过程基于[这篇文章][15],根据我的情况进行了调整。
|
||||||
|
|
||||||
|
**1. 安装软件包**
|
||||||
|
|
||||||
|
我们需要安装下面的软件包:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt-get install hostapd
|
||||||
|
sudo apt-get install dnsmasq
|
||||||
|
```
|
||||||
|
|
||||||
|
hostapd 用来使用内置的 WiFi 来创建 AP,dnsmasp 是一个组合的 DHCP 和 DNS 服务其,很容易设置。
|
||||||
|
|
||||||
|
**2. 编辑 dhcpcd.conf**
|
||||||
|
|
||||||
|
通过以太网连接树莓派,树莓派上的网络接口配置由 `dhcpd` 控制,因此我们首先忽略这一点,将 `wlan0` 设置为一个静态的 IP。
|
||||||
|
|
||||||
|
用 `sudo nano /etc/dhcpcd.conf` 命令打开 dhcpcd 的配置文件,在最后一行添加上如下内容:
|
||||||
|
|
||||||
|
```
|
||||||
|
denyinterfaces wlan0
|
||||||
|
```
|
||||||
|
|
||||||
|
注意:它必须放在如果已经有的其它接口行**之上**。
|
||||||
|
|
||||||
|
**3. 编辑接口**
|
||||||
|
|
||||||
|
现在设置静态 IP,使用 `sudo nano /etc/network/interfaces` 打开接口配置文件,按照如下信息编辑`wlan0`部分:
|
||||||
|
|
||||||
|
```
|
||||||
|
allow-hotplug wlan0
|
||||||
|
iface wlan0 inet static
|
||||||
|
address 192.168.1.1
|
||||||
|
netmask 255.255.255.0
|
||||||
|
network 192.168.1.0
|
||||||
|
broadcast 192.168.1.255
|
||||||
|
# wpa-conf /etc/wpa_supplicant/wpa_supplicant.conf
|
||||||
|
```
|
||||||
|
|
||||||
|
同样,然后 `wlan1` 编辑如下:
|
||||||
|
|
||||||
|
```
|
||||||
|
#allow-hotplug wlan1
|
||||||
|
#iface wlan1 inet manual
|
||||||
|
# wpa-conf /etc/wpa_supplicant/wpa_supplicant.conf
|
||||||
|
```
|
||||||
|
|
||||||
|
重要: 使用 `sudo service dhcpcd restart` 命令重启 `dhcpd`服务,然后用 `sudo ifdown eth0; sudo ifup wlan0` 命令来重载`wlan0`的配置。
|
||||||
|
|
||||||
|
**4. 配置 Hostapd**
|
||||||
|
|
||||||
|
接下来,我们需要配置 hostapd,使用 `sudo nano /etc/hostapd/hostapd.conf` 命令创建一个新的配置文件,内容如下:
|
||||||
|
|
||||||
|
```
|
||||||
|
interface=wlan0
|
||||||
|
|
||||||
|
# Use the nl80211 driver with the brcmfmac driver
|
||||||
|
driver=nl80211
|
||||||
|
|
||||||
|
# This is the name of the network
|
||||||
|
ssid=YOUR_NETWORK_NAME_HERE
|
||||||
|
|
||||||
|
# Use the 2.4GHz band
|
||||||
|
hw_mode=g
|
||||||
|
|
||||||
|
# Use channel 6
|
||||||
|
channel=6
|
||||||
|
|
||||||
|
# Enable 802.11n
|
||||||
|
ieee80211n=1
|
||||||
|
|
||||||
|
# Enable QoS Support
|
||||||
|
wmm_enabled=1
|
||||||
|
|
||||||
|
# Enable 40MHz channels with 20ns guard interval
|
||||||
|
ht_capab=[HT40][SHORT-GI-20][DSSS_CCK-40]
|
||||||
|
|
||||||
|
# Accept all MAC addresses
|
||||||
|
macaddr_acl=0
|
||||||
|
|
||||||
|
# Use WPA authentication
|
||||||
|
auth_algs=1
|
||||||
|
|
||||||
|
# Require clients to know the network name
|
||||||
|
ignore_broadcast_ssid=0
|
||||||
|
|
||||||
|
# Use WPA2
|
||||||
|
wpa=2
|
||||||
|
|
||||||
|
# Use a pre-shared key
|
||||||
|
wpa_key_mgmt=WPA-PSK
|
||||||
|
|
||||||
|
# The network passphrase
|
||||||
|
wpa_passphrase=YOUR_NEW_WIFI_PASSWORD_HERE
|
||||||
|
|
||||||
|
# Use AES, instead of TKIP
|
||||||
|
rsn_pairwise=CCMP
|
||||||
|
```
|
||||||
|
|
||||||
|
配置完成后,我们需要告诉`dhcpcd` 在系统启动运行时到哪里寻找配置文件。 使用 `sudo nano /etc/default/hostapd` 命令打开默认配置文件,然后找到`#DAEMON_CONF=""` 替换成`DAEMON_CONF="/etc/hostapd/hostapd.conf"`。
|
||||||
|
|
||||||
|
**5. 配置 Dnsmasq**
|
||||||
|
|
||||||
|
自带的 dnsmasp 配置文件包含很多信息方便你使用它,但是我们不需要那么多选项,我建议把它移动到别的地方(而不要删除它),然后自己创建一个新文件:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo mv /etc/dnsmasq.conf /etc/dnsmasq.conf.orig
|
||||||
|
sudo nano /etc/dnsmasq.conf
|
||||||
|
```
|
||||||
|
|
||||||
|
粘贴下面的信息到新文件中:
|
||||||
|
|
||||||
|
```
|
||||||
|
interface=wlan0 # Use interface wlan0
|
||||||
|
listen-address=192.168.1.1 # Explicitly specify the address to listen on
|
||||||
|
bind-interfaces # Bind to the interface to make sure we aren't sending things elsewhere
|
||||||
|
server=8.8.8.8 # Forward DNS requests to Google DNS
|
||||||
|
domain-needed # Don't forward short names
|
||||||
|
bogus-priv # Never forward addresses in the non-routed address spaces.
|
||||||
|
dhcp-range=192.168.1.50,192.168.1.100,12h # Assign IP addresses in that range with a 12 hour lease time
|
||||||
|
```
|
||||||
|
|
||||||
|
**6. 设置 IPv4 转发**
|
||||||
|
|
||||||
|
最后我们需要做的事就是配置包转发,用 `sudo nano /etc/sysctl.conf` 命令打开 `sysctl.conf` 文件,将包含 `net.ipv4.ip_forward=1`的那一行之前的#号删除,它将在下次重启时生效。
|
||||||
|
|
||||||
|
我们还需要给连接到树莓派的设备通过 WIFI 分享互联网连接,做一个 `wlan0`和 `eth0` 之间的 NAT。我们可以参照下面的脚本来实现。
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo iptables -t nat -A POSTROUTING -o eth0 -j MASQUERADE
|
||||||
|
sudo iptables -A FORWARD -i eth0 -o wlan0 -m state --state RELATED,ESTABLISHED -j ACCEPT
|
||||||
|
sudo iptables -A FORWARD -i wlan0 -o eth0 -j ACCEPT
|
||||||
|
```
|
||||||
|
|
||||||
|
我命名这个脚本名为 `hotspot-boot.sh`,然后让它可以执行:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo chmod 755 hotspot-boot.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
该脚本应该在树莓派启动的时候运行。有很多方法实现,下面是我实现的方式:
|
||||||
|
|
||||||
|
1. 把文件放到`/home/pi/scripts`目录下。
|
||||||
|
2. 输入`sudo nano /etc/rc.local`命令编辑 `rc.local` 文件,将运行该脚本的命令放到 `exit 0`之前。(更多信息参照[这里][16])。
|
||||||
|
|
||||||
|
编辑后`rc.local`看起来像这样:
|
||||||
|
|
||||||
|
```
|
||||||
|
#!/bin/sh -e
|
||||||
|
#
|
||||||
|
# rc.local
|
||||||
|
#
|
||||||
|
# This script is executed at the end of each multiuser runlevel.
|
||||||
|
# Make sure that the script will "exit 0" on success or any other
|
||||||
|
# value on error.
|
||||||
|
#
|
||||||
|
# In order to enable or disable this script just change the execution
|
||||||
|
# bits.
|
||||||
|
#
|
||||||
|
# By default this script does nothing.
|
||||||
|
|
||||||
|
# Print the IP address
|
||||||
|
_IP=$(hostname -I) || true
|
||||||
|
if [ "$_IP" ]; then
|
||||||
|
printf "My IP address is %s\n" "$_IP"
|
||||||
|
fi
|
||||||
|
|
||||||
|
sudo /home/pi/scripts/hotspot-boot.sh &
|
||||||
|
|
||||||
|
exit 0
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 安装 Samba 服务和 NTFS 兼容驱动
|
||||||
|
|
||||||
|
我们要安装下面几个软件来启用 samba 协议,使[文件浏览器][20]能够访问树莓派分享的文件夹,`ntfs-3g` 可以使我们能够访问移动硬盘中 ntfs 文件系统的文件。
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt-get install ntfs-3g
|
||||||
|
sudo apt-get install samba samba-common-bin
|
||||||
|
```
|
||||||
|
|
||||||
|
你可以参照[这些文档][17]来配置 Samba。
|
||||||
|
|
||||||
|
重要提示:参考的文档介绍的是挂载外置硬盘到树莓派上,我们不这样做,是因为在这篇文章写作的时候,树莓派在启动时的 auto-mounts 功能同时将 SD 卡和优盘挂载到`/media/pi/`上,该文章有一些多余的功能我们也不会采用。
|
||||||
|
|
||||||
|
### 2. Python 脚本
|
||||||
|
|
||||||
|
树莓派配置好后,我们需要开发脚本来实际拷贝和备份照片。注意,这个脚本只是提供了特定的自动化备份进程,如果你有基本的 Linux/树莓派命令行操作的技能,你可以 ssh 进树莓派,然后创建需要的文件夹,使用`cp`或`rsync`命令拷贝你自己的照片从一个设备到另外一个设备上。在脚本里我们用`rsync`命令,这个命令比较可靠而且支持增量备份。
|
||||||
|
|
||||||
|
这个过程依赖两个文件,脚本文件自身和`backup_photos.conf`这个配置文件,后者只有几行包含被挂载的目的驱动器(优盘)和应该挂载到哪个目录,它看起来是这样的:
|
||||||
|
|
||||||
|
```
|
||||||
|
mount folder=/media/pi/
|
||||||
|
destination folder=PDRIVE128GB
|
||||||
|
```
|
||||||
|
|
||||||
|
重要提示:在这个符号`=`前后不要添加多余的空格,否则脚本会失效。
|
||||||
|
|
||||||
|
下面是这个 Python 脚本,我把它命名为`backup_photos.py`,把它放到了`/home/pi/scripts/`目录下,我在每行都做了注释可以方便的查看各行的功能。
|
||||||
|
|
||||||
|
```
|
||||||
|
#!/usr/bin/python3
|
||||||
|
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
from sh import rsync
|
||||||
|
|
||||||
|
'''
|
||||||
|
脚本将挂载到 /media/pi 的 SD 卡上的内容复制到目的磁盘的同名目录下,目的磁盘的名字在 .conf文件里定义好了。
|
||||||
|
|
||||||
|
|
||||||
|
Argument: label/name of the mounted SD Card.
|
||||||
|
'''
|
||||||
|
|
||||||
|
CONFIG_FILE = '/home/pi/scripts/backup_photos.conf'
|
||||||
|
ORIGIN_DEV = sys.argv[1]
|
||||||
|
|
||||||
|
def create_folder(path):
|
||||||
|
|
||||||
|
print ('attempting to create destination folder: ',path)
|
||||||
|
if not os.path.exists(path):
|
||||||
|
try:
|
||||||
|
os.mkdir(path)
|
||||||
|
print ('Folder created.')
|
||||||
|
except:
|
||||||
|
print ('Folder could not be created. Stopping.')
|
||||||
|
return
|
||||||
|
else:
|
||||||
|
print ('Folder already in path. Using that instead.')
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
confFile = open(CONFIG_FILE,'rU')
|
||||||
|
#重要:: rU 选项将以统一换行模式打开文件,
|
||||||
|
#所以 \n 和/或 \r 都被识别为一个新行。
|
||||||
|
|
||||||
|
confList = confFile.readlines()
|
||||||
|
confFile.close()
|
||||||
|
|
||||||
|
|
||||||
|
for line in confList:
|
||||||
|
line = line.strip('\n')
|
||||||
|
|
||||||
|
try:
|
||||||
|
name , value = line.split('=')
|
||||||
|
|
||||||
|
if name == 'mount folder':
|
||||||
|
mountFolder = value
|
||||||
|
elif name == 'destination folder':
|
||||||
|
destDevice = value
|
||||||
|
|
||||||
|
|
||||||
|
except ValueError:
|
||||||
|
print ('Incorrect line format. Passing.')
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
destFolder = mountFolder+destDevice+'/'+ORIGIN_DEV
|
||||||
|
create_folder(destFolder)
|
||||||
|
|
||||||
|
print ('Copying files...')
|
||||||
|
|
||||||
|
# 取消这行备注将删除不在源处的文件
|
||||||
|
# rsync("-av", "--delete", mountFolder+ORIGIN_DEV, destFolder)
|
||||||
|
rsync("-av", mountFolder+ORIGIN_DEV+'/', destFolder)
|
||||||
|
|
||||||
|
print ('Done.')
|
||||||
|
```
|
||||||
|
|
||||||
|
### 3. iPad Pro 的配置
|
||||||
|
|
||||||
|
因为重活都由树莓派干了,文件不通过 iPad Pro 传输,这比我[之前尝试的一种方案][18]有巨大的优势。我们在 iPad 上只需要安装上 [Prompt2][19] 来通过 SSH 连接树莓派就行了,这样你既可以运行 Python 脚本也可以手动复制文件了。
|
||||||
|
|
||||||
|
![](http://www.movingelectrons.net/images/bkup_photos_ipad&rpi_prompt.jpg)
|
||||||
|
|
||||||
|
*iPad 用 Prompt2 通过 SSH 连接树莓派*
|
||||||
|
|
||||||
|
因为我们安装了 Samba,我们可以以更图形化的方式访问连接到树莓派的 USB 设备,你可以看视频,在不同的设备之间复制和移动文件,[文件浏览器][20]对于这种用途非常完美。
|
||||||
|
|
||||||
|
### 4. 将它们结合在一起
|
||||||
|
|
||||||
|
我们假设`SD32GB-03`是连接到树莓派 USB 端口之一的 SD 卡的卷标,`PDRIVE128GB`是那个优盘的卷标,也连接到设备上,并在上面指出的配置文件中定义好。如果我们想要备份 SD 卡上的图片,我们需要这么做:
|
||||||
|
|
||||||
|
1. 给树莓派加电打开,将驱动器自动挂载好。
|
||||||
|
2. 连接树莓派配置好的 WIFI 网络。
|
||||||
|
3. 用 [Prompt2][21] 这个 app 通过 SSH 连接到树莓派。
|
||||||
|
4. 连接好后输入下面的命令:`python3 backup_photos.py SD32GB-03`
|
||||||
|
|
||||||
|
首次备份需要一些时间,这依赖于你的 SD 卡使用了多少容量。这意味着你需要一直保持树莓派和 iPad 设备连接不断,你可以在脚本运行之前通过 `nohup` 命令解决:
|
||||||
|
|
||||||
|
```
|
||||||
|
nohup python3 backup_photos.py SD32GB-03 &
|
||||||
|
```
|
||||||
|
|
||||||
|
![](http://www.movingelectrons.net/images/bkup_photos_ipad&rpi_finished.png)
|
||||||
|
|
||||||
|
*运行完成的脚本如图所示*
|
||||||
|
|
||||||
|
### 未来的定制
|
||||||
|
|
||||||
|
我在树莓派上安装了 vnc 服务,这样我可以通过其它计算机或在 iPad 上用 [Remoter App][23]连接树莓派的图形界面,我安装了 [BitTorrent Sync][24] 用来远端备份我的图片,当然需要先设置好。当我有了可以运行的解决方案之后,我会补充我的文章。
|
||||||
|
|
||||||
|
你可以在下面发表你的评论和问题,我会在此页下面回复。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.movingelectrons.net/blog/2016/06/26/backup-photos-while-traveling-with-a-raspberry-pi.html
|
||||||
|
|
||||||
|
作者:[Lenin][a]
|
||||||
|
译者:[jiajia9linuxer](https://github.com/jiajia9linuxer)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.movingelectrons.net/blog/2016/06/26/backup-photos-while-traveling-with-a-raspberry-pi.html
|
||||||
|
[1]: http://bit.ly/1MVVtZi
|
||||||
|
[2]: http://www.amazon.com/dp/B01D3NZIMA/?tag=movinelect0e-20
|
||||||
|
[3]: http://www.amazon.com/dp/B01CD5VC92/?tag=movinelect0e-20
|
||||||
|
[4]: http://www.amazon.com/dp/B010Q57T02/?tag=movinelect0e-20
|
||||||
|
[5]: http://www.amazon.com/dp/B01F1PSFY6/?tag=movinelect0e-20
|
||||||
|
[6]: http://amzn.to/293kPqX
|
||||||
|
[7]: http://amzn.to/290syFY
|
||||||
|
[8]: http://amzn.to/290syFY
|
||||||
|
[9]: http://amzn.to/290syFY
|
||||||
|
[10]: http://amzn.to/290syFY
|
||||||
|
[11]: http://amzn.to/293kPqX
|
||||||
|
[12]: https://www.raspberrypi.org/downloads/noobs/
|
||||||
|
[13]: https://www.raspberrypi.org/documentation/remote-access/ssh/passwordless.md
|
||||||
|
[14]: https://www.iterm2.com/
|
||||||
|
[15]: https://frillip.com/using-your-raspberry-pi-3-as-a-wifi-access-point-with-hostapd/
|
||||||
|
[16]: https://www.raspberrypi.org/documentation/linux/usage/rc-local.md
|
||||||
|
[17]: http://www.howtogeek.com/139433/how-to-turn-a-raspberry-pi-into-a-low-power-network-storage-device/
|
||||||
|
[18]: http://bit.ly/1MVVtZi
|
||||||
|
[19]: https://itunes.apple.com/us/app/prompt-2/id917437289?mt=8&uo=4&at=11lqkH
|
||||||
|
[20]: https://itunes.apple.com/us/app/filebrowser-access-files-on/id364738545?mt=8&uo=4&at=11lqkH
|
||||||
|
[21]: https://itunes.apple.com/us/app/prompt-2/id917437289?mt=8&uo=4&at=11lqkH
|
||||||
|
[22]: https://en.m.wikipedia.org/wiki/Nohup
|
||||||
|
[23]: https://itunes.apple.com/us/app/remoter-pro-vnc-ssh-rdp/id519768191?mt=8&uo=4&at=11lqkH
|
||||||
|
[24]: https://getsync.com/
|
@ -0,0 +1,88 @@
|
|||||||
|
适用于所有发行版的 Linux 应用程序——是否真的有好处呢?
|
||||||
|
============================================================================
|
||||||
|
|
||||||
|
![](http://www.iwillfolo.com/wordpress/wp-content/uploads/2016/06/Bundled-applications.jpg)
|
||||||
|
|
||||||
|
让我们回顾一下 Linux 社区最新的愿景——推动去中心化的应用来解决发行版的碎片化。
|
||||||
|
|
||||||
|
继上周的文章:“[Snap、Flatpak 这种通吃所有发行版的打包方式真的有用吗?][1]” 之后,一系列新观点浮出水面,其中可能包含关于这样应用是否有用的重要信息。
|
||||||
|
|
||||||
|
### 缺点
|
||||||
|
|
||||||
|
就这个话题在[这里][2]的评论,一个叫 Till 的 [Gentoo][3] 使用者,对于上一次我们未能完全解释的问题给出了一些新的观点。
|
||||||
|
|
||||||
|
对于上一次我们选择仅仅称之为膨胀的的东西,Till 从另一方面做了剖析膨胀将来的发展,这可以帮助我们更好的理解它的组成和其影响。
|
||||||
|
|
||||||
|
这些被称之为“捆绑应用”的应用程序能够工作在所有发行版上的机制是——将它依赖的库都包含在它们的应用软件之中,Till 说:
|
||||||
|
|
||||||
|
> “捆绑应用装载了大量的并不被应用开发者所维护的软件。如果其中的某个函数库被发现了一个安全问题而需要更新的话,你得为每一个独立的应用程序安装更新来确保你的系统安全。”
|
||||||
|
|
||||||
|
本质上,Till 提出了一个**重要的安全问题**。但是它并不仅仅与安全有关系,它还关系到许多方面,比如说系统维护、原子更新等等。
|
||||||
|
|
||||||
|
此外,如果我们进一步假设:依赖的开发者们也许会合作,将他们的软件与使用它的应用程序一起发布(一种理想状况),但这将导致整个平台的开发整体放缓。
|
||||||
|
|
||||||
|
另一个将会导致的问题是**透明的依赖关系变得模糊**,就是说,如果你想知道一个应用程序捆绑了哪些依赖关系,你必须依靠开发者发布这些数据。
|
||||||
|
|
||||||
|
或者就像 Till 说的:“比如说像某某包是否已经包含了更新的某函数库这样的问题将会是你每天需要面对的。”
|
||||||
|
|
||||||
|
与之相反,对于 Linux 现行的标准的包管理方法(包括二进制包和源码包),你能够很容易的注意到哪些函数库已经在系统中更新了。
|
||||||
|
|
||||||
|
并且,你也可以很轻松的知道其它哪些应用使用了这个函数库,这就将你从繁琐的单独检查每一个应用程序的工作中解救了出来。
|
||||||
|
|
||||||
|
其他可能由膨胀导致的缺点包括:**更大的包体积**(每一个应用程序捆绑了依赖),**更高的内存占用**(没有共享函数库),并且,**少了一个包过滤机制**来防止恶意软件:发行版的包维护者也充当了一个在开发者和用户之间的过滤者,他保障了用户获得高质量的软件。
|
||||||
|
|
||||||
|
而在捆绑应用中就不再是这种情况了。
|
||||||
|
|
||||||
|
最后一点,Till 声称,尽管在某些情况下很有用,但是在大多数情况下,**捆绑应用程序将弱化自由软件在发行版中的地位**(专有软件供应商将被能够发布他们的软件而不用把它放到公共软件仓库中)。
|
||||||
|
|
||||||
|
除此之外,它引出了许多其他问题。很多问题都可以简单归结到开发人员身上。
|
||||||
|
|
||||||
|
### 优点
|
||||||
|
|
||||||
|
相比之下,另一个名叫 Sven 的人的评论试图反驳目前普遍反对使用捆绑应用程序的观点,从而证明和支持使用它。
|
||||||
|
|
||||||
|
“浪费空间?”——Sven 声称在当今世界我们有**很多其他事情在浪费磁盘空间**,比如电影存储在硬盘上、本地安装等等……
|
||||||
|
|
||||||
|
最终,这些事情浪费的空间要远远多于仅仅“ 100 MB 而你每天都要使用的程序。……因此浪费空间的说法实在很荒谬。”
|
||||||
|
|
||||||
|
“浪费运行内存?”——主要的观点有:
|
||||||
|
|
||||||
|
- **共享库浪费的内存要远远少于程序的运行时数据所占用的**。
|
||||||
|
|
||||||
|
- 而今运行**内存已经很便宜**了。
|
||||||
|
|
||||||
|
“安全梦魇”——不是每个应用程序的运行**真正的要注重安全**。
|
||||||
|
|
||||||
|
而且,**许多应用程序甚至从来没有过任何安全更新**,除非在“滚动更新的发行版”。
|
||||||
|
|
||||||
|
除了 Sven 这种从实用出发的观点以外,Till 其实也指出了捆绑应用在一些情况下也有着其优点:
|
||||||
|
|
||||||
|
- 专有软件的供应商想要保持他们的代码游离于公共仓库之外将更加容易。
|
||||||
|
- 没有被你的发行版打包进去的小众应用程序将变得更加可行。
|
||||||
|
- 在没有 Beta 包的二进制发行版中测试应用将变得简单。
|
||||||
|
- 将用户从复杂的依赖关系中解放出来。
|
||||||
|
|
||||||
|
### 最后的思考
|
||||||
|
|
||||||
|
虽然关于此问题有着不同的想法,但是有一个被大家共同接受的观点是:**捆绑应用对于填补 Linux 生态系统有着其独到的作用。**
|
||||||
|
|
||||||
|
虽然如此,它的定位,不论是主流的还是边缘的,都变得愈发清晰,至少理论上是这样。
|
||||||
|
|
||||||
|
想要尽可能优化其系统的用户,在大多数情况下应该要避免使用捆绑应用。
|
||||||
|
|
||||||
|
而讲究易用性、尽可能在维护系统上少费劲的用户,可能应该会感觉这种新应用十分舒爽。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.iwillfolo.com/linux-applications-that-works-on-all-distributions-are-they-any-good/
|
||||||
|
|
||||||
|
作者:[Liron][a]
|
||||||
|
译者:[Chao-zhi](https://github.com/Chao-zhi)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创翻译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.iwillfolo.com/category/editorials/
|
||||||
|
[1]: https://linux.cn/article-7783-1.html
|
||||||
|
[2]: http://www.proli.net/2016/06/25/gnulinux-bundled-application-ramblings/
|
||||||
|
[3]: http://www.iwillfolo.com/5-reasons-use-gentoo-linux/
|
36
published/201609/20160803 The revenge of Linux.md
Normal file
36
published/201609/20160803 The revenge of Linux.md
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
Linux 的逆袭
|
||||||
|
========================
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/images/life/penguin%20swimming.jpg?itok=mfhEdRdM)
|
||||||
|
|
||||||
|
Linux 系统在早期的时候被人们嘲笑,它什么也干不了。而现在,Linux 无处不在!
|
||||||
|
|
||||||
|
我当时还是个在巴西学习计算机工程的大三学生,并同时在一个全球审计和顾问公司兼职系统管理员。公司决定用 Oracle 数据库开发一些企业资源计划(ERP)软件。因此,我得以在 Digital UNIX OS (DEC Alpha) 进行训练,这个训练颠覆了我的三观。
|
||||||
|
|
||||||
|
UNIX 系统非常的强大,而且给予了我们对机器上包括存储系统、网络、应用和其他一切的绝对控制权。
|
||||||
|
|
||||||
|
我开始在 ksh 和 Bash 里编写大量的脚本让系统进行自动备份、文件传输、提取转换加载(ETL)操作、自动化 DBA 日常工作,还为各种不同的项目创建了许多服务。此外,调整数据库和操作系统的工作让我更好的理解了如何让服务器以最佳方式运行。在那时,我在自己的个人电脑上使用的是 Windows 95 系统,而我非常想要在我的个人电脑里放进一个 Digital UNIX,或者哪怕是 Solaris 或 HP-UX 也行,但是那些 UNIX 系统都得在特定的硬件才能上运行。我阅读了所有的系统文档,还找过其它的书籍以求获得更多的信息,也在我们的开发环境里对这些疯狂的想法进行了实验。
|
||||||
|
|
||||||
|
后来在大学里,我从我的同事那听说了 Linux。我那时非常激动的从还在用拨号方式连接的因特网上下载了它。在我的正宗的个人电脑里装上 UNIX 这类系统的这个想法真是太酷了!
|
||||||
|
|
||||||
|
Linux 不同于 UNIX 系统,它设计用来在各种常见个人电脑硬件上运行,在起初,让它开始工作确实有点困难,Linux 针对的用户群只有系统管理员和极客们。我为了让它能运行,甚至用 C 语言修改了驱动软件。我之前使用 UNIX 的经历让我在编译 Linux 内核,排错这些过程中非常的顺手。由于它不同于那些只适合特定硬件配置的封闭系统,所以让 Linux 跟各种意料之外的硬件配置一起工作真的是件非常具有挑战性的事。
|
||||||
|
|
||||||
|
我曾见过 Linux 在数据中心获得一席之地。一些具有冒险精神的系统管理员使用它来帮他们完成每天监视和管理基础设施的工作,随后,Linux 作为 DNS 和 DHCP 服务器、打印管理和文件服务器等赢得了更多的使用。企业曾对 Linux 有着很多顾虑(恐惧,不确定性,怀疑(FUD:fear, uncertainty and doubt))和诟病:谁是它的拥有者?由谁来支持它?有适用于它的应用吗?
|
||||||
|
|
||||||
|
但现在看来,Linux 在各个地方进行着逆袭!从开发者的个人电脑到大企业的服务器;我们能在智能手机、智能手表以及像树莓派这样的物联网(IoT)设备里找到它。甚至 Mac OS X 有些命令跟我们所熟悉的命令一样。微软在制造它自己的发行版,在 Azure 上运行,然后…… Windows 10 要装备 Bash。
|
||||||
|
|
||||||
|
有趣的是 IT 市场会不断地创造并迅速的用新技术替代,但是我们所掌握的 Digital UNIX、HP-UX 和 Solaris 这些旧系统的知识还依然有效并跟 Linux 息息相关,不论是为了工作还是玩。现在我们能完全的掌控我们的系统,并使它发挥最大的效用。此外,Linux 有个充满热情的社区。
|
||||||
|
|
||||||
|
我真的建议想在计算机方面发展的年轻人学习 Linux,不论你处于 IT 界里的哪个分支。如果你深入了解了一个普通的家用个人电脑是如何工作的,你就可以以基本相同的方式来面对任何机器。你可以通过 Linux 学习最基本的计算机知识,并通过它建立能在 IT 界任何地方都有用的能力!
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://opensource.com/life/16/8/revenge-linux
|
||||||
|
|
||||||
|
作者:[Daniel Carvalho][a]
|
||||||
|
译者:[H-mudcup](https://github.com/H-mudcup)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创翻译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://opensource.com/users/danielscarvalho
|
@ -0,0 +1,36 @@
|
|||||||
|
为什么计量 IT 的生产力如此具有挑战性?
|
||||||
|
===========================
|
||||||
|
|
||||||
|
![](https://enterprisersproject.com/sites/default/files/styles/620x350/public/images/cio_talent_6.png?itok=JV-zSor3)
|
||||||
|
|
||||||
|
在某些行业里,人们可以根据一些测量标准判定一个人的生产力。比如,如果你是一个零件制造商,可以通过一个月你能够制造的零件数量来确定你的生产效率。如果你在客户服务中心工作,你解答了多少个客户来电,你的平均解答时间都会成为评判你的生产效率的依据。这些都是相当简单的案例,但即便你是一位医生,也可以通过你主刀的临床手术次数或者一个月你确诊的患者数量来确定你的生产效率。无论这些评判标准正确与否,但它们提供了一个通用的方法来评断一个人在给定时间内的执行能力。
|
||||||
|
|
||||||
|
然而在 IT 这方面,通过上述方法来衡量一个人的生产力是不可能的,因为 IT 有太多的变化性。比如,通过一个开发者编写的代码行数来衡量开发者所用的时间看起来很诱人。但是,编程的语言很大程度上能影响到根据这种方法得到的结论。因为一种编程语言的一行代码比用其他编程语言编写所花费的时间和难度可能会明显的多或少。
|
||||||
|
|
||||||
|
它总是这样不可捉摸吗?多年以前,你可能听说过或者经历过根据功能点来衡量 IT 工作人员的生产效率。这些措施是针对开发者们能够创建的关键特征来衡量开发者的生产效率的。但这种方法在今天也变得逐渐难以实施,开发者经常将可能已有的逻辑封装进内部,比如,按供应商来整合功能点。这使得仅仅是基于功能点的数目来估量生产效率难度加大。
|
||||||
|
|
||||||
|
这两个例子能够阐述为什么当我们 CIO 之间谈论 IT 生产效率的时候有时会引起争论。考虑以下这个假想中的谈话:
|
||||||
|
|
||||||
|
> IT leader:“天啊,我想这些开发者一定很厉害。”
|
||||||
|
> HR:“真的假的,他们做了什么?”
|
||||||
|
> IT leader:“他们做了个相当好的应用。”
|
||||||
|
> HR:“好吧,那他们比那些做了 10 个应用的开发者更好吗”
|
||||||
|
> IT leader:“这要看你怎么理解 ‘更好’。”
|
||||||
|
|
||||||
|
这个对话比较有代表性。当我们处于上述的这种交谈时,这里面有太多的主观因素导致我们很难回答这个问题。当我们用一种有意义的方法来测试 IT 的效率时,类似上述谈话的这种问题仅仅是冰山一角。这不仅仅使谈话更加困难-它还会使 CIO 们很难展示他们的团队在商业上的价值。
|
||||||
|
|
||||||
|
确实这不是一个新出现的问题。我已经花费差不多 30 年的时间来思考这个问题。我得出的结论是我们真的不应该在谈论 IT 的生产效率这件事上面相互烦扰-因为我们永远不可能有结论。
|
||||||
|
|
||||||
|
我认为我们需要在改变这种对话同时停止根据生产能力和成本来谈论 IT 的生产效率,将目光集中于衡量 IT 的整体商业价值上。重申一下,这个过程不会很容易。商业价值的实现是一件困难的事情。但如果 CIO 们能够和商业人员合作来解决这个问题,就可以将实际价值变的更加科学而非一种艺术形式。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://enterprisersproject.com/article/2016/8/why-measuring-it-productivity-so-challenging
|
||||||
|
|
||||||
|
作者:[Anil Cheriyan][a]
|
||||||
|
译者:[LemonDemo](https://github.com/LemonDemo) [WangYueScream](https://github.com/WangYueScream)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://enterprisersproject.com/user/anil-cheriyan
|
219
published/201609/20160809 How to build your own Git server.md
Normal file
219
published/201609/20160809 How to build your own Git server.md
Normal file
@ -0,0 +1,219 @@
|
|||||||
|
Git 系列(六):如何搭建你自己的 Git 服务器
|
||||||
|
====================
|
||||||
|
|
||||||
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/images/business/bus-big-data.png?itok=sOQHDuID)
|
||||||
|
|
||||||
|
现在我们将要学习如何搭建 git 服务器,如何编写自定义的 Git 钩子来在特定的事件触发相应的动作(例如通知),或者是发布你的代码到一个站点。
|
||||||
|
|
||||||
|
直到现在,我们主要讨论的还是以一个使用者的身份与 Git 进行交互。这篇文章中我将讨论 Git 的管理,并且设计一个灵活的 Git 框架。你可能会觉得这听起来是 “高阶 Git 技术” 或者 “只有狂热粉才能阅读”的一句委婉的说法,但是事实是这里面的每个任务都不需要很深的知识或者其他特殊的训练,就能基本理解 Git 的工作原理,有可能需要一丁点关于 Linux 的知识。
|
||||||
|
|
||||||
|
### 共享 Git 服务器
|
||||||
|
|
||||||
|
创建你自己的共享 Git 服务器意外地简单,而且在很多情况下,遇到的这点麻烦是完全值得的。不仅仅是因为它保证你有权限查看自己的代码,它还可以通过扩展为 Git 的使用敞开了一扇大门,例如个人 Git 钩子、无限制的数据存储、和持续集成与分发(CI & CD)。
|
||||||
|
|
||||||
|
如果你知道如何使用 Git 和 SSH,那么你已经知道怎么创建一个 Git 服务器了。Git 的设计方式,就是让你在创建或者 clone 一个仓库的时候,就完成了一半服务器的搭建。然后允许用 SSH 访问仓库,而且任何有权限访问的人都可以使用你的仓库作为 clone 的新仓库的基础。
|
||||||
|
|
||||||
|
但是,这是一个小的点对点环境(ad-hoc)。按照一些方案你可以创建一些带有同样的功能的设计优良的 Git 服务器,同时有更好的拓展性。
|
||||||
|
|
||||||
|
首要之事:确认你的用户们,现在的用户以及之后的用户都要考虑。如果你是唯一的用户那么没有任何改动的必要。但是如果你试图邀请其他的代码贡献者使用,那么你应该允许一个专门的分享系统用户给你的开发者们。
|
||||||
|
|
||||||
|
假定你有一个可用的服务器(如果没有,这不成问题,Git 会帮忙解决,CentOS 的 [树莓派 3][3] 是个不错的开始),然后第一步就是只允许使用 SSH 密钥认证的 SSH 登录。这比使用密码登录安全得多,因为这可以免于暴力破解,也可以通过直接删除用户密钥而禁用用户。
|
||||||
|
|
||||||
|
一旦你启用了 SSH 密钥认证,创建 `gituser` 用户。这是给你的所有授权的用户们的公共用户:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ su -c 'adduser gituser'
|
||||||
|
```
|
||||||
|
|
||||||
|
然后切换到刚创建的 `gituser` 用户,创建一个 `~/.ssh` 的框架,并设置好合适的权限。这很重要,如果权限设置得太开放会使自己所保护的 SSH 没有意义。
|
||||||
|
|
||||||
|
```
|
||||||
|
$ su - gituser
|
||||||
|
$ mkdir .ssh && chmod 700 .ssh
|
||||||
|
$ touch .ssh/authorized_keys
|
||||||
|
$ chmod 600 .ssh/authorized_keys
|
||||||
|
```
|
||||||
|
|
||||||
|
`authorized_keys` 文件里包含所有你的开发者们的 SSH 公钥,你开放权限允许他们可以在你的 Git 项目上工作。他们必须创建他们自己的 SSH 密钥对然后把他们的公钥给你。复制公钥到 gituser 用户下的 `authorized_keys` 文件中。例如,为一个叫 Bob 的开发者,执行以下命令:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ cat ~/path/to/id_rsa.bob.pub >> /home/gituser/.ssh/authorized_keys
|
||||||
|
```
|
||||||
|
|
||||||
|
只要开发者 Bob 有私钥并且把相对应的公钥给你,Bob 就可以用 `gituser` 用户访问服务器。
|
||||||
|
|
||||||
|
但是,你并不是想让你的开发者们能使用服务器,即使只是以 `gituser` 的身份访问。你只是想给他们访问 Git 仓库的权限。因为这个特殊的原因,Git 提供了一个限制的 shell,准确的说是 `git-shell`。以 root 身份执行以下命令,把 `git-shell` 添加到你的系统中,然后设置成 `gituser` 用户的默认 shell。
|
||||||
|
|
||||||
|
```
|
||||||
|
# grep git-shell /etc/shells || su -c "echo `which git-shell` >> /etc/shells"
|
||||||
|
# su -c 'usermod -s git-shell gituser'
|
||||||
|
```
|
||||||
|
|
||||||
|
现在 `gituser` 用户只能使用 SSH 来 push 或者 pull Git 仓库,并且无法使用任何一个可以登录的 shell。你应该把你自己添加到和 `gituser` 一样的组中,在我们的样例服务器中这个组的名字也是 `gituser`。
|
||||||
|
|
||||||
|
举个例子:
|
||||||
|
|
||||||
|
```
|
||||||
|
# usermod -a -G gituser seth
|
||||||
|
```
|
||||||
|
|
||||||
|
仅剩下的一步就是创建一个 Git 仓库。因为没有人能在服务器上直接与 Git 交互(也就是说,你之后不能 SSH 到服务器然后直接操作这个仓库),所以创建一个空的仓库 。如果你想使用这个放在服务器上的仓库来完成工作,你可以从它的所在处 `clone` 下来,然后在你的 home 目录下进行工作。
|
||||||
|
|
||||||
|
严格地讲,你不是必须创建这个空的仓库;它和一个正常的仓库一样工作。但是,一个空的仓库没有工作分支(working tree) (也就是说,使用 `checkout` 并没有任何分支显示)。这很重要,因为不允许远程使用者们 `push` 到一个有效的分支上(如果你正在 `dev` 分支工作然后突然有人把一些变更 `push` 到你的工作分支,你会有怎么样的感受?)。因为一个空的仓库可以没有有效的分支,所以这不会成为一个问题。
|
||||||
|
|
||||||
|
你可以把这个仓库放到任何你想放的地方,只要你想要放开权限给用户和用户组,让他们可以在仓库下工作。千万不要保存目录到比如说一个用户的 home 目录下,因为那里有严格的权限限制。保存到一个常规的共享地址,例如 `/opt` 或者 `/usr/local/share`。
|
||||||
|
|
||||||
|
以 root 身份创建一个空的仓库:
|
||||||
|
|
||||||
|
```
|
||||||
|
# git init --bare /opt/jupiter.git
|
||||||
|
# chown -R gituser:gituser /opt/jupiter.git
|
||||||
|
# chmod -R 770 /opt/jupiter.git
|
||||||
|
```
|
||||||
|
|
||||||
|
现在任何一个用户,只要他被认证为 `gituser` 或者在 `gituser` 组中,就可以从 jupiter.git 库中读取或者写入。在本地机器尝试以下操作:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ git clone gituser@example.com:/opt/jupiter.git jupiter.clone
|
||||||
|
Cloning into 'jupiter.clone'...
|
||||||
|
Warning: you appear to have cloned an empty repository.
|
||||||
|
```
|
||||||
|
|
||||||
|
谨记:开发者们**一定**要把他们的 SSH 公钥加入到 `gituser` 用户下的 `authorized_keys` 文件里,或者说,如果他们有服务器上的用户(如果你给了他们用户),那么他们的用户必须属于 `gituser` 用户组。
|
||||||
|
|
||||||
|
### Git 钩子
|
||||||
|
|
||||||
|
运行你自己的 Git 服务器最赞的一件事之一就是可以使用 Git 钩子。Git 托管服务有时提供一个钩子类的接口,但是他们并不会给你真正的 Git 钩子来让你访问文件系统。Git 钩子是一个脚本,它将在一个 Git 过程的某些点运行;钩子可以运行在当一个仓库即将接收一个 commit 时、或者接受一个 commit 之后,或者即将接收一次 push 时,或者一次 push 之后等等。
|
||||||
|
|
||||||
|
这是一个简单的系统:任何放在 `.git/hooks` 目录下的脚本、使用标准的命名体系,就可按设计好的时间运行。一个脚本是否应该被运行取决于它的名字; `pre-push` 脚本在 `push` 之前运行,`post-receive` 脚本在接受 `commit` 之后运行等等。这或多或少的可以从名字上看出来。
|
||||||
|
|
||||||
|
脚本可以用任何语言写;如果在你的系统上有可以执行的脚本语言,例如输出 ‘hello world’ ,那么你就可以这个语言来写 Git 钩子脚本。Git 默认带了一些例子,但是并不有启用。
|
||||||
|
|
||||||
|
想要动手试一个?这很简单。如果你没有现成的 Git 仓库,首先创建一个 Git 仓库:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ mkdir jupiter
|
||||||
|
$ cd jupiter
|
||||||
|
$ git init .
|
||||||
|
```
|
||||||
|
|
||||||
|
然后写一个 “hello world” 的 Git 钩子。因为我为了支持老旧系统而使用 tsch,所以我仍然用它作为我的脚本语言,你可以自由的使用自己喜欢的语言(Bash,Python,Ruby,Perl,Rust,Swift,Go):
|
||||||
|
|
||||||
|
```
|
||||||
|
$ echo "#\!/bin/tcsh" > .git/hooks/post-commit
|
||||||
|
$ echo "echo 'POST-COMMIT SCRIPT TRIGGERED'" >> ~/jupiter/.git/hooks/post-commit
|
||||||
|
$ chmod +x ~/jupiter/.git/hooks/post-commit
|
||||||
|
```
|
||||||
|
|
||||||
|
现在测试它的输出:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ echo "hello world" > foo.txt
|
||||||
|
$ git add foo.txt
|
||||||
|
$ git commit -m 'first commit'
|
||||||
|
! POST-COMMIT SCRIPT TRIGGERED
|
||||||
|
[master (root-commit) c8678e0] first commit
|
||||||
|
1 file changed, 1 insertion(+)
|
||||||
|
create mode 100644 foo.txt
|
||||||
|
```
|
||||||
|
|
||||||
|
现在你已经实现了:你的第一个有功能的 Git 钩子。
|
||||||
|
|
||||||
|
### 有名的 push-to-web 钩子
|
||||||
|
|
||||||
|
Git 钩子最流行的用法就是自动 `push` 更改的代码到一个正在使用中的产品级 Web 服务器目录下。这是摆脱 FTP 的很好的方式,对于正在使用的产品保留完整的版本控制,整合并自动化内容的发布。
|
||||||
|
|
||||||
|
如果操作正确,网站发布工作会像以前一样很好的完成,而且在某种程度上,很精准。Git 真的好棒。我不知道谁最初想到这个主意,但是我是从 Emacs 和 Git 方面的专家,IBM 的 Bill von Hagen 那里第一次听到它的。他的文章包含关于这个过程的权威介绍:[Git 改变了分布式网页开发的游戏规则][1]。
|
||||||
|
|
||||||
|
### Git 变量
|
||||||
|
|
||||||
|
每一个 Git 钩子都有一系列不同的变量对应触发钩子的不同 Git 行为。你需不需要这些变量,主要取决于你写的程序。如果你只是需要一个当某人 push 代码时候的通用邮件通知,那么你就不需要什么特殊的东西,甚至也不需要编写额外的脚本,因为已经有现成的适合你的样例脚本。如果你想在邮件里查看 commit 信息和 commit 的作者,那么你的脚本就会变得相对麻烦些。
|
||||||
|
|
||||||
|
Git 钩子并不是被用户直接执行,所以要弄清楚如何收集可能会混淆的重要信息。事实上,Git 钩子脚本类似于其他的脚本,像 BASH、Python、C++ 等等一样从标准输入读取参数。不同的是,我们不会给它提供这个输入,所以,你在使用的时候,需要知道可能的输入参数。
|
||||||
|
|
||||||
|
在写 Git 钩子之前,看一下 Git 在你的项目目录下 `.git/hooks` 目录中提供的一些例子。举个例子,在这个 `pre-push.sample` 文件里,注释部分说明了如下内容:
|
||||||
|
|
||||||
|
```
|
||||||
|
# $1 -- 即将 push 的远程仓库的名字
|
||||||
|
# $2 -- 即将 push 的远程仓库的 URL
|
||||||
|
# 如果 push 的时候,并没有一个命名的远程仓库,那么这两个参数将会一样。
|
||||||
|
#
|
||||||
|
# 提交的信息将以下列形式按行发送给标准输入
|
||||||
|
# <local ref> <local sha1> <remote ref> <remote sha1>
|
||||||
|
```
|
||||||
|
|
||||||
|
并不是所有的例子都是这么清晰,而且关于钩子获取变量的文档依旧缺乏(除非你去读 Git 的源码)。但是,如果你有疑问,你可以从线上[其他用户的尝试中][2]学习,或者你只是写一些基本的脚本,比如 `echo $1, $2, $3` 等等。
|
||||||
|
|
||||||
|
### 分支检测示例
|
||||||
|
|
||||||
|
我发现,对于生产环境来说有一个共同的需求,就是需要一个只有在特定分支被修改之后,才会触发事件的钩子。以下就是如何跟踪分支的示例。
|
||||||
|
|
||||||
|
首先,Git 钩子本身是不受版本控制的。 Git 并不会跟踪它自己的钩子,因为对于钩子来说,它是 Git 的一部分,而不是你仓库的一部分。所以,Git 钩子可以监控你的 Git 服务器上的一个空仓库的 commit 记录和 push 记录,而不是你本地仓库的一部分。
|
||||||
|
|
||||||
|
我们来写一个 `post-receive`(也就是说,在 `commit` 被接受之后触发)钩子。第一步就是需要确定分支名:
|
||||||
|
|
||||||
|
```
|
||||||
|
#!/bin/tcsh
|
||||||
|
|
||||||
|
foreach arg ( $< )
|
||||||
|
set argv = ( $arg )
|
||||||
|
set refname = $1
|
||||||
|
end
|
||||||
|
```
|
||||||
|
|
||||||
|
这个 for 循环用来读入第一个参数 `$1` ,然后循环用第二个参数 `$2` 去覆盖它,然后用第三个参数 `$3` 再这样。在 Bash 中有一个更好的方法,使用 `read` 命令,并且把值放入数组里。但是,这里是 tcsh,并且变量的顺序可以预测的,所以,这个方法也是可行的。
|
||||||
|
|
||||||
|
当我们有了 commit 记录的 `refname`,我们就能使用 Git 去找到这个分支的供人看的名字:
|
||||||
|
|
||||||
|
```
|
||||||
|
set branch = `git rev-parse --symbolic --abbrev-ref $refname`
|
||||||
|
echo $branch #DEBUG
|
||||||
|
```
|
||||||
|
|
||||||
|
然后把这个分支名和我们想要触发的事件的分支名关键字进行比较:
|
||||||
|
|
||||||
|
```
|
||||||
|
if ( "$branch" == "master" ) then
|
||||||
|
echo "Branch detected: master"
|
||||||
|
git \
|
||||||
|
--work-tree=/path/to/where/you/want/to/copy/stuff/to \
|
||||||
|
checkout -f $branch || echo "master fail"
|
||||||
|
else if ( "$branch" == "dev" ) then
|
||||||
|
echo "Branch detected: dev"
|
||||||
|
Git \
|
||||||
|
--work-tree=/path/to/where/you/want/to/copy/stuff/to \
|
||||||
|
checkout -f $branch || echo "dev fail"
|
||||||
|
else
|
||||||
|
echo "Your push was successful."
|
||||||
|
echo "Private branch detected. No action triggered."
|
||||||
|
endif
|
||||||
|
```
|
||||||
|
|
||||||
|
给这个脚本分配可执行权限:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ chmod +x ~/jupiter/.git/hooks/post-receive
|
||||||
|
```
|
||||||
|
|
||||||
|
现在,当一个用户提交到服务器的 master 分支,那些代码就会被复制到一个生产环境的目录,提交到 dev 分支则会被复制到另外的地方,其他分支将不会触发这些操作。
|
||||||
|
|
||||||
|
同时,创造一个 `pre-commit` 脚本也很简单。比如,判断一个用户是否在他们不该 `push` 的分支上 `push` 代码,或者对 commit 信息进行解析等等。
|
||||||
|
|
||||||
|
Git 钩子也可以变得复杂,而且它们因为 Git 的工作流的抽象层次不同而变得难以理解,但是它们确实是一个强大的系统,让你能够在你的 Git 基础设施上针对所有的行为进行对应的操作。如果你是一个 Git 重度用户,或者一个全职 Git 管理员,那么 Git 钩子是值得学习的,只有当你熟悉这个过程,你才能真正掌握它。
|
||||||
|
|
||||||
|
在我们这个系列下一篇也是最后一篇文章中,我们将会学习如何使用 Git 来管理非文本的二进制数据,比如音频和图片。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://opensource.com/life/16/8/how-construct-your-own-git-server-part-6
|
||||||
|
|
||||||
|
作者:[Seth Kenlon][a]
|
||||||
|
译者:[maywanting](https://github.com/maywanting)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://opensource.com/users/seth
|
||||||
|
[1]: http://www.ibm.com/developerworks/library/wa-git/
|
||||||
|
[2]: https://www.analysisandsolutions.com/code/git-hooks-summary-cheat-sheet.htm
|
||||||
|
[3]: https://wiki.centos.org/SpecialInterestGroup/AltArch/Arm32/RaspberryPi3
|
212
published/201609/20160813 Journey-to-HTTP2.md
Normal file
212
published/201609/20160813 Journey-to-HTTP2.md
Normal file
@ -0,0 +1,212 @@
|
|||||||
|
漫游 HTTP/2
|
||||||
|
===================
|
||||||
|
|
||||||
|
自从我写了上一篇博文之后,就再也找不到空闲时间写文章了。今天我终于可以抽出时间写一些关于 HTTP 的东西。
|
||||||
|
|
||||||
|
我认为每一个 web 开发者都应该对这个支撑了整个 Web 世界的 HTTP 协议有所了解,这样才能帮助你更好的完成开发任务。
|
||||||
|
|
||||||
|
在这篇文章中,我将讨论什么是 HTTP,它是怎么产生的,它的地位,以及我们应该怎么使用它。
|
||||||
|
|
||||||
|
### HTTP 是什么
|
||||||
|
|
||||||
|
首先我们要明白 HTTP 是什么。HTTP 是一个基于 `TCP/IP` 的应用层通信协议,它是客户端和服务端在互联网互相通讯的标准。它定义了内容是如何通过互联网进行请求和传输的。HTTP 是在应用层中抽象出的一个标准,使得主机(客户端和服务端)之间的通信得以通过 `TCP/IP` 来进行请求和响应。TCP 默认使用的端口是 `80`,当然也可以使用其它端口,比如 HTTPS 使用的就是 `443` 端口。
|
||||||
|
|
||||||
|
### `HTTP/0.9` - 单行协议 (1991)
|
||||||
|
|
||||||
|
HTTP 最早的规范可以追溯到 1991 年,那时候的版本是 `HTTP/0.9`,该版本极其简单,只有一个叫做 `GET` 的 请求方式。如果客户端要访问服务端上的一个页面,只需要如下非常简单的请求:
|
||||||
|
|
||||||
|
```
|
||||||
|
GET /index.html
|
||||||
|
```
|
||||||
|
|
||||||
|
服务端的响应类似如下:
|
||||||
|
|
||||||
|
```
|
||||||
|
(response body)
|
||||||
|
(connection closed)
|
||||||
|
```
|
||||||
|
|
||||||
|
就这么简单,服务端捕获到请求后立马返回 HTML 并且关闭连接,在这之中:
|
||||||
|
|
||||||
|
- 没有头信息(headers)
|
||||||
|
- 仅支持 `GET` 这一种请求方法
|
||||||
|
- 必须返回 HTML
|
||||||
|
|
||||||
|
如同你所看到的,当时的 HTTP 协议只是一块基础的垫脚石。
|
||||||
|
|
||||||
|
### HTTP/1.0 - 1996
|
||||||
|
|
||||||
|
在 1996 年,新版本的 HTTP 对比之前的版本有了极大的改进,同时也被命名为 `HTTP/1.0`。
|
||||||
|
|
||||||
|
与 `HTTP/0.9` 只能返回 HTML 不同的是,`HTTP/1.0` 支持处理多种返回的格式,比如图片、视频、文本或者其他格式的文件。它还增加了更多的请求方法(如 `POST` 和 `HEAD`),请求和响应的格式也相应做了改变,两者都增加了头信息;引入了状态码来定义返回的特征;引入了字符集支持;支持多段类型(multi-part)、用户验证信息、缓存、内容编码格式等等。
|
||||||
|
|
||||||
|
一个简单的 HTTP/1.0 请求大概是这样的:
|
||||||
|
|
||||||
|
```
|
||||||
|
GET / HTTP/1.0
|
||||||
|
Host: kamranahmed.info
|
||||||
|
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_10_5)
|
||||||
|
Accept: */*
|
||||||
|
```
|
||||||
|
|
||||||
|
正如你所看到的,在请求中附带了客户端中的一些个人信息、响应类型要求等内容。这些是在 `HTTP/0.9` 无法实现的,因为那时候没有头信息。
|
||||||
|
|
||||||
|
一个对上述请求的响应例子如下所示:
|
||||||
|
|
||||||
|
```
|
||||||
|
HTTP/1.0 200 OK
|
||||||
|
Content-Type: text/plain
|
||||||
|
Content-Length: 137582
|
||||||
|
Expires: Thu, 05 Dec 1997 16:00:00 GMT
|
||||||
|
Last-Modified: Wed, 5 August 1996 15:55:28 GMT
|
||||||
|
Server: Apache 0.84
|
||||||
|
|
||||||
|
(response body)
|
||||||
|
(connection closed)
|
||||||
|
```
|
||||||
|
|
||||||
|
从 `HTTP/1.0` (HTTP 后面跟的是版本号)早期开始,在状态码 `200` 之后就附带一个原因短语(你可以用来描述状态码)。
|
||||||
|
|
||||||
|
在这个较新一点的版本中,请求和响应的头信息仍然必须是 `ASCII` 编码,但是响应的内容可以是任意类型,如图片、视频、HTML、文本或其他类型,服务器可以返回任意内容给客户端。所以这之后,`HTTP` 中的“超文本(Hyper Text)”成了名不副实。 `HMTP` (超媒体传输协议 - Hypermedia transfer protocol)可能会更有意义,但是我猜我们还是会一直沿用这个名字。
|
||||||
|
|
||||||
|
`HTTP/1.0` 的一个主要缺点就是它不能在一个连接内拥有多个请求。这意味着,当客户端需要从服务器获取东西时,必须建立一个新的 TCP 连接,并且处理完单个请求后连接即被关闭。需要下一个东西时,你必须重新建立一个新的连接。这样的坏处在哪呢?假设你要访问一个有 `10` 张图片,`5` 个样式表(stylesheet)和 `5` 个 JavaScript 的总计 `20` 个文件才能完整展示的一个页面。由于一个连接在处理完成一次请求后即被关闭,所以将有 `20` 个单独的连接,每一个文件都将通过各自对应的连接单独处理。当连接数量变得庞大的时候就会面临严重的性能问题,因为 `TCP` 启动需要经过三次握手,才能缓慢开始。
|
||||||
|
|
||||||
|
#### 三次握手
|
||||||
|
|
||||||
|
三次握手是一个简单的模型,所有的 `TCP` 连接在传输应用数据之前都需要在三次握手中传输一系列数据包。
|
||||||
|
|
||||||
|
- `SYN` - 客户端选取一个随机数,我们称为 `x`,然后发送给服务器。
|
||||||
|
- `SYN ACK` - 服务器响应对应请求的 `ACK` 包中,包含了一个由服务器随机产生的数字,我们称为 `y`,并且把客户端发送的 `x+1`,一并返回给客户端。
|
||||||
|
- `ACK` - 客户端在从服务器接受到 `y` 之后把 `y` 加上 `1` 作为一个 `ACK` 包返回给服务器。
|
||||||
|
|
||||||
|
一旦三次握手完成后,客户端和服务器之间就可以开始交换数据。值得注意的是,当客户端发出最后一个 `ACK` 数据包后,就可以立刻向服务器发送应用数据包,而服务器则需要等到收到这个 `ACK` 数据包后才能接受应用数据包。
|
||||||
|
|
||||||
|
![](http://i.imgur.com/uERG2G2.png)
|
||||||
|
|
||||||
|
> 请注意,上图有点小问题,客户端发回的最后一个 ACK 包仅包含 `y+1`,上图应该是 `ACK:y+1` 而不是 `ACK:x+1,y+1`
|
||||||
|
|
||||||
|
然而,某些 HTTP/1.0 的实现试图通过新引入一个称为 `Connection: keep-alive` 的头信息来克服这一问题,这个头信息意味着告诉服务器“嘿,服务器,请不要关闭此连接,我还要用它”。但是,这并没有得到广泛的支持,问题依然存在。
|
||||||
|
|
||||||
|
除了无连接之外,HTTP 还是一个无状态的协议,即服务器不维护有关客户端的信息。因此每个请求必须给服务器必要的信息才能完成请求,每个请求都与之前的旧的请求无关。所以,这增加了推波助澜的作用,客户端除了需要新建大量连接之外,在每次连接中还需要发送许多重复的数据,这导致了带宽的大量浪费。
|
||||||
|
|
||||||
|
### `HTTP/1.1` - 1999
|
||||||
|
|
||||||
|
`HTTP/1.0` 经过仅仅 3 年,下一个版本,即 `HTTP/1.1` 就在 1999 年发布了,改进了它的前身很多问题,主要的改进包括:
|
||||||
|
|
||||||
|
- **增加了许多 HTTP 请求方法**,包括 `PUT`、`PATCH`、`HEAD`、`OPTIONS`、`DELETE`。
|
||||||
|
- **主机标识符** `Host` 在 `HTTP/1.0` 并不是必须的,而在 `HTTP/1.1` 是必须的。
|
||||||
|
- 如上所述的**持久连接**。在 `HTTP/1.0` 中每个连接只有一个请求并在该请求结束后被立即关闭,这导致了性能问题和增加了延迟。 `HTTP/1.1` 引入了持久连接,即连接在默认情况下是不关闭并保持开放的,这允许多个连续的请求使用这个连接。要关闭该连接只需要在头信息加入 `Connection: close`,客户通常在最后一个请求里发送这个头信息就能安全地关闭连接。
|
||||||
|
- 新版本还引入了“**管线化(pipelining)**”的支持,客户端可以不用等待服务器返回响应,就能在同一个连接内发送多个请求给服务器,而服务器必须以接收到的请求相同的序列发送响应。但是你可能会问了,客户端如何知道哪里是第一个响应下载完成而下一个响应内容开始的地方呢?要解决这个问题,头信息必须有 `Content-Length`,客户可以使用它来确定哪些响应结束之后可以开始等待下一个响应。
|
||||||
|
- 值得注意的是,为了从持久连接或管线化中受益, 头部信息必须包含 `Content-Length`,因为这会使客户端知道什么时候完成了传输,然后它可以发送下一个请求(持久连接中,以正常的依次顺序发送请求)或开始等待下一个响应(启用管线化时)。
|
||||||
|
- 但是,使用这种方法仍然有一个问题。那就是,如果数据是动态的,服务器无法提前知道内容长度呢?那么在这种情况下,你就不能使用这种方法中获益了吗?为了解决这个问题,`HTTP/1.1` 引进了分块编码。在这种情况下,服务器可能会忽略 `Content-Length` 来支持分块编码(更常见一些)。但是,如果它们都不可用,那么连接必须在请求结束时关闭。
|
||||||
|
- 在动态内容的情况下**分块传输**,当服务器在传输开始但无法得到 `Content-Length` 时,它可能会开始按块发送内容(一块接一块),并在传输时为每一个小块添加 `Content-Length`。当发送完所有的数据块后,即整个传输已经完成后,它发送一个空的小块,比如设置 `Content-Length` 为 0 ,以便客户端知道传输已完成。为了通知客户端块传输的信息,服务器在头信息中包含了 `Transfer-Encoding: chunked`。
|
||||||
|
- 不像 HTTP/1.0 中只有 Basic 身份验证方式,`HTTP/1.1` 包括摘要验证方式(digest authentication)和代理验证方式(proxy authentication)。
|
||||||
|
- 缓存。
|
||||||
|
- 范围请求(Byte Ranges)。
|
||||||
|
- 字符集。
|
||||||
|
- 内容协商(Content Negotiation)。
|
||||||
|
- 客户端 cookies。
|
||||||
|
- 支持压缩。
|
||||||
|
- 新的状态码。
|
||||||
|
- 等等。
|
||||||
|
|
||||||
|
我不打算在这里讨论所有 `HTTP/1.1` 的特性,因为你可以围绕这个话题找到很多关于这些的讨论。我建议你阅读 [`HTTP/1.0` 和 `HTTP/1.1` 版本之间的主要差异][5],希望了解更多可以读[原始的 RFC][6]。
|
||||||
|
|
||||||
|
`HTTP/1.1` 在 1999 年推出,到现在已经是多年前的标准。虽然,它比前一代改善了很多,但是网络日新月异,它已经垂垂老矣。相比之前,加载网页更是一个资源密集型任务,打开一个简单的网页已经需要建立超过 30 个连接。你或许会说,`HTTP/1.1` 具有持久连接,为什么还有这么多连接呢?其原因是,在任何时刻 `HTTP/1.1` 只能有一个未完成的连接。 `HTTP/1.1` 试图通过引入管线来解决这个问题,但它并没有完全地解决。因为一旦管线遇到了缓慢的请求或庞大的请求,后面的请求便被阻塞住,它们必须等待上一个请求完成。为了克服 `HTTP/1.1` 的这些缺点,开发人员开始实现一些解决方法,例如使用 spritesheets、在 CSS 中编码图像、单个巨型 CSS / JavaScript 文件、[域名切分][7]等。
|
||||||
|
|
||||||
|
### SPDY - 2009
|
||||||
|
|
||||||
|
谷歌走在业界前列,为了使网络速度更快,提高网络安全,同时减少网页的等待时间,他们开始实验替代的协议。在 2009 年,他们宣布了 `SPDY`。
|
||||||
|
|
||||||
|
> `SPDY` 是谷歌的商标,而不是一个缩写。
|
||||||
|
|
||||||
|
显而易见的是,如果我们继续增加带宽,网络性能开始的时候能够得到提升,但是到了某个阶段后带来的性能提升就很有限了。但是如果把这些优化放在等待时间上,比如减少等待时间,将会有持续的性能提升。这就是 `SPDY` 优化之前的协议的核心思想,减少等待时间来提升网络性能。
|
||||||
|
|
||||||
|
> 对于那些不知道其中区别的人,等待时间就是延迟,即数据从源到达目的地需要多长时间(单位为毫秒),而带宽是每秒钟数据的传输量(比特每秒)。
|
||||||
|
|
||||||
|
`SPDY` 的特点包括:复用、压缩、优先级、安全性等。我不打算展开 `SPDY` 的细节。在下一章节,当我们将介绍 `HTTP/2`,这些都会被提到,因为 `HTTP/2` 大多特性是从 `SPDY` 受启发的。
|
||||||
|
|
||||||
|
`SPDY` 没有试图取代 HTTP,它是处于应用层的 HTTP 之上的一个传输层,它只是在请求被发送之前做了一些修改。它开始成为事实标准,大多数浏览器都开始支持了。
|
||||||
|
|
||||||
|
2015年,谷歌不想有两个相互竞争的标准,所以他们决定将其合并到 HTTP 协议,这样就导致了 `HTTP/2` 的出现和 `SPDY` 的废弃。
|
||||||
|
|
||||||
|
### `HTTP/2` - 2015
|
||||||
|
|
||||||
|
现在想必你明白了为什么我们需要另一个版本的 HTTP 协议了。 `HTTP/2` 是专为了低延迟地内容传输而设计。主要特点和与 `HTTP/1.1` 的差异包括:
|
||||||
|
|
||||||
|
- 使用二进制替代明文
|
||||||
|
- 多路传输 - 多个异步 HTTP 请求可以使用单一连接
|
||||||
|
- 报头使用 HPACK 压缩
|
||||||
|
- 服务器推送 - 单个请求多个响应
|
||||||
|
- 请求优先级
|
||||||
|
- 安全性
|
||||||
|
|
||||||
|
![](http://i.imgur.com/S85j8gg.png)
|
||||||
|
|
||||||
|
#### 1. 二进制协议
|
||||||
|
|
||||||
|
`HTTP/2` 通过使其成为一个二进制协议以解决 HTTP/1.x 中存在的延迟问题。作为一个二进制协议,它更容易解析,但可读性却不如 `HTTP/1.x`。帧(frames)和流(stream)的概念组成了 `HTTP/2` 的主要部分。
|
||||||
|
|
||||||
|
**帧和流**
|
||||||
|
|
||||||
|
现在 HTTP 消息是由一个或多个帧组成的。`HEADERS` 帧承载了元数据(meta data),`DATA` 帧则承载了内容。还有其他类型的帧(`HEADERS`、`DATA`、`RST_STREAM`、`SETTINGS`、`PRIORITY` 等等),这些你可以通过[HTTP/2 规范][3]来了解。
|
||||||
|
|
||||||
|
每个 `HTTP/2` 请求和响应都被赋予一个唯一的流 ID,并切分成帧。帧就是一小片二进制数据。帧的集合称为流,每个帧都有个标识了其所属流的流 ID,所以在同一个流下的每个帧具有共同的报头。值得注意的是,除了流 ID 是唯一的之外,由客户端发起的请求使用了奇数作为流 ID,从来自服务器的响应使用了偶数作为流 ID。
|
||||||
|
|
||||||
|
除了 `HEADERS` 帧和 `DATA` 帧,另一个值得一提的帧是 `RST_STREAM`。这是一个特殊的帧类型,用来中止流,即客户可以发送此帧让服务器知道,我不再需要这个流了。在 `HTTP/1.1` 中让服务器停止给客户端发送响应的唯一方法是关闭连接,这样造成了延迟增加,因为之后要发送请求时,就要必须打开一个新的请求。而在 `HTTP/2` ,客户端可以使用 `RST_STREAM` 来停止接收特定的数据流,而连接仍然打开着,可以被其他请求使用。
|
||||||
|
|
||||||
|
#### 2. 多路传输
|
||||||
|
|
||||||
|
因为 `HTTP/2` 是一个二进制协议,而且如上所述它使用帧和流来传输请求与响应,一旦建立了 TCP 连接,相同连接内的所有流都可以同过这个 TCP 连接异步发送,而不用另外打开连接。反过来说,服务器也可以使用同样的异步方式返回响应,也就是说这些响应可以是无序的,客户端使用分配的流 ID 来识别数据包所属的流。这也解决了 HTTP/1.x 中请求管道被阻塞的问题,即客户端不必等待占用时间的请求而其他请求仍然可以被处理。
|
||||||
|
|
||||||
|
#### 3. HPACK 请求头部压缩
|
||||||
|
|
||||||
|
RFC 花了一篇文档的篇幅来介绍针对发送的头信息的优化,它的本质是当我们在同一客户端上不断地访问服务器时,许多冗余数据在头部中被反复发送,有时候仅仅是 cookies 就能增加头信息的大小,这会占用许多宽带和增加传输延迟。为了解决这个问题,`HTTP/2` 引入了头信息压缩。
|
||||||
|
|
||||||
|
![](http://i.imgur.com/3IPWXvR.png)
|
||||||
|
|
||||||
|
不像请求和响应那样,头信息中的信息不会以 `gzip` 或者 `compress` 等格式压缩。而是采用一种不同的机制来压缩头信息,客户端和服务器同时维护一张头信息表,储存了使用了哈夫曼编码进行编码后的头信息的值,并且后续请求中若出现同样的字段则忽略重复值(例如用户代理(user agent)等),只发送存在两边信息表中它的引用即可。
|
||||||
|
|
||||||
|
我们说的头信息,它们同 HTTP/1.1 中一样,并在此基础上增加了一些伪头信息,如 `:scheme`,`:host` 和 `:path`。
|
||||||
|
|
||||||
|
#### 4. 服务器推送
|
||||||
|
|
||||||
|
服务器推送是 `HTTP/2` 的另一个巨大的特点。对于服务器来说,当它知道客户端需要一定的资源后,它可以把数据推送到客户端,即使客户端没有请求它。例如,假设一个浏览器在加载一个网页时,它解析了整个页面,发现有一些内容必须要从服务端获取,然后发送相应的请求到服务器以获取这些内容。
|
||||||
|
|
||||||
|
服务器推送减少了传输这些数据需要来回请求的次数。它是如何做到的呢?服务器通过发送一个名字为 `PUSH_PROMISE` 特殊的帧通知到客户端“嘿,我准备要发送这个资源给你了,不要再问我要了。”这个 `PUSH_PROMISE` 帧与要产生推送的流联系在一起,并包含了要推送的流 ID,也就是说这个流将会被服务器推送到客户端上。
|
||||||
|
|
||||||
|
#### 5. 请求优先级
|
||||||
|
|
||||||
|
当流被打开的时候,客户端可以在 `HEADERS` 帧中包含优先级信息来为流指定优先级。在任何时候,客户端都可以发送 `PRIORITY` 帧来改变流的优先级。
|
||||||
|
|
||||||
|
如果没有任何优先级信息,服务器将异步地无序地处理这些请求。如果流分配了优先级,服务器将基于这个优先级来决定需要分配多少资源来处理这个请求。
|
||||||
|
|
||||||
|
#### 6. 安全性
|
||||||
|
|
||||||
|
在是否强制使用 `TLS` 来增加安全性的问题上产生了大范围的讨论,讨论的结果是不强制使用。然而大多数厂商只有在使用 `TLS` 时才能使用 `HTTP/2`。所以 `HTTP/2` 虽然规范上不要求加密,但是加密已经约定俗成了。这样,在 `TLS` 之上实现 `HTTP/2` 就有了一些强制要求,比如,`TLS` 的最低版本为 `1.2`,必须达到某种级别的最低限度的密钥大小,需要布署 ephemeral 密钥等等。
|
||||||
|
|
||||||
|
到现在 `HTTP/2` 已经[完全超越了 SPDY][4],并且还在不断成长,HTTP/2 有很多关系性能的提升,我们应该开始布署它了。
|
||||||
|
|
||||||
|
如果你想更深入的了解细节,请访问[该规范的链接][1]和[ HTTP/2 性能提升演示的链接][2]。请在留言板写下你的疑问或者评论,最后如果你发现有错误,请同样留言指出。
|
||||||
|
|
||||||
|
这就是全部了,我们之后再见~
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://kamranahmed.info/blog/2016/08/13/http-in-depth/
|
||||||
|
|
||||||
|
作者:[Kamran Ahmed][a]
|
||||||
|
译者:[NearTan](https://github.com/NearTan)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://github.com/kamranahmedse
|
||||||
|
|
||||||
|
[1]: https://http2.github.io/http2-spec
|
||||||
|
[2]: http://www.http2demo.io/
|
||||||
|
[3]: https://http2.github.io/http2-spec/#FrameTypes
|
||||||
|
[4]: http://caniuse.com/#search=http2
|
||||||
|
[5]: http://www.ra.ethz.ch/cdstore/www8/data/2136/pdf/pd1.pdf
|
||||||
|
[6]: https://tools.ietf.org/html/rfc2616
|
||||||
|
[7]: https://www.maxcdn.com/one/visual-glossary/domain-sharding-2/
|
@ -0,0 +1,58 @@
|
|||||||
|
我的 Android 开发者之路以及我在其中学到了什么
|
||||||
|
=============================
|
||||||
|
|
||||||
|
大家都说所有的关系都需要经历两年、七年甚至十年的磨砺。我忘了是谁说的这句话,但肯定有人在几年前这么跟我说过。
|
||||||
|
|
||||||
|
下周是我来悉尼两周年,所以我想现在正是我写这篇文章的好时候。
|
||||||
|
|
||||||
|
去年五月份参加 I/O 年会的时候,我遇到了亚斯曼女士,她十分漂亮。她向我询问我是如何成长为一名安卓开发者的,当我说完我的经历时,她认为我应该写个博客记下来。所以亚斯曼,如你所愿,虽然迟了点,但好过没做。;)
|
||||||
|
|
||||||
|
### 故事的开始
|
||||||
|
|
||||||
|
如果有件事我可能希望你知道,那就是我发现自己有选择困难症。你最好的朋友是谁?你最喜欢的食物是什么?你应该给你的玩具熊猫命名吗?我连这些问题都不知道该怎么回答才好。所以你可以想象到,16 岁的、即将高中毕业的我对于专业选择根本就没有任何想法。那我最初申请的大学是?在交表给注册员前,我在她面前逐字掂量着写下这个打算申请的专业(商业经济学)。
|
||||||
|
|
||||||
|
可我最后去了另外一间学校,就读电子与通信工程。大一时我有一门计算机编程课程。但我很讨厌编程,十分地讨厌。关于编程的一切我都一无所知。我曾发誓再也不要写代码了。
|
||||||
|
|
||||||
|
我大学毕业后的第一份工作是在英特尔做产品工程师并在那呆了两年。我很迷茫,无所适从,整天长时间工作。这在我意料之中,身为成年人难道不该努力工作吗?可之后菲律宾的半导体行业开始呈现颓势,大批工厂纷纷倒闭,以前由我们维护一些产品被转移到其他分公司。我便决定去找另一份工作而不是等着被裁员,因为被裁员后我都不知道自己多久才能找到另一份工作。
|
||||||
|
|
||||||
|
### 现在呢?
|
||||||
|
|
||||||
|
我想留在城市里找到一份工作,但我不想呆在正在没落的半导体行业里了。但话说回来,我又不知道该做什么好。对了,我可是拿了毕业证书的工程师,所以从技术上来说我可以在电信运营商或电视台找到工作。可这种时候,如果想入职电信运营商,我应该在大学毕业之际就去他们那实习,这样更容易被录用。可惜我没有,所以我放弃了这个想法。虽然有很多软件开发人员的招聘信息,但我讨厌编程,所以我真的不知道怎么做才好。
|
||||||
|
|
||||||
|
接下来是我第一个幸运的机遇,我很幸运地遇到了信任我的上司,我也和她坦诚了我什么都不会。之后我不得不边工作边学习,一开始这个过程很漫长。无需多言,我在这份工作上学到了很多,也结识了很多很好的人,与我一起的是一群很厉害的同事(我们曾开发出安装在 SIM 卡上的 APP)。但更重要的是我开始踏上了软件开发的征途。
|
||||||
|
|
||||||
|
最后我做得更多是一些公司的琐事(十分无聊)直到项目完结。换句话说,我总在是在办公室里闲逛并坐等发薪。之后我发现这确实是在浪费时间,2009 年的时候,我不停地接触到关于谷歌的新系统 Android 的消息,并得知它的 SDK 已经公布!是时候尝试一波了。于是我安装了所有相关软件并着手 Android 开发。
|
||||||
|
|
||||||
|
### 事情变得有趣了
|
||||||
|
|
||||||
|
所以现在我能够构建一个在运行在仿真器的 Hello World 应用,在我看来意味着我有胜任安卓开发工作的能力。我加入了一个创业公司,并且再次坦诚我不知道该怎么做,我只是接触过一些;但如果你们愿意付薪水给我继续尝试,我们就可以成为朋友。然后我很幸运地遇到另一个机遇。
|
||||||
|
|
||||||
|
那时成为开发者是一件令人欣喜的事。StackOverflow 上的 Android 开发社区非常小,我们都在相互交流学习,说真的,我认为里面的所有人都很友好、很豪迈(注 1)!
|
||||||
|
|
||||||
|
我最后去了一家企业,这家企业的移动开发团队在马尼拉、悉尼、纽约都设有办公地点。而我是马尼拉办公地点的第一个安卓开发人员,但那时我很习惯,并没有在意。
|
||||||
|
|
||||||
|
在那里我认识了最后令我永远感激的引荐我参与 Domain 项目的人。Domain 项目不管在个人或职业上对我来说都意味深重。我和一支很有才华的团队一起共事,也从没见过一个公司能如此执着于一款产品。Domain 让我实现了参加 I/O 年会的梦。与他们共事后我懂得了很多之前没想到的可爱特性(注 2)。这是又一个幸运的机遇,我是说最大限度地利用它。
|
||||||
|
|
||||||
|
### 然后呢?
|
||||||
|
|
||||||
|
我想说的是,虽然这些年都在晃荡,但至少我很诚实,对吧?如上就是我所学到的全部东西。说一句「我不懂」没什么可怕的。有时候我们是该装懂,但更多时候我们需要坦诚地接受这样一个事实:我们还不懂。
|
||||||
|
|
||||||
|
别害怕尝试新事物,不管它让你感觉多害怕。我知道说比做简单。但总有一些东西能让你鼓起勇气动手去尝试的(注3)。Lundagin mo, baby!(LCTT 译注:一首歌名)
|
||||||
|
|
||||||
|
|
||||||
|
注 1: 我翻阅着以前在 StackOverflow 提的问题,认真想想,如果现在我问他们这些,估计会收到很多「你是谁啊,傻瓜」的评论。我不知道是不是因为我老了,而且有些愤世妒俗。但关键是,我们有缘在同一个社区中,所以大家相互之间友善些,好吗?
|
||||||
|
注 2: 这一点写在另一篇文章里了。
|
||||||
|
注 3: 我还清晰地记得第一次申请 Android 开发职位的情形:我写完求职信后又通读了一遍,提交前鼠标在发送按钮上不断徘徊,深呼吸之后我趁改变主意之前把它发出去了。
|
||||||
|
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.zdominguez.com/2016/08/winging-it-how-i-got-to-be-android-dev.html
|
||||||
|
|
||||||
|
作者:[Zarah Dominguez][a]
|
||||||
|
译者:[JianhuanZhuo](https://github.com/JianhuanZhuo)
|
||||||
|
校对:[PurlingNayuki](https://github.com/PurlingNayuki)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://plus.google.com/102371834744366149197
|
@ -0,0 +1,102 @@
|
|||||||
|
LXDE、Xfce 及 MATE 桌面环境下的又一系统监视器应用:Multiload-ng
|
||||||
|
======
|
||||||
|
|
||||||
|
[Multiload-ng][1] 是一个 GTK2 图形化系统监视器应用,可集成到 Xfce、LXDE 及 MATE 的桌面面板中, 它 fork 自原来的 GNOME Multiload 应用。它也可以运行在一个独立的窗口中。
|
||||||
|
|
||||||
|
![](https://2.bp.blogspot.com/-U8CFzhSPJho/V7GigDbcLWI/AAAAAAAAYS0/pJMM6Rt5-HkbKljmxzP4-v0oGGxjvH8AgCLcB/s1600/multiload-ng-lxde.png)
|
||||||
|
|
||||||
|
Multiload-ng 的特点有:
|
||||||
|
|
||||||
|
- 支持以下资源的图形块: CPU,内存,网络,交换空间,平均负载,磁盘以及温度;
|
||||||
|
- 高度可定制;
|
||||||
|
- 支持配色方案;
|
||||||
|
- 自动适应容器(面板或窗口)的改变;
|
||||||
|
- 极低的 CPU 和内存占用;
|
||||||
|
- 提供基本或详细的提示信息;
|
||||||
|
- 可自定义双击触发的动作。
|
||||||
|
|
||||||
|
相比于原来的 Multiload 应用,Multiload-ng 含有一个额外的图形块(温度),以及更多独立的图形自定义选项,例如独立的边框颜色,支持配色方案,可根据自定义的动作对鼠标的点击做出反应,图形块的方向可以被设定为与面板的方向无关。
|
||||||
|
|
||||||
|
它也可以运行在一个独立的窗口中,而不需要面板:
|
||||||
|
|
||||||
|
![](https://1.bp.blogspot.com/-hHoipwFlHrg/V7Gw2s107zI/AAAAAAAAYTQ/fS5OtiL7VvwDEzr6qO_gdEA_qB9YvJa5gCLcB/s400/multiload-ng-standalone.png)
|
||||||
|
|
||||||
|
另外,它的 GitHub page 上说还会带来更多的图形块支持。
|
||||||
|
|
||||||
|
下图展示的是在带有一个垂直面板的 Xubuntu 16.04 中,该应用分别处于水平和垂直方向的效果:
|
||||||
|
|
||||||
|
![](https://3.bp.blogspot.com/-xa0OML8T-lg/V7Gixksbt8I/AAAAAAAAYS4/Jxo-MukDh3sYlOOk9A1YGtARmte490g8ACLcB/s400/multiload-ng-xfce-horizontal.png)
|
||||||
|
|
||||||
|
![](https://1.bp.blogspot.com/-WAD5MdDObD8/V7GixgVU0DI/AAAAAAAAYS8/uMhHJri1GJccRWvmf_tZkYeenVdxiENQwCLcB/s400/multiload-ng-xfce-vertical.png)
|
||||||
|
|
||||||
|
这个应用的偏好设置窗口虽然不是非常好看,但是有计划去改进它:
|
||||||
|
|
||||||
|
![](https://2.bp.blogspot.com/-P-ophDpc-gI/V7Gi_54b7JI/AAAAAAAAYTA/AHQck_JF_RcwZ1KbgHbaO2JRt24ZZdO3gCLcB/s320/multiload-ng-preferences.png)
|
||||||
|
|
||||||
|
Multiload-ng 当前使用的是 GTK2,所以它不能在构建自 GTK3 下的 Xfce 或 MATE 桌面环境(面板)下工作。
|
||||||
|
|
||||||
|
对于 Ubuntu 系统而言,只有 Ubuntu MATE 16.10 使用 GTK3。但是鉴于 MATE 的系统监视器应用也是 Multiload GNOME 的一个分支,所以它们大多数的功能相同(除了 Multiload-ng 提供的额外自定义选项和温度图形块)。
|
||||||
|
|
||||||
|
该应用的[愿望清单][2] 中提及到了计划支持 GTK3 的集成以及各种各样的改进,例如温度块资料的更多来源,能够显示十进制(KB、MB、GB……)或二进制(KiB、MiB、GiB……)单位等等。
|
||||||
|
|
||||||
|
### 安装 Multiload-ng
|
||||||
|
|
||||||
|
请注意因为依赖的关系, Multiload-ng 不能在 Lubuntu 14.04 上构建。
|
||||||
|
|
||||||
|
Multiload-ng 可在 WebUpd8 的主 PPA (针对 Ubuntu 14.04 - 16.04 / Linux Mint 17.x 和 18)中获取到。可以使用下面的命令来添加 PPA 并更新软件源:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo add-apt-repository ppa:nilarimogard/webupd8
|
||||||
|
sudo apt update
|
||||||
|
```
|
||||||
|
|
||||||
|
然后可以使用下面的命令来安装这个应用:
|
||||||
|
|
||||||
|
- 对于 LXDE (Lubuntu):
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt install lxpanel-multiload-ng-plugin
|
||||||
|
```
|
||||||
|
|
||||||
|
- 对于 Xfce (Xubuntu,Linux Mint Xfce):
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt install xfce4-multiload-ng-plugin
|
||||||
|
```
|
||||||
|
|
||||||
|
- 对于 MATE (Ubuntu MATE,Linux Mint MATE):
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt install mate-multiload-ng-applet
|
||||||
|
```
|
||||||
|
|
||||||
|
- 独立安装 (不需要集成到面板):
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt install multiload-ng-standalone
|
||||||
|
```
|
||||||
|
|
||||||
|
一旦安装完毕,便可以像其他应用那样添加到桌面面板中了。需要注意的是在 LXDE 中,Multiload-ng 不能马上出现在面板清单中,除非重新启动面板。你可以通过重启会话(登出后再登录)或者使用下面的命令来重启面板:
|
||||||
|
|
||||||
|
```
|
||||||
|
lxpanelctl restart
|
||||||
|
```
|
||||||
|
|
||||||
|
独立的 Multiload-ng 应用可以像其他正常应用那样从菜单中启动。
|
||||||
|
|
||||||
|
如果要下载源码或报告 bug 等,请看 Multiload-ng 的 [GitHub page][3]。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.webupd8.org/2016/08/alternative-system-monitor-applet-for.html
|
||||||
|
|
||||||
|
作者:[Andrew][a]
|
||||||
|
译者:[FSSlc](https://github.com/FSSlc)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.webupd8.org/p/about.html
|
||||||
|
[1]: https://github.com/udda/multiload-ng
|
||||||
|
[2]: https://github.com/udda/multiload-ng/wiki/Wishlist
|
||||||
|
[3]: https://github.com/udda/multiload-ng
|
@ -0,0 +1,111 @@
|
|||||||
|
使用 HTTP/2 服务端推送技术加速 Node.js 应用
|
||||||
|
=========================================================
|
||||||
|
|
||||||
|
四月份,我们宣布了对 [HTTP/2 服务端推送技术][3]的支持,我们是通过 HTTP 的 [Link 头部](https://www.w3.org/wiki/LinkHeader)来实现这项支持的。我的同事 John 曾经通过一个例子演示了[在 PHP 里支持服务端推送功能][4]是多么的简单。
|
||||||
|
|
||||||
|
![](https://blog.cloudflare.com/content/images/2016/08/489477622_594bf9e3d9_z.jpg)
|
||||||
|
|
||||||
|
我们想让现今使用 Node.js 构建的网站能够更加轻松的获得性能提升。为此,我们开发了 [netjet][1] 中间件,它可以解析应用生成的 HTML 并自动添加 Link 头部。当在一个示例的 Express 应用中使用这个中间件时,我们可以看到应用程序的输出多了如下 HTTP 头:
|
||||||
|
|
||||||
|
![](https://blog.cloudflare.com/content/images/2016/08/2016-08-11_13-32-45.png)
|
||||||
|
|
||||||
|
[本博客][5]是使用 [Ghost](https://ghost.org/)(LCTT 译注:一个博客发布平台)进行发布的,因此如果你的浏览器支持 HTTP/2,你已经在不知不觉中享受了服务端推送技术带来的好处了。接下来,我们将进行更详细的说明。
|
||||||
|
|
||||||
|
netjet 使用了带有定制插件的 [PostHTML](https://github.com/posthtml/posthtml) 来解析 HTML。目前,netjet 用它来查找图片、脚本和外部 CSS 样式表。你也可以用其它的技术来实现这个。
|
||||||
|
|
||||||
|
在响应过程中增加 HTML 解析器有个明显的缺点:这将增加页面加载的延时(到加载第一个字节所花的时间)。大多数情况下,所新增的延时被应用里的其他耗时掩盖掉了,比如数据库访问。为了解决这个问题,netjet 包含了一个可调节的 LRU 缓存,该缓存以 HTTP 的 ETag 头部作为索引,这使得 netjet 可以非常快的为已经解析过的页面插入 Link 头部。
|
||||||
|
|
||||||
|
不过,如果我们现在从头设计一款全新的应用,我们就应该考虑把页面内容和页面中的元数据分开存放,从而整体地减少 HTML 解析和其它可能增加的延时了。
|
||||||
|
|
||||||
|
任意的 Node.js HTML 框架,只要它支持类似 Express 这样的中间件,netjet 都是能够兼容的。只要把 netjet 像下面这样加到中间件加载链里就可以了。
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
var express = require('express');
|
||||||
|
var netjet = require('netjet');
|
||||||
|
var root = '/path/to/static/folder';
|
||||||
|
|
||||||
|
express()
|
||||||
|
.use(netjet({
|
||||||
|
cache: {
|
||||||
|
max: 100
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
.use(express.static(root))
|
||||||
|
.listen(1337);
|
||||||
|
```
|
||||||
|
|
||||||
|
稍微加点代码,netjet 也可以摆脱 HTML 框架,独立工作:
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
var http = require('http');
|
||||||
|
var netjet = require('netjet');
|
||||||
|
|
||||||
|
var port = 1337;
|
||||||
|
var hostname = 'localhost';
|
||||||
|
var preload = netjet({
|
||||||
|
cache: {
|
||||||
|
max: 100
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
var server = http.createServer(function (req, res) {
|
||||||
|
preload(req, res, function () {
|
||||||
|
res.statusCode = 200;
|
||||||
|
res.setHeader('Content-Type', 'text/html');
|
||||||
|
res.end('<!doctype html><h1>Hello World</h1>');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
server.listen(port, hostname, function () {
|
||||||
|
console.log('Server running at http://' + hostname + ':' + port+ '/');
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
[netjet 文档里][1]有更多选项的信息。
|
||||||
|
|
||||||
|
### 查看推送了什么数据
|
||||||
|
|
||||||
|
![](https://blog.cloudflare.com/content/images/2016/08/2016-08-02_10-49-33.png)
|
||||||
|
|
||||||
|
访问[本文][5]时,通过 Chrome 的开发者工具,我们可以轻松的验证网站是否正在使用服务器推送技术(LCTT 译注: Chrome 版本至少为 53)。在“Network”选项卡中,我们可以看到有些资源的“Initiator”这一列中包含了`Push`字样,这些资源就是服务器端推送的。
|
||||||
|
|
||||||
|
不过,目前 Firefox 的开发者工具还不能直观的展示被推送的资源。不过我们可以通过页面响应头部里的`cf-h2-pushed`头部看到一个列表,这个列表包含了本页面主动推送给浏览器的资源。
|
||||||
|
|
||||||
|
希望大家能够踊跃为 netjet 添砖加瓦,我也乐于看到有人正在使用 netjet。
|
||||||
|
|
||||||
|
### Ghost 和服务端推送技术
|
||||||
|
|
||||||
|
Ghost 真是包罗万象。在 Ghost 团队的帮助下,我把 netjet 也集成到里面了,而且作为测试版内容可以在 Ghost 的 0.8.0 版本中用上它。
|
||||||
|
|
||||||
|
如果你正在使用 Ghost,你可以通过修改 config.js、并在`production`配置块中增加 `preloadHeaders` 选项来启用服务端推送。
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
production: {
|
||||||
|
url: 'https://my-ghost-blog.com',
|
||||||
|
preloadHeaders: 100,
|
||||||
|
// ...
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Ghost 已经为其用户整理了[一篇支持文档][2]。
|
||||||
|
|
||||||
|
### 总结
|
||||||
|
|
||||||
|
使用 netjet,你的 Node.js 应用也可以使用浏览器预加载技术。并且 [CloudFlare][5] 已经使用它在提供了 HTTP/2 服务端推送了。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://blog.cloudflare.com/accelerating-node-js-applications-with-http-2-server-push/
|
||||||
|
|
||||||
|
作者:[Terin Stock][a]
|
||||||
|
译者:[echoma](https://github.com/echoma)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://blog.cloudflare.com/author/terin-stock/
|
||||||
|
[1]: https://www.npmjs.com/package/netjet
|
||||||
|
[2]: http://support.ghost.org/preload-headers/
|
||||||
|
[3]: https://www.cloudflare.com/http2/server-push/
|
||||||
|
[4]: https://blog.cloudflare.com/using-http-2-server-push-with-php/
|
||||||
|
[5]: https://blog.cloudflare.com/accelerating-node-js-applications-with-http-2-server-push/
|
54
published/201609/20160817 Build an integration for GitHub.md
Normal file
54
published/201609/20160817 Build an integration for GitHub.md
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
为 Github 创造集成件(Integration)
|
||||||
|
=============
|
||||||
|
|
||||||
|
![](https://cloud.githubusercontent.com/assets/4432363/17537572/934e2f9c-5e52-11e6-9c24-50c7be5b5ae3.png)
|
||||||
|
|
||||||
|
现在你可以从我们的 [集成件目录][1]里面找到更多工具。这个目录目前有超过 15 个分类 — 从 [API 管理][2] 到 [应用监控][3], Github 的集成件可以支持您的开发周期的每一个阶段。
|
||||||
|
|
||||||
|
我们邀请了具有不同层面的专长的开发人员,来创造有助于开发者更好的工作的集成件。如果你曾经为 Github 构建过一个很棒的集成件,我们希望来让更多人知道它! [Gitter][4]、[AppVeyor][5] 和 [ZenHub][6] 都做到了,你也可以!
|
||||||
|
|
||||||
|
### 我们在寻找什么?
|
||||||
|
|
||||||
|
良好的软件开发依赖于上乘的工具,开发人员如今有了更多的选择,无论是语言、框架、工作流程,还是包含了其他因素的环境。我们正在寻找能够创造更好的整体开发体验的开发工具。
|
||||||
|
|
||||||
|
#### 进入集成件目录清单的指南:
|
||||||
|
|
||||||
|
- 稳步增多( 你的 Github OAuth 接口当前可以支持超过 500 个用户 )
|
||||||
|
- 查看我们的 [技术要求][7]
|
||||||
|
- 遵从[服务条款][8] 和[隐私政策][9]
|
||||||
|
- 专注于软件开发生命周期
|
||||||
|
|
||||||
|
### 有帮助的资源
|
||||||
|
|
||||||
|
如果想要被列在目录里,请按照[列出需求页][10]中概述的步骤。
|
||||||
|
|
||||||
|
你也应该阅读我们的[营销指南][11]和[已有目录清单][12]来更好的了解如何把它们全都放在一起。请把你的列表的内容记录在一个[私密 gist][13] 中(markdown 格式),并且通过邮件联系我们 <partnerships@github.com>。 如果你有任何问题,不要犹疑,请联系 <partnerships@github.com>。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://github.com/blog/2226-build-an-integration-for-github
|
||||||
|
|
||||||
|
作者:[chobberoni][a]
|
||||||
|
译者:[Bestony](https://github.com/Bestony)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://github.com/chobberoni
|
||||||
|
|
||||||
|
[1]: https://github.com/integrations
|
||||||
|
[2]: https://github.com/integrations/feature/api-management
|
||||||
|
[3]: https://github.com/integrations/feature/monitoring
|
||||||
|
[4]: https://github.com/integrations/feature/monitoring
|
||||||
|
[5]: https://github.com/integrations/appveyor
|
||||||
|
[6]: https://github.com/integrations/zenhub
|
||||||
|
[7]: https://developer.github.com/integrations-directory/getting-listed/#technical-requirements
|
||||||
|
[8]: https://help.github.com/articles/github-terms-of-service/
|
||||||
|
[9]: https://help.github.com/articles/github-privacy-policy/
|
||||||
|
[10]: https://developer.github.com/integrations-directory/getting-listed/
|
||||||
|
[11]: https://developer.github.com/integrations-directory/marketing-guidelines/
|
||||||
|
[12]: https://github.com/integrations
|
||||||
|
[13]: https://gist.github.com/
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -0,0 +1,79 @@
|
|||||||
|
Turtl:安全、开源的 Evernote 替代品
|
||||||
|
=============================
|
||||||
|
|
||||||
|
Turtl 是一个安全、开源的 Evernote 替代品,在Linux、Windows、Mac 和 Android 等系统上都能使用。iOS版本仍在开发当中,Firefox 和 Chrome 也有扩展程序可以使用。
|
||||||
|
|
||||||
|
![](https://3.bp.blogspot.com/-cNoUUjaU4A0/V7MFKCasZJI/AAAAAAAAYTk/r7oWe-z_HB87hDvlKLViiiHUMfagnC6LQCLcB/s400/turtl-desktop-linux.png)
|
||||||
|
|
||||||
|
这个产品仍在测试阶段,它能够让你把你的笔记(便签编辑器支持 Markdown)、网站书签、密码、文档、图片等单独放在一个隐秘地方。
|
||||||
|
|
||||||
|
笔记可以按模块组织起来,支持嵌套,也可以和其他 Turtl 用户分享。
|
||||||
|
|
||||||
|
![](https://2.bp.blogspot.com/-G-Ln3T1c2QA/V7MFmrqkukI/AAAAAAAAYTs/dXMPEB9MPREicixlEJlQVqg9SFjBX1pwgCLcB/s400/turtl-boards.png)
|
||||||
|
|
||||||
|
你可以给你的笔记打上标签。Turtl 通过创建时间、最后修改时间或者标签来找你的笔记。
|
||||||
|
|
||||||
|
这个是便签编辑器(文件便签):
|
||||||
|
|
||||||
|
![](https://1.bp.blogspot.com/-8cNHV69iCWM/V7MFX7sBlMI/AAAAAAAAYTo/ZUVTYwiCSy8uzrVKdf6NcsQZlHtylIyvgCEw/s400/turtl-edit-note.png)
|
||||||
|
|
||||||
|
那么安全性如何呢?Turtl 会在保存数据之前加密,使用的是一个加密密钥,而密码并不保存在服务器上。只有你和你想要分享的人能获取数据。你可以从[这里][1]获得更多关于 Turtl 安全和加密的信息。
|
||||||
|
|
||||||
|
更新(感谢 Dimitry!):根据[错误反馈][2],Turtl 有个严重的安全性问题。Turtl 允许创建多个相同用户名的账号,却只使用密码来区分它们。希望能马上修复这个问题。
|
||||||
|
|
||||||
|
Turtl 团队提供了一个托管服务来同步你的记录,它是完全免费的,”除非你的文件足够大,或者你需要更好的服务”,在我写这篇文章的时候这个高级服务还不能用。
|
||||||
|
|
||||||
|
并且你也不一定要用这个托管服务,因为就像其桌面应用和手机应用一样,这个自托管服务器也是一个自由、开源的软件,所以你可以自己搭建一个 [Turtl 服务器][3]。
|
||||||
|
|
||||||
|
Turtl 没有像 Evernote 那么多的功能,但它在它的[计划][4]中也有一些新的功能,比如:支持导入/导出文本和Evernote 格式的数据、原生支持 PDF 阅读器、界面锁定等。
|
||||||
|
|
||||||
|
不得不提醒的是,每次启动都要输入密码,虽然安全,但有时候实在是麻烦。
|
||||||
|
|
||||||
|
###下载 Turtl
|
||||||
|
|
||||||
|
[下载 Turtl 应用][5](二进制文件支持 Linux (32位/64位)、Windows 64 位、Mac 64位、Android,以及 Chrome 和Firefox 浏览器插件)
|
||||||
|
|
||||||
|
**更新**:Turtl 用了一个新的服务器,注销然后在登录框的下面选择高级设置,把 Turtl 服务器设置为 "https://api.turtlapp.com/v2"(没有引号)。
|
||||||
|
|
||||||
|
下载源代码(桌面应用、移动应用和服务器)、反馈问题等,参见 Turtl 的 [GitHub][6] 项目站点。
|
||||||
|
|
||||||
|
Arch Linux 用户可以通过 [AUR][7] 来安装 Turtl。
|
||||||
|
|
||||||
|
要在 Linux 上安装,把安装包解压后运行 install.sh,安装之前请确保 ~/.local/share/applications 目录存在,若不存在请自行创建:
|
||||||
|
|
||||||
|
```
|
||||||
|
mkdir -p ~/.local/share/applications
|
||||||
|
```
|
||||||
|
|
||||||
|
注意:如果使用 sudo 命令安装,那么只有 root 用户才能使用。所以,要么不用 sudo 命令安装,要么在安装完成后修改权限。你可以参考[AUR 软件包的设置][8]来了解如何修改权限。
|
||||||
|
|
||||||
|
使用如下命令把 Turtl 安装到 ~/turtl 文件夹下(假定你已经把安装包解压在你的家目录下了):
|
||||||
|
|
||||||
|
````
|
||||||
|
~/turtl-*/install.sh ~/turtl
|
||||||
|
```
|
||||||
|
|
||||||
|
可以使用 ~/.turtl 代替 ~/turtl 把 Turtl 安装到你的家目录的隐藏文件夹下。你也可以用些小技巧把它隐藏起来。
|
||||||
|
|
||||||
|
如果 Turtl 没有在你的 Unity Dash 上显示出来,请注销/登录以重启会话。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.webupd8.org/2016/08/turtl-secure-open-source-evernote.html
|
||||||
|
|
||||||
|
作者:[Andrew][a]
|
||||||
|
译者:[chisper](https://github.com/chisper)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.webupd8.org/p/about.html
|
||||||
|
[1]: https://turtl.it/docs/security/
|
||||||
|
[2]: https://github.com/turtl/api/issues/20
|
||||||
|
[3]: https://turtl.it/docs/server/
|
||||||
|
[4]: https://trello.com/b/yIQGkHia/turtl-product-dev
|
||||||
|
[5]: https://turtl.it/download/
|
||||||
|
[6]: https://github.com/turtl
|
||||||
|
[7]: https://aur.archlinux.org/packages/turtl/
|
||||||
|
[8]: https://aur.archlinux.org/cgit/aur.git/tree/PKGBUILD?h=turtl
|
||||||
|
[9]: https://turtlapp.com/
|
@ -0,0 +1,554 @@
|
|||||||
|
MySQL 中你应该使用什么数据类型表示时间?
|
||||||
|
==========================================================
|
||||||
|
|
||||||
|
![](http://www.vertabelo.com/_file/blog/what-datatype-should-you-use-to-represent-time-in-mysql-we-compare-datetime-timestamp-and-int/clock.jpg)
|
||||||
|
|
||||||
|
_当你需要保存日期时间数据时,一个问题来了:你应该使用 MySQL 中的什么类型?使用 MySQL 原生的 DATE 类型还是使用 INT 字段把日期和时间保存为一个纯数字呢?_
|
||||||
|
|
||||||
|
在这篇文章中,我将解释 MySQL 原生的方案,并给出一个最常用数据类型的对比表。我们也将对一些典型的查询做基准测试,然后得出在给定场景下应该使用什么数据类型的结论。
|
||||||
|
|
||||||
|
## 原生的 MySQL Datetime 数据类型
|
||||||
|
|
||||||
|
Datetime 数据表示一个时间点。这可以用作日志记录、物联网时间戳、日历事件数据,等等。MySQL 有两种原生的类型可以将这种信息保存在单个字段中:Datetime 和 Timestamp。MySQL 文档中是这么介绍这些数据类型的:
|
||||||
|
|
||||||
|
> DATETIME 类型用于保存同时包含日期和时间两部分的值。MySQL 以 'YYYY-MM-DD HH:MM:SS' 形式接收和显示 DATETIME 类型的值。
|
||||||
|
|
||||||
|
> TIMESTAMP 类型用于保存同时包含日期和时间两部分的值。
|
||||||
|
|
||||||
|
> DATETIME 或 TIMESTAMP 类型的值可以在尾部包含一个毫秒部分,精确度最高到微秒(6 位数)。
|
||||||
|
|
||||||
|
> TIMESTAMP 和 DATETIME 数据类型提供自动初始化和更新到当前的日期和时间的功能,只需在列的定义中设置 DEFAULT CURRENT_TIMESTAMP 和 ON UPDATE CURRENT_TIMESTAMP。
|
||||||
|
|
||||||
|
作为一个例子:
|
||||||
|
|
||||||
|
```SQL
|
||||||
|
CREATE TABLE `datetime_example` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`measured_on` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
KEY `measured_on` (`measured_on`)
|
||||||
|
) ENGINE=InnoDB;
|
||||||
|
```
|
||||||
|
|
||||||
|
```SQL
|
||||||
|
CREATE TABLE `timestamp_example` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`measured_on` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
KEY `measured_on` (`measured_on`)
|
||||||
|
) ENGINE=InnoDB;
|
||||||
|
```
|
||||||
|
|
||||||
|
除了原生的日期时间表示方法,还有另一种常用的存储日期和时间信息的方法。即使用 INT 字段保存 Unix 时间(从1970 年 1 月 1 日协调世界时(UTC)建立所经过的秒数)。
|
||||||
|
|
||||||
|
MySQL 也提供了只保存时间信息中的一部分的方式,通过使用 Date、Year 或 Time 类型。由于这篇文章是关于保存准确时间点的最佳方式的,我们没有讨论这些不那么精确的局部类型。
|
||||||
|
|
||||||
|
## 使用 INT 类型保存 Unix 时间
|
||||||
|
|
||||||
|
使用一个简单的 INT 列保存 Unix 时间是最普通的方法。使用 INT,你可以确保你要保存的数字可以快速、可靠地插入到表中,就像这样:
|
||||||
|
|
||||||
|
```SQL
|
||||||
|
INSERT INTO `vertabelo`.`sampletable`
|
||||||
|
(
|
||||||
|
`id`,
|
||||||
|
`measured_on` ### INT 类型的列
|
||||||
|
)
|
||||||
|
VALUES
|
||||||
|
(
|
||||||
|
1,
|
||||||
|
946684801
|
||||||
|
### 至 01/01/2000 @ 12:00am (UTC) 的 UNIX 时间戳 http://unixtimestamp.com
|
||||||
|
);
|
||||||
|
```
|
||||||
|
|
||||||
|
这就是关于它的所有内容了。它仅仅是个简单的 INT 列,MySQL 的处理方式是这样的:在内部使用 4 个字节保存那些数据。所以如果你在这个列上使用 SELECT 你将会得到一个数字。如果你想把这个列用作日期进行比较,下面的查询并不能正确工作:
|
||||||
|
|
||||||
|
```SQL
|
||||||
|
SELECT
|
||||||
|
id, measured_on, FROM_UNIXTIME(measured_on)
|
||||||
|
FROM
|
||||||
|
vertabelo.inttimestampmeasures
|
||||||
|
WHERE
|
||||||
|
measured_on > '2016-01-01' ### measured_on 会被作为字符串比较以进行查询
|
||||||
|
LIMIT 5;
|
||||||
|
```
|
||||||
|
|
||||||
|
这是因为 MySQL 把 INT 视为数字,而非日期。为了进行日期比较,你必须要么获取( LCTT 译注:从 1970-01-01 00:00:00)到 2016-01-01 经过的秒数,要么使用 MySQL 的 FROM\_UNIXTIME() 函数把 INT 列转为 Date 类型。下面的查询展示了 FROM\_UNIXTIME() 函数的用法:
|
||||||
|
|
||||||
|
```SQL
|
||||||
|
SELECT
|
||||||
|
id, measured_on, FROM_UNIXTIME(measured_on)
|
||||||
|
FROM
|
||||||
|
vertabelo.inttimestampmeasures
|
||||||
|
WHERE
|
||||||
|
FROM_UNIXTIME(measured_on) > '2016-01-01'
|
||||||
|
LIMIT 5;
|
||||||
|
```
|
||||||
|
|
||||||
|
这会正确地获取到日期在 2016-01-01 之后的记录。你也可以直接比较数字和 2016-01-01 的 Unix 时间戳表示形式,即 1451606400。这样做意味着不用使用任何特殊的函数,因为你是在直接比较数字。查询如下:
|
||||||
|
|
||||||
|
```SQL
|
||||||
|
SELECT
|
||||||
|
id, measured_on, FROM_UNIXTIME(measured_on)
|
||||||
|
FROM
|
||||||
|
vertabelo.inttimestampmeasures
|
||||||
|
WHERE
|
||||||
|
measured_on > 1451606400
|
||||||
|
LIMIT 5;
|
||||||
|
```
|
||||||
|
|
||||||
|
假如这种方式不够高效甚至提前做这种转换是不可行的话,那该怎么办?例如,你想获取 2016 年所有星期三的记录。要做到这样而不使用任何 MySQL 日期函数,你就不得不查出 2016 年每个星期三的开始和结束时间的 Unix 时间戳。然后你不得不写很大的查询,至少要在 WHERE 中包含 104 个比较。(2016 年有 52 个星期三,你不得不考虑一天的开始(0:00 am)和结束(11:59:59 pm)...)
|
||||||
|
|
||||||
|
结果是你很可能最终会使用 FROM\_UNIXTIME() 转换函数。既然如此,为什么不试下真正的日期类型呢?
|
||||||
|
|
||||||
|
## 使用 Datetime 和 Timestamp
|
||||||
|
|
||||||
|
Datetime 和 Timestamp 几乎以同样的方式工作。两种都保存日期和时间信息,毫秒部分最高精确度都是 6 位数。同时,使用人类可读的日期形式如 "2016-01-01" (为了便于比较)都能工作。查询时两种类型都支持“宽松格式”。宽松的语法允许任何标点符号作为分隔符。例如,"YYYY-MM-DD HH:MM:SS" 和 "YY-MM-DD HH:MM:SS" 两种形式都可以。在宽松格式情况下以下任何一种形式都能工作:
|
||||||
|
|
||||||
|
```
|
||||||
|
2012-12-31 11:30:45
|
||||||
|
2012^12^31 11+30+45
|
||||||
|
2012/12/31 11*30*45
|
||||||
|
2012@12@31 11^30^45
|
||||||
|
```
|
||||||
|
|
||||||
|
其它宽松格式也是允许的;你可以在 [MySQL 参考手册](https://dev.mysql.com/doc/refman/5.7/en/date-and-time-literals.html) 找到所有的格式。
|
||||||
|
|
||||||
|
默认情况下,Datetime 和 Timestamp 两种类型查询结果都以标准输出格式显示 —— 年-月-日 时:分:秒 (如 2016-01-01 23:59:59)。如果使用了毫秒部分,它们应该以小数值出现在秒后面 (如 2016-01-01 23:59:59.5)。
|
||||||
|
|
||||||
|
Timestamp 和 Datetime 的核心不同点主要在于 MySQL 在内部如何表示这些信息:两种都以二进制而非字符串形式存储,但在表示日期/时间部分时 Timestamp (4 字节) 比 Datetime (5 字节) 少使用 1 字节。当保存毫秒部分时两种都使用额外的空间 (1-3 字节)。如果你存储 150 万条记录,这种 1 字节的差异是微不足道的:
|
||||||
|
|
||||||
|
> 150 万条记录 * 每条记录 1 字节 / (1048576 字节/MB) = __1.43 MB__
|
||||||
|
|
||||||
|
Timestamp 节省的 1 字节是有代价的:你只能存储从 '1970-01-01 00:00:01.000000' 到 '2038-01-19 03:14:07.999999' 之间的时间。而 Datetime 允许你存储从 '1000-01-01 00:00:00.000000' 到 '9999-12-31 23:59:59.999999' 之间的任何时间。
|
||||||
|
|
||||||
|
另一个重要的差别 —— 很多 MySQL 开发者没意识到的 —— 是 MySQL 使用__服务器的时区__转换 Timestamp 值到它的 UTC 等价值再保存。当获取值是它会再次进行时区转换,所以你得回了你“原始的”日期/时间值。有可能,下面这些情况会发生。
|
||||||
|
|
||||||
|
理想情况下,如果你一直使用同一个时区,MySQL 会获取到和你存储的同样的值。以我的经验,如果你的数据库涉及时区变换,你可能会遇到问题。例如,服务器变化(比如,你把数据库从都柏林的一台服务器迁移到加利福尼亚的一台服务器上,或者你只是修改了一下服务器的时区)时可能会发生这种情况。不管哪种方式,如果你获取数据时的时区是不同的,数据就会受影响。
|
||||||
|
|
||||||
|
Datetime 列不会被数据库改变。无论时区怎样配置,每次都会保存和获取到同样的值。就我而言,我认为这是一个更可靠的选择。
|
||||||
|
|
||||||
|
> __MySQL 文档:__
|
||||||
|
|
||||||
|
> MySQL 把 TIMESTAMP 值从当前的时区转换到 UTC 再存储,获取时再从 UTC 转回当前的时区。(其它类型如 DATETIME 不会这样,它们会“原样”保存。) 默认情况下,每个连接的当前时区都是服务器的时区。时区可以基于连接设置。只要时区设置保持一致,你就能得到和保存的相同的值。如果你保存了一个 TIMESTAMP 值,然后改变了时区再获取这个值,获取到的值和你存储的是不同的。这是因为在写入和查询的会话上没有使用同一个时区。当前时区可以通过系统变量 [time_zone](https://dev.mysql.com/doc/refman/5.7/en/server-system-variables.html#sysvar_time_zone) 的值得到。更多信息,请查看 [MySQL Server Time Zone Support](https://dev.mysql.com/doc/refman/5.7/en/time-zone-support.html)。
|
||||||
|
|
||||||
|
## 对比总结
|
||||||
|
|
||||||
|
在深入探讨使用各数据类型的性能差异之前,让我们先看一个总结表格以给你更多了解。每种类型的弱点以红色显示。
|
||||||
|
|
||||||
|
特性 | Datetime | Timestamp | Int (保存 Unix 时间)
|
||||||
|
:--|:--|:--|:--
|
||||||
|
原生时间表示 | 是 | 是 | 否,所以大多数操作需要先使用转换函数,如 FROM_UNIXTIME()
|
||||||
|
能保存毫秒 | 是,最高 6 位精度 | 是,最高 6 位精度 | **否**
|
||||||
|
合法范围 | '1000-01-01 00:00:00.000000' 到 '9999-12-31 23:59:59.999999 | **'1970-01-01 00:00:01.000000' 到 '2038-01-19 03:14:07.999999'** | 若使用 unsigned, '1970-01-01 00:00:01.000000; 理论上最大到 '2106-2-07 06:28:15'
|
||||||
|
自动初始化(MySQL 5.6.5+) | 是 | 是 | **否**
|
||||||
|
宽松解释 ([MySQL docs](https://dev.mysql.com/doc/refman/5.7/en/date-and-time-literals.html)) | 是 | 是 | **否,必须使用正确的格式**
|
||||||
|
值被转换到 UTC 存储 | 否 | **是** | 否
|
||||||
|
可转换到其它类型 | 是,如果值在合法的 Timestamp 范围中 | 是,总是 | 是,如果值在合法的范围中并使用转换函数
|
||||||
|
存储需求([MySQL 5.6.4+](https://dev.mysql.com/doc/refman/5.7/en/storage-requirements.html)) | **5 字节**(如果使用了毫秒部分,再加最多 3 字节) | 4 字节 (如果使用了毫秒部分,再加最多 3 字节) | 4 字节 (不允许毫秒部分)
|
||||||
|
无需使用函数即可作为真实日期可读 | 是 | 是 | **否,你必须格式化输出**
|
||||||
|
数据分区 | 是 | 是,使用 [UNIX_TIMESTAMP()](https://dev.mysql.com/doc/refman/5.7/en/date-and-time-functions.html#function_unix-timestamp);在 MySQL 5.7 中其它表达式是不允许包含 [TIMESTAMP](https://dev.mysql.com/doc/refman/5.7/en/datetime.html) 值的。同时,注意[分区裁剪时的这些考虑](https://dev.mysql.com/doc/refman/5.7/en/partitioning-pruning.html) | 是,使用 INT 上的任何合法操作
|
||||||
|
|
||||||
|
## 基准测试 INT、Timestamp 和 Datetime 的性能
|
||||||
|
|
||||||
|
为了比较这些类型的性能,我会使用我创建的一个天气预报网络的 150 万记录(准确说是 1,497,421)。这个网络每分钟都收集数据。为了让这些测试可复现,我已经删除了一些私有列,所以你可以使用这些数据运行你自己的测试。
|
||||||
|
|
||||||
|
基于我原始的表格,我创建了三个版本:
|
||||||
|
|
||||||
|
- `datetimemeasures` 表在 `measured_on` 列使用 Datetime 类型,表示天气预报记录的测量时间
|
||||||
|
- `timestampmeasures` 表在 `measured_on` 列使用 Timestamp 类型
|
||||||
|
- `inttimestampmeasures` 表在 `measured_on` 列使用 INT (unsigned) 类型
|
||||||
|
|
||||||
|
这三个表拥有完全相同的数据;唯一的差别就是 `measured_on` 字段的类型。所有表都在 `measured_on` 列上设置了一个索引。
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### 基准测试工具
|
||||||
|
|
||||||
|
为了评估这些数据类型的性能,我使用了两种方法。一种基于 [Sysbench](https://github.com/akopytov/sysbench),它的官网是这么描述的:
|
||||||
|
|
||||||
|
_“... 一个模块化、跨平台和多线程的基准测试工具,用以评估那些对运行高负载数据库的系统非常重要的系统参数。”_
|
||||||
|
|
||||||
|
这个工具是 [MySQL 文档](https://dev.mysql.com/downloads/benchmarks.html)中推荐的。
|
||||||
|
|
||||||
|
如果你使用 Windows (就像我),你可以下载一个包含可执行文件和我使用的测试查询的 zip 文件。它们基于 [一种推荐的基准测试方法](https://dba.stackexchange.com/questions/39221/stress-test-mysql-with-queries-captured-with-general-log-in-mysql)。
|
||||||
|
|
||||||
|
为了执行一个给定的测试,你可以使用下面的命令(插入你自己的连接参数):
|
||||||
|
|
||||||
|
```bash
|
||||||
|
sysbench --MySQL-table-engine=innodb --MySQL-db=vertabelo --MySQL-user=root --MySQL-host=localhost --MySQL-password= --test=sysbench_test_file.lua --num-threads=8 --max-requests=100 run
|
||||||
|
```
|
||||||
|
|
||||||
|
这会正常工作,这里 `sysbench_test_file.lua` 是测试文件,并包含了各个测试中指向各个表的 SQL 查询。
|
||||||
|
|
||||||
|
为了进一步验证结果,我也运行了 [mysqlslap](https://dev.mysql.com/doc/refman/5.7/en/mysqlslap.html)。它的官网是这么描述的:
|
||||||
|
|
||||||
|
_“[mysqlslap](https://dev.mysql.com/doc/refman/5.7/en/mysqlslap.html) 是一个诊断程序,为模拟 MySQL 服务器的客户端负载并报告各个阶段的用时而设计。它工作起来就像是很多客户端在同时访问服务器。_
|
||||||
|
|
||||||
|
记得这些测试中最重要的不是所需的_绝对_时间。而是在不同数据类型上执行相同查询时的_相对_时间。这两个基准测试工具的测试时间不一定相同,因为不同工具的工作方式不同。重要的是数据类型的比较,随着我们深入到测试中,这将会变得清楚。
|
||||||
|
|
||||||
|
### 基准测试
|
||||||
|
|
||||||
|
我将使用三种可以评估几个性能方面的查询:
|
||||||
|
|
||||||
|
* 时间范围选择
|
||||||
|
* 在 Datetime 和 Timestamp 数据类型上这允许我们直接比较而不需要使用任何特殊的日期函数。
|
||||||
|
* 同时,我们可以评估在 INT 类型的列上使用日期函数相对于使用简单的数值比较的影响。为了做到这些我们需要把范围转换为 Unix 时间戳数值。
|
||||||
|
* 日期函数选择
|
||||||
|
* 与前个测试中比较操作针对一个简单的 DATE 值相反,这个测试使得我们可以评估使用日期函数作为 “WHERE” 子句的一部分的性能。
|
||||||
|
* 我们还可以测试一个场景,即我们必须使用一个函数将 INT 列转换为一个合法的 DATE 类型然后执行查询。
|
||||||
|
* count() 查询
|
||||||
|
* 作为对前面测试的补充,这将评估在三种不同的表示类型上进行典型的统计查询的性能。
|
||||||
|
|
||||||
|
我们将在这些测试中覆盖一些常见的场景,并看到三种类型上的性能表现。
|
||||||
|
|
||||||
|
### 关于 SQL\_NO\_CACHE
|
||||||
|
|
||||||
|
当在查询中使用 SQL\_NO\_CACHE 时,服务器不使用查询缓存。它既不检查查询缓存以确认结果是不是已经在那儿了,也不会保存查询结果。因此,每个查询将反映真实的性能影响,就像每次查询都是第一次被调用。
|
||||||
|
|
||||||
|
### 测试 1:选择一个日期范围中的值
|
||||||
|
|
||||||
|
这个查询返回总计 1,497,421 行记录中的 75,706 行。
|
||||||
|
|
||||||
|
#### 查询 1 和 Datetime:
|
||||||
|
|
||||||
|
```SQL
|
||||||
|
SELECT SQL_NO_CACHE
|
||||||
|
measured_on
|
||||||
|
FROM
|
||||||
|
vertabelo.datetimemeasures m
|
||||||
|
WHERE
|
||||||
|
m.measured_on > '2016-01-01 00:00:00.0'
|
||||||
|
AND m.measured_on < '2016-02-01 00:00:00.0';
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 性能
|
||||||
|
|
||||||
|
| 响应时间 (ms) | Sysbench | mysqlslap |
|
||||||
|
|-------------|----------|-----------|
|
||||||
|
| 最小 | 152 | 296 |
|
||||||
|
| 最大 | 1261 | 3203 |
|
||||||
|
| 平均 | 362 | 809 |
|
||||||
|
|
||||||
|
```bash
|
||||||
|
Sysbench cmd> sysbench --MySQL-table-engine=innodb --MySQL-db=vertabelo --MySQL-user=root --MySQL-host=localhost --MySQL-password= --test=datetime.lua --num-threads=8 --max-requests=100 run
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash
|
||||||
|
mysqlslap cmd> mysqlslap --query="SELECT SQL_NO_CACHE measured_on FROM vertabelo.datetimemeasures m WHERE m.measured_on > '2016-01-01 00:00:00.0' AND m.measured_on < '2016-02-01 00:00:00.0'" --host=localhost --user=root --concurrency=8 --iterations=100 --no-drop --create-schema=vertabelo
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 查询 1 和 Timestamp:
|
||||||
|
|
||||||
|
```SQL
|
||||||
|
SELECT SQL_NO_CACHE
|
||||||
|
measured_on
|
||||||
|
FROM
|
||||||
|
vertabelo.timestampmeasures m
|
||||||
|
WHERE
|
||||||
|
m.measured_on > '2016-01-01 00:00:00.0'
|
||||||
|
AND m.measured_on < '2016-02-01 00:00:00.0';
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 性能
|
||||||
|
|
||||||
|
| 响应时间 (ms) | Sysbench | mysqlslap |
|
||||||
|
|-------------|----------|-----------|
|
||||||
|
| 最小 | 214 | 359 |
|
||||||
|
| 最大 | 1389 | 3313 |
|
||||||
|
| 平均 | 431 | 1004 |
|
||||||
|
|
||||||
|
```bash
|
||||||
|
Sysbench cmd> sysbench --MySQL-table-engine=innodb --MySQL-db=vertabelo --MySQL-user=root --MySQL-host=localhost --MySQL-password= --test=timestamp.lua --num-threads=8 --max-requests=100 run
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash
|
||||||
|
mysqlslap cmd> mysqlslap --query="SELECT SQL_NO_CACHE measured_on FROM vertabelo.timestampmeasures m WHERE m.measured_on > '2016-01-01 00:00:00.0' AND m.measured_on < '2016-02-01 00:00:00.0'" --host=localhost --user=root --concurrency=8 --iterations=100 --no-drop --create-schema=vertabelo
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 查询 1 和 INT:
|
||||||
|
|
||||||
|
```SQL
|
||||||
|
SELECT SQL_NO_CACHE
|
||||||
|
measured_on
|
||||||
|
FROM
|
||||||
|
vertabelo.inttimestampmeasures m
|
||||||
|
WHERE
|
||||||
|
FROM_UNIXTIME(m.measured_on) > '2016-01-01 00:00:00.0'
|
||||||
|
AND FROM_UNIXTIME(m.measured_on) < '2016-02-01 00:00:00.0';
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 性能
|
||||||
|
|
||||||
|
| 响应时间 (ms) | Sysbench | mysqlslap |
|
||||||
|
|-------------|----------|-----------|
|
||||||
|
| 最小 | 2472 | 7968 |
|
||||||
|
| 最大 | 6554 | 10312 |
|
||||||
|
| 平均 | 4107 | 8527 |
|
||||||
|
|
||||||
|
```bash
|
||||||
|
Sysbench cmd> sysbench --MySQL-table-engine=innodb --MySQL-db=vertabelo --MySQL-user=root --MySQL-host=localhost --MySQL-password= --test=int.lua --num-threads=8 --max-requests=100 run
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash
|
||||||
|
mysqlslap cmd> mysqlslap --query="SELECT SQL_NO_CACHE measured_on FROM vertabelo.inttimestampmeasures m WHERE FROM_UNIXTIME(m.measured_on) > '2016-01-01 00:00:00.0' AND FROM_UNIXTIME(m.measured_on) < '2016-02-01 00:00:00.0'" --host=localhost --user=root --concurrency=8 --iterations=100 --no-drop --create-schema=vertabelo
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 另一种 INT 上的查询 1:
|
||||||
|
|
||||||
|
由于这是个相当直接的范围搜索,而且查询中的日期可以轻易地转为简单的数值比较,我将它包含在了这个测试中。结果证明这是最快的方法 (你大概已经预料到了),因为它仅仅是比较数字而没有使用任何日期转换函数:
|
||||||
|
|
||||||
|
```SQL
|
||||||
|
SELECT SQL_NO_CACHE
|
||||||
|
measured_on
|
||||||
|
FROM
|
||||||
|
vertabelo.inttimestampmeasures m
|
||||||
|
WHERE
|
||||||
|
m.measured_on > 1451617200
|
||||||
|
AND m.measured_on < 1454295600;
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 性能
|
||||||
|
|
||||||
|
| 响应时间 (ms) | Sysbench | mysqlslap |
|
||||||
|
|-------------|----------|-----------|
|
||||||
|
| 最小 | 88 | 171 |
|
||||||
|
| 最大 | 275 | 2157 |
|
||||||
|
| 平均 | 165 | 514 |
|
||||||
|
|
||||||
|
```bash
|
||||||
|
Sysbench cmd> sysbench --MySQL-table-engine=innodb --MySQL-db=vertabelo --MySQL-user=root --MySQL-host=localhost --MySQL-password= --test=basic_int.lua --num-threads=8 --max-requests=100 run
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash
|
||||||
|
mysqlslap cmd> mysqlslap --query="SELECT SQL_NO_CACHE measured_on FROM vertabelo.inttimestampmeasures m WHERE m.measured_on > 1451617200 AND m.measured_on < 1454295600" --host=localhost --user=root --concurrency=8 --iterations=100 --no-drop --create-schema=vertabelo
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 测试 1 总结
|
||||||
|
|
||||||
|
| 平均响应时间 (ms) | Sysbench | 相对于 Datetime 的速度 | mysqlslap | 相对于 Datetime 的速度 |
|
||||||
|
|-----------------|----------|----------------------|-----------|----------------------|
|
||||||
|
| Datetime | 362 | - | 809 | - |
|
||||||
|
| Timestamp | 431 | 慢 19% | 1004 | 慢 24% |
|
||||||
|
| INT | 4107 | 慢 1134% | 8527 | 慢 1054% |
|
||||||
|
| 另一种 INT 查询 | 165 | 快 55% | 514 | 快 36% |
|
||||||
|
|
||||||
|
两种基准测试工具都显示 Datetime 比 Timestamp 和 INT 更快。但 Datetime 没有我们在另一种 INT 查询中使用的简单数值比较快。
|
||||||
|
|
||||||
|
### 测试 2:选择星期一产生的记录
|
||||||
|
|
||||||
|
这个查询返回总计 1,497,421 行记录中的 221,850 行。
|
||||||
|
|
||||||
|
#### 查询 2 和 Datetime:
|
||||||
|
|
||||||
|
```SQL
|
||||||
|
SELECT SQL_NO_CACHE measured_on
|
||||||
|
FROM
|
||||||
|
vertabelo.datetimemeasures m
|
||||||
|
WHERE
|
||||||
|
WEEKDAY(m.measured_on) = 0; # MONDAY
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 性能
|
||||||
|
|
||||||
|
| 响应时间 (ms) | Sysbench | mysqlslap |
|
||||||
|
|-------------|----------|-----------|
|
||||||
|
| 最小 | 1874 | 4343 |
|
||||||
|
| 最大 | 6168 | 7797 |
|
||||||
|
| 平均 | 3127 | 6103 |
|
||||||
|
|
||||||
|
```bash
|
||||||
|
Sysbench cmd> sysbench --MySQL-table-engine=innodb --MySQL-db=vertabelo --MySQL-user=root --MySQL-host=localhost --MySQL-password= --test=datetime_1.lua --num-threads=8 --max-requests=100 run
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash
|
||||||
|
mysqlslap cmd> mysqlslap --query="SELECT SQL_NO_CACHE measured_on FROM vertabelo.datetimemeasures m WHERE WEEKDAY(m.measured_on) = 0" --host=localhost --user=root --concurrency=8 --iterations=25 --no-drop --create-schema=vertabelo
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 查询 2 和 Timestamp:
|
||||||
|
|
||||||
|
```SQL
|
||||||
|
SELECT SQL_NO_CACHE
|
||||||
|
measured_on
|
||||||
|
FROM
|
||||||
|
vertabelo.timestampmeasures m
|
||||||
|
WHERE
|
||||||
|
WEEKDAY(m.measured_on) = 0; # MONDAY
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 性能
|
||||||
|
|
||||||
|
| 响应时间 (ms) | Sysbench | mysqlslap |
|
||||||
|
|-------------|----------|-----------|
|
||||||
|
| 最小 | 2688 | 5953 |
|
||||||
|
| 最大 | 6666 | 13531 |
|
||||||
|
| 平均 | 3653 | 8412 |
|
||||||
|
|
||||||
|
```bash
|
||||||
|
Sysbench cmd> sysbench --MySQL-table-engine=innodb --MySQL-db=vertabelo --MySQL-user=root --MySQL-host=localhost --MySQL-password= --test=timestamp_1.lua --num-threads=8 --max-requests=100 run
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash
|
||||||
|
mysqlslap cmd> mysqlslap --query="SELECT SQL_NO_CACHE measured_on FROM vertabelo.timestampmeasures m WHERE WEEKDAY(m.measured_on) = 0" --host=localhost --user=root --concurrency=8 --iterations=25 --no-drop --create-schema=vertabelo
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 查询 2 和 INT:
|
||||||
|
|
||||||
|
```SQL
|
||||||
|
SELECT SQL_NO_CACHE
|
||||||
|
measured_on
|
||||||
|
FROM
|
||||||
|
vertabelo.inttimestampmeasures m
|
||||||
|
WHERE
|
||||||
|
WEEKDAY(FROM_UNIXTIME(m.measured_on)) = 0; # MONDAY
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 性能
|
||||||
|
|
||||||
|
| 响应时间 (ms) | Sysbench | mysqlslap |
|
||||||
|
|-------------|----------|-----------|
|
||||||
|
| 最小 | 2051 | 5844 |
|
||||||
|
| 最大 | 7007 | 10469 |
|
||||||
|
| 平均 | 3486 | 8088 |
|
||||||
|
|
||||||
|
|
||||||
|
```bash
|
||||||
|
Sysbench cmd> sysbench --MySQL-table-engine=innodb --MySQL-db=vertabelo --MySQL-user=root --MySQL-host=localhost --MySQL-password= --test=int_1.lua --num-threads=8 --max-requests=100 run
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash
|
||||||
|
mysqlslap cmd> mysqlslap --query="SELECT SQL_NO_CACHE measured_on FROM vertabelo.inttimestampmeasures m WHERE WEEKDAY(FROM_UNIXTIME(m.measured_on)) = 0" --host=localhost --user=root --concurrency=8 --iterations=25 --no-drop --create-schema=vertabelo
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 测试 2 总结
|
||||||
|
|
||||||
|
| 平均响应时间 (ms) | Sysbench | 相对于 Datetime 的速度 | mysqlslap | 相对于 Datetime 的速度 |
|
||||||
|
|-----------------|----------|----------------------|-----------|----------------------|
|
||||||
|
| Datetime | 3127 | - | 6103 | - |
|
||||||
|
| Timestamp | 3653 | 慢 17% | 8412 | 慢 38% |
|
||||||
|
| INT | 3486 | 慢 11% | 8088 | 慢 32% |
|
||||||
|
|
||||||
|
再次,在两个基准测试工具中 Datetime 比 Timestamp 和 INT 快。但在这个测试中,INT 查询 —— 即使它使用了一个函数以转换日期 —— 比 Timestamp 查询更快得到结果。
|
||||||
|
|
||||||
|
### 测试 3:选择星期一产生的记录总数
|
||||||
|
|
||||||
|
这个查询返回一行,包含产生于星期一的所有记录的总数(从总共 1,497,421 行可用记录中)。
|
||||||
|
|
||||||
|
#### 查询 3 和 Datetime:
|
||||||
|
|
||||||
|
```SQL
|
||||||
|
SELECT SQL_NO_CACHE
|
||||||
|
COUNT(measured_on)
|
||||||
|
FROM
|
||||||
|
vertabelo.datetimemeasures m
|
||||||
|
WHERE
|
||||||
|
WEEKDAY(m.measured_on) = 0; # MONDAY
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 性能
|
||||||
|
|
||||||
|
| 响应时间 (ms) | Sysbench | mysqlslap |
|
||||||
|
|-------------|----------|-----------|
|
||||||
|
| 最小 | 1720 | 4063 |
|
||||||
|
| 最大 | 4594 | 7812 |
|
||||||
|
| 平均 | 2797 | 5540 |
|
||||||
|
|
||||||
|
```bash
|
||||||
|
Sysbench cmd> sysbench --MySQL-table-engine=innodb --MySQL-db=vertabelo --MySQL-user=root --MySQL-host=localhost --MySQL-password= --test=datetime_1_count.lua --num-threads=8 --max-requests=100 run
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash
|
||||||
|
mysqlslap cmd> mysqlslap --query="SELECT SQL_NO_CACHE COUNT(measured_on) FROM vertabelo.datetimemeasures m WHERE WEEKDAY(m.measured_on) = 0" --host=localhost --user=root --concurrency=8 --iterations=25 --no-drop --create-schema=vertabelo
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 查询 3 和 Timestamp:
|
||||||
|
|
||||||
|
```SQL
|
||||||
|
SELECT SQL_NO_CACHE
|
||||||
|
COUNT(measured_on)
|
||||||
|
FROM
|
||||||
|
vertabelo.timestampmeasures m
|
||||||
|
WHERE
|
||||||
|
WEEKDAY(m.measured_on) = 0; # MONDAY
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 性能
|
||||||
|
|
||||||
|
| 响应时间 (ms) | Sysbench | mysqlslap |
|
||||||
|
|-------------|----------|-----------|
|
||||||
|
| 最小 | 1907 | 4578 |
|
||||||
|
| 最大 | 5437 | 10235 |
|
||||||
|
| 平均 | 3408 | 7102 |
|
||||||
|
|
||||||
|
```bash
|
||||||
|
Sysbench cmd> sysbench --MySQL-table-engine=innodb --MySQL-db=vertabelo --MySQL-user=root --MySQL-host=localhost --MySQL-password= --test=timestamp_1_count.lua --num-threads=8 --max-requests=100 run
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash
|
||||||
|
mysqlslap cmd> mysqlslap --query="SELECT SQL_NO_CACHE COUNT(measured_on) FROM vertabelo.timestampmeasures m WHERE WEEKDAY(m.measured_on) = 0" --host=localhost --user=root --concurrency=8 --iterations=25 --no-drop --create-schema=vertabelo
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 查询 3 和 INT:
|
||||||
|
|
||||||
|
```SQL
|
||||||
|
SELECT SQL_NO_CACHE
|
||||||
|
COUNT(measured_on)
|
||||||
|
FROM
|
||||||
|
vertabelo.inttimestampmeasures m
|
||||||
|
WHERE
|
||||||
|
WEEKDAY(FROM_UNIXTIME(m.measured_on)) = 0; # MONDAY
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 性能
|
||||||
|
|
||||||
|
| 响应时间 (ms) | Sysbench | mysqlslap |
|
||||||
|
|-------------|----------|-----------|
|
||||||
|
| 最小 | 2108 | 5609 |
|
||||||
|
| 最大 | 4764 | 9735 |
|
||||||
|
| 平均 | 3307 | 7416 |
|
||||||
|
|
||||||
|
```bash
|
||||||
|
Sysbench cmd> sysbench --MySQL-table-engine=innodb --MySQL-db=vertabelo --MySQL-user=root --MySQL-host=localhost --MySQL-password= --test=int_1_count.lua --num-threads=8 --max-requests=100 run
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash
|
||||||
|
mysqlslap cmd> mysqlslap --query="SELECT SQL_NO_CACHE COUNT(measured_on) FROM vertabelo.inttimestampmeasures m WHERE WEEKDAY(FROM_UNIXTIME(m.measured_on)) = 0" --host=localhost --user=root --concurrency=8 --iterations=25 --no-drop --create-schema=vertabelo
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 测试 3 总结
|
||||||
|
|
||||||
|
| 平均响应时间 (ms) | Sysbench | 相对于 Datetime 的速度 | mysqlslap | 相对于 Datetime 的速度 |
|
||||||
|
|-----------------|----------|----------------------|-----------|----------------------|
|
||||||
|
| Datetime | 2797 | - | 5540 | - |
|
||||||
|
| Timestamp | 3408 | 慢 22% | 7102 | 慢 28% |
|
||||||
|
| INT | 3307 | 慢 18% | 7416 | 慢 33% |
|
||||||
|
|
||||||
|
再一次,两个基准测试工具都显示 Datetime 比 Timestamp 和 INT 快。不能判断 INT 是否比 Timestamp 快,因为 mysqlslap 显示 INT 比 Timestamp 略快而 Sysbench 却相反。
|
||||||
|
|
||||||
|
_注意:_ 所有测试都是在一台 Windows 10 机器上本地运行的,这台机器拥有一个双核 i7 CPU,16GB 内存,运行 MariaDB v10.1.9,使用 innoDB 引擎。
|
||||||
|
|
||||||
|
## 结论
|
||||||
|
|
||||||
|
基于这些数据,我确信 Datetime 是大多数场景下的最佳选择。原因是:
|
||||||
|
|
||||||
|
* 更快(根据我们的三个基准测试)。
|
||||||
|
* 无需任何转换即是人类可读的。
|
||||||
|
* 不会因为时区变换产生问题。
|
||||||
|
* 只比它的对手们多用 1 字节
|
||||||
|
* 支持更大的日期范围(从 1000 年到 9999 年)
|
||||||
|
|
||||||
|
如果你只是存储 Unix 时间戳(并且在它的合法日期范围内),而且你真的不打算在它上面使用任何基于日期的查询,我觉得使用 INT 是可以的。我们已经看到,它执行简单数值比较查询时非常快,因为只是在处理简单的数字。
|
||||||
|
|
||||||
|
Timestamp 怎么样呢?如果 Datetime 相对于 Timestamp 的优势不适用于你特殊的场景,你最好使用时间戳。阅读这篇文章后,你对三种类型间的区别应该有了更好的理解,可以根据你的需要做出最佳的选择。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.vertabelo.com/blog/technical-articles/what-datatype-should-you-use-to-represent-time-in-mysql-we-compare-datetime-timestamp-and-int
|
||||||
|
|
||||||
|
作者:[Francisco Claria][a]
|
||||||
|
译者:[bianjp](https://github.com/bianjp)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.axones.com.ar/
|
@ -0,0 +1,49 @@
|
|||||||
|
使用 Github Pages 发布你的项目文档
|
||||||
|
=====
|
||||||
|
|
||||||
|
你可能比较熟悉[如何用 Github Pages 来分享你的工作][3],又或许你看过[一堂][4]教你建立你的第一个 Github Pages 网站的教程。近期 Github Pages 的改进使得[从不同的数据源来发布您的网站][5]更加的方便,其中的来源之一就是你的仓库的 /docs 目录。
|
||||||
|
|
||||||
|
文档的质量是一个软件项目健康发展的标志。对于开源项目来说,维护一个可靠而不出错的知识库、详细说明所有的细节是至关重要的。精心策划的文档可以让增加项目的亲切感,提供一步步的指导并促进各种方式的合作可以推动开源软件开发的协作进程。
|
||||||
|
|
||||||
|
在 Web 上托管你的文档是一个消耗时间的挑战,而且对于它的发布和维护也没有省事的办法,然而这是并非不可避免的。面对多种不同的发布工具,又是 FTP 服务器,又是数据库,文件以各种不同的方式存放在不同的位置下,而这些都需要你手动来调整。需要说明的是,传统的 Web 发布方式提供了无与伦比的灵活性和性能,但是在许多情况下,这是以牺牲简单易用为代价的。
|
||||||
|
|
||||||
|
当作为文档使用时,麻烦更少的方式显然更容易去维护。
|
||||||
|
|
||||||
|
[GitHub Pages][2] 可以以指定的方式为你的项目创建网站,这使得它天然地适合发布和维护文档。因为 Github Pages 支持 Jekyll,所以你可以使用纯文本或 Markdown 来书写你的文档,从而降低你维护的成本、减少维护时的障碍。Jekyll 还支持许多有用的工具比如变量、模板、以及自动代码高亮等等,它会给你更多的灵活性而不会增加复杂性,这些你在一些笨重的平台是见不到的。
|
||||||
|
|
||||||
|
最重要的是,在 Github 上使用 GitHub Pages 意味着你的文档和代码可以使用诸如 Issues 和 Pull Requests 来确保它得到应有的高水平维护,而且因为 GitHub Pages 允许您发布代码库主分支上的 /docs 目录,这样您就可以在同一分支同时维护你的代码库及其文档。
|
||||||
|
|
||||||
|
### 现在开始!
|
||||||
|
|
||||||
|
发布你的第一个文档页面只需要短短几分钟。
|
||||||
|
|
||||||
|
1. 在你的仓库的主分支里创建一个 /docs/index.md 文件。
|
||||||
|
|
||||||
|
2. 把你的内容以 Jekyll 格式添加进去,并提交你的修改。
|
||||||
|
|
||||||
|
![](https://cloud.githubusercontent.com/assets/3477155/17778793/47c5a586-6533-11e6-982c-ebd41ec6968c.gif)
|
||||||
|
|
||||||
|
3. 查看你的仓库的设置分支然后选择主分支 /docs 目录,将其设置为 GitHub Pages 的源 ,点击保存,你就搞定了。
|
||||||
|
|
||||||
|
![](https://cloud.githubusercontent.com/assets/3477155/17778792/47c2ecc4-6533-11e6-828a-91980daa7297.gif)
|
||||||
|
|
||||||
|
GitHub Pages 将会从你的 /docs 目录中读取内容,转换 index.md 为 HTML。然后把它发布到你的 GitHub Pages 的 URL 上。
|
||||||
|
|
||||||
|
这样将会创建并输出一个最基础的 HTML ,而且你可以使用 Jekyll 的自定义模板、CSS 和其他特性。如果想要看所有的可能,你可以看看 [GitHub Pages Showcase][1]。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://github.com/blog/2233-publish-your-project-documentation-with-github-pages
|
||||||
|
|
||||||
|
作者:[loranallensmith][a]
|
||||||
|
译者:[Bestony](https://github.com/bestony)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://github.com/loranallensmith
|
||||||
|
[1]: https://github.com/showcases/github-pages-examples
|
||||||
|
[2]: https://pages.github.com/
|
||||||
|
[3]: https://www.youtube.com/watch?v=2MsN8gpT6jY
|
||||||
|
[4]: https://www.youtube.com/watch?v=RaKX4A5EiQo
|
||||||
|
[5]: https://help.github.com/articles/configuring-a-publishing-source-for-github-pages/
|
@ -0,0 +1,48 @@
|
|||||||
|
百度运用 FPGA 方法大规模加速 SQL 查询
|
||||||
|
===================================================================
|
||||||
|
|
||||||
|
尽管我们对百度今年工作焦点的关注集中在这个中国搜索巨头在深度学习方面的举措上,许多其他的关键的,尽管不那么前沿的应用表现出了大数据带来的挑战。
|
||||||
|
|
||||||
|
正如百度的欧阳剑在本周 Hot Chips 大会上谈论的,百度坐拥超过 1 EB 的数据,每天处理大约 100 PB 的数据,每天更新 100 亿的网页,每 24 小时更新处理超过 1 PB 的日志更新,这些数字和 Google 不分上下,正如人们所想象的。百度采用了类似 Google 的方法去大规模地解决潜在的瓶颈。
|
||||||
|
|
||||||
|
正如刚刚我们谈到的,Google 寻找一切可能的方法去打败摩尔定律,百度也在进行相同的探索,而令人激动的、使人着迷的机器学习工作是迷人的,业务的核心关键任务的加速同样也是,因为必须如此。欧阳提到,公司基于自身的数据提供高端服务的需求和 CPU 可以承载的能力之间的差距将会逐渐增大。
|
||||||
|
|
||||||
|
![](http://www.nextplatform.com/wp-content/uploads/2016/08/BaiduFPGA1.png)
|
||||||
|
|
||||||
|
对于百度的百亿亿级问题,在所有数据的接受端是一系列用于数据分析的框架和平台,从该公司的海量知识图谱,多媒体工具,自然语言处理框架,推荐引擎,和点击流分析都是这样。简而言之,大数据的首要问题就是这样的:一系列各种应用和与之匹配的具有压倒性规模的数据。
|
||||||
|
|
||||||
|
当谈到加速百度的大数据分析,所面临的几个挑战,欧阳谈到抽象化运算核心去寻找一个普适的方法是困难的。“大数据应用的多样性和变化的计算类型使得这成为一个挑战,把所有这些整合成为一个分布式系统是困难的,因为有多变的平台和编程模型(MapReduce,Spark,streaming,user defined,等等)。将来还会有更多的数据类型和存储格式。”
|
||||||
|
|
||||||
|
尽管存在这些障碍,欧阳讲到他们团队找到了(它们之间的)共同线索。如他所指出的那样,那些把他们的许多数据密集型的任务相连系在一起的就是传统的 SQL。“我们的数据分析任务大约有 40% 是用 SQL 写的,而其他的用 SQL 重写也是可用做到的。” 更进一步,他讲道他们可以享受到现有的 SQL 系统的好处,并可以和已有的框架相匹配,比如 Hive,Spark SQL,和 Impala 。下一步要做的事情就是 SQL 查询加速,百度发现 FPGA 是最好的硬件。
|
||||||
|
|
||||||
|
![](http://www.nextplatform.com/wp-content/uploads/2016/08/BaiduFPGA2.png)
|
||||||
|
|
||||||
|
这些主板,被称为处理单元( 下图中的 PE ),当执行 SQL 时会自动地处理关键的 SQL 功能。这里所说的都是来自演讲,我们不承担责任。确切的说,这里提到的 FPGA 有点神秘,或许是故意如此。如果百度在基准测试中得到了如下图中的提升,那这可是一个有竞争力的信息。后面我们还会继续介绍这里所描述的东西。简单来说,FPGA 运行在数据库中,当其收到 SQL 查询的时候,该团队设计的软件就会与之紧密结合起来。
|
||||||
|
|
||||||
|
![](http://www.nextplatform.com/wp-content/uploads/2016/08/BaiduFPGA3.png)
|
||||||
|
|
||||||
|
欧阳提到了一件事,他们的加速器受限于 FPGA 的带宽,不然性能表现本可以更高,在下面的评价中,百度安装了 2 块12 核心,主频 2.0 GHz 的 intl E26230 CPU,运行在 128G 内存。SDA 具有 5 个处理单元,(上图中的 300MHz FPGA 主板)每个分别处理不同的核心功能(筛选(filter),排序(sort),聚合(aggregate),联合(join)和分组(group by))
|
||||||
|
|
||||||
|
为了实现 SQL 查询加速,百度针对 TPC-DS 的基准测试进行了研究,并且创建了称做处理单元(PE)的特殊引擎,用于在基准测试中加速 5 个关键功能,这包括筛选(filter),排序(sort),聚合(aggregate),联合(join)和分组(group by),(我们并没有把这些单词都像 SQL 那样大写)。SDA 设备使用卸载模型,具有多个不同种类的处理单元的加速卡在 FPGA 中组成逻辑,SQL 功能的类型和每张卡的数量由特定的工作量决定。由于这些查询在百度的系统中执行,用来查询的数据被以列格式推送到加速卡中(这会使得查询非常快速),而且通过一个统一的 SDA API 和驱动程序,SQL 查询工作被分发到正确的处理单元而且 SQL 操作实现了加速。
|
||||||
|
|
||||||
|
SDA 架构采用一种数据流模型,加速单元不支持的操作被退回到数据库系统然后在那里本地运行,比其他任何因素,百度开发的 SQL 加速卡的性能被 FPGA 卡的内存带宽所限制。加速卡跨整个集群机器工作,顺便提一下,但是数据和 SQL 操作如何分发到多个机器的准确原理没有被百度披露。
|
||||||
|
|
||||||
|
![](http://www.nextplatform.com/wp-content/uploads/2016/08/BaiduFPGA4.png)
|
||||||
|
|
||||||
|
![](http://www.nextplatform.com/wp-content/uploads/2016/08/BaiduFPGA5.png)
|
||||||
|
|
||||||
|
我们受限与百度所愿意披露的细节,但是这些基准测试结果是十分令人鼓舞的,尤其是 Terasort 方面,我们将在 Hot Chips 大会之后跟随百度的脚步去看看我们是否能得到关于这是如何连接到一起的和如何解决内存带宽瓶颈的细节。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.nextplatform.com/2016/08/24/baidu-takes-fpga-approach-accelerating-big-sql/
|
||||||
|
|
||||||
|
作者:[Nicole Hemsoth][a]
|
||||||
|
译者:[LinuxBars](https://github.com/LinuxBars)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.nextplatform.com/author/nicole/
|
||||||
|
[1]: http://www.nextplatform.com/?s=baidu+deep+learning
|
||||||
|
[2]: http://www.hotchips.org/wp-content/uploads/hc_archives/hc26/HC26-12-day2-epub/HC26.12-5-FPGAs-epub/HC26.12.545-Soft-Def-Acc-Ouyang-baidu-v3--baidu-v4.pdf
|
@ -0,0 +1,53 @@
|
|||||||
|
Azure SQL 数据库已经支持 JSON
|
||||||
|
===========
|
||||||
|
|
||||||
|
我们很高兴地宣布你现在可以在 Azure SQL 中查询及存储关系型数据或者 JSON 了、Azure SQL 数据库提供了读取 JSON 文本数据的简单的内置函数,将 JSON 文本转化成表,以及将表的数据转化成 JSON。
|
||||||
|
|
||||||
|
![](https://azurecomcdn.azureedge.net/mediahandler/acomblog/media/Default/blog/1cc536a5-a822-467b-a4a2-4557746f70cc.png)
|
||||||
|
|
||||||
|
你可以使用 JSON 函数来从 JSON 文本中提取值([JSON_VALUE][4])、提取对象([JSON_QUERY][5]), 更新JSON 中的值([JSON_MODIFY][6]),并且验证 JSON 文本的正确性([ISJSON][7])。[OPENJSON][8] 函数让你可以将 JSON 文本转化成表结构。最后,JSON 功能函数可以让你很简单地从 SQL 查询中使用 [FOR JSON][9] 从句来获得 JSON 文本结果。
|
||||||
|
|
||||||
|
### 你可以用 JSON 做什么?
|
||||||
|
|
||||||
|
Azure SQL 数据库中的 JSON 可以让您构建并与现代 web、移动设备和 HTML5/单页应用、诸如 Azure DocumentDB 等包含 JSON 格式化数据的 NoSQL 存储等交换数据,分析来自不同系统和服务的日志和消息。现在你可以轻易地将 Azure SQL 数据库与任何使用使用 JSON 的服务集成。
|
||||||
|
|
||||||
|
#### 轻易地开放数据给现代框架和服务
|
||||||
|
|
||||||
|
你有没有在使用诸如 REST 或者 Azure App 使用 JSON 来交换数据的服务?你有使用诸如 AngularJS、ReactJS、D3 或者 JQuery 等使用 JSON 的组件或框架么?使用新的 JSON 功能函数,你可以轻易地格式化存储在 Azure SQL 数据库中的数据,并将它用在任何现代服务或者应用中。
|
||||||
|
|
||||||
|
#### 轻松采集 JSON 数据
|
||||||
|
|
||||||
|
你有在使用移动设备、传感器、如 Azure Stream Analytics 或者 Insight 这样产生 JSON 的服务、如 Azure DocumentDB 或者 MongoDB 这样存储 JSON 的系统么?你需要在 Azure SQL 数据中使用熟悉的 SQL 语句来查询并分析 JSON 数据么?现在你可以轻松采集 JSON 数据并存储到 Azure SQL 数据库中,并且可以使用任何 Azure SQL 数据库支持的语言或者工具来查询和分析加载的数据。
|
||||||
|
|
||||||
|
#### 简化你的数据模型
|
||||||
|
|
||||||
|
你需要同时存储及查询数据库中关系型及半结构化的数据么?你需简化像 NoSQL 平台下的数据模型么?现在你可以在一张表中同时存储结构化数据及非结构化数据了。在 Azure SQL 数据库中,你可以同时从关系型及 NoSQL 的世界中使用最好的方法来调整你的数据模型。Azure SQL 数据库让你可以使用 Transact-SQL 语言来查询关系及 JSON 数据。程序和工具将不会在从表中取出的值及 JSON 文本中提取的值看出差别。
|
||||||
|
|
||||||
|
### 下一步
|
||||||
|
|
||||||
|
要学习如何在你的应用中集成 JSON,查看我们的[开始学习][1]页面或者 [Channel 9的视频][2]。要了解不同的情景下如何集成 JSON,观看 Channel 9 的视频或者在这些 [JSON 分类文章][3]中查找你感兴趣的使用情景。
|
||||||
|
|
||||||
|
我们将持续增加新的 JSON 特性并让 JSON 支持的更好。请敬请关注。
|
||||||
|
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://azure.microsoft.com/en-us/blog/json-support-is-generally-available-in-azure-sql-database/
|
||||||
|
|
||||||
|
作者:[Jovan Popovic][a]
|
||||||
|
译者:[geekpi](https://github.com/geekpi)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://azure.microsoft.com/en-us/blog/author/jovanpop/
|
||||||
|
[1]: https://azure.microsoft.com/en-us/documentation/articles/sql-database-json-features/
|
||||||
|
[2]: https://channel9.msdn.com/Shows/Data-Exposed/SQL-Server-2016-and-JSON-Support
|
||||||
|
[3]: http://blogs.msdn.com/b/sqlserverstorageengine/archive/tags/json/
|
||||||
|
[4]: https://msdn.microsoft.com/en-us/library/dn921898.aspx
|
||||||
|
[5]: https://msdn.microsoft.com/en-us/library/dn921884.aspx
|
||||||
|
[6]: https://msdn.microsoft.com/en-us/library/dn921892.aspx
|
||||||
|
[7]: https://msdn.microsoft.com/en-us/library/dn921896.aspx
|
||||||
|
[8]: https://msdn.microsoft.com/en-us/library/dn921885.aspx
|
||||||
|
[9]: https://msdn.microsoft.com/en-us/library/dn921882.aspx
|
||||||
|
|
@ -0,0 +1,217 @@
|
|||||||
|
理解 Linux 下 Shell 命令的不同分类及它们的用法
|
||||||
|
====================
|
||||||
|
|
||||||
|
当你打算真正操纵好你的 Linux 系统,没有什么能比命令行界面更让你做到这一点。为了成为一个 Linux 高手,你必须能够理解 [Shell 命令的不同类型][1],并且会在终端下正确的使用它们。
|
||||||
|
|
||||||
|
在 Linux 下,命令有几种类型,对于一个 Linux 新手来说,知道不同命令的意思才能够高效和准确的使用它们。因此,在这篇文章里,我们将会遍及各种不同分类的 Linux Shell 命令。
|
||||||
|
|
||||||
|
需要注意一件非常重要的事:命令行界面和 Shell 是不同的,命令行界面只是为你提供一个访问 Shell 的方式。而 Shell ,它是可编程的,这使得它可以通过命令与内核进行交流。
|
||||||
|
|
||||||
|
下面列出了 Linux 下命令的不同种类:
|
||||||
|
|
||||||
|
### 1. 程序可执行文件(文件系统 中的命令)
|
||||||
|
|
||||||
|
当你执行一条命令的时候,Linux 通过从左到右搜索存储在 `$PATH` 环境变量中的目录来找到这条命令的可执行文件。
|
||||||
|
|
||||||
|
你可以像下面这样查看存储在 `$PATH` 中的目录:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ echo $PATH
|
||||||
|
/home/aaronkilik/bin:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games
|
||||||
|
```
|
||||||
|
|
||||||
|
在上面的命令中,目录 `/home/aaronkilik/bin` 将会被首先搜索,紧跟着是 `/usr/local/sbin`,然后一直接着下去。在搜索过程中,搜索顺序是至关重要的。
|
||||||
|
|
||||||
|
比如在 `/usr/bin` 目录里的文件系统中的命令:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ ll /bin/
|
||||||
|
```
|
||||||
|
|
||||||
|
样本输出:
|
||||||
|
|
||||||
|
```
|
||||||
|
total 16284
|
||||||
|
drwxr-xr-x 2 root root 4096 Jul 31 16:30 ./
|
||||||
|
drwxr-xr-x 23 root root 4096 Jul 31 16:29 ../
|
||||||
|
-rwxr-xr-x 1 root root 6456 Apr 14 18:53 archdetect*
|
||||||
|
-rwxr-xr-x 1 root root 1037440 May 17 16:15 bash*
|
||||||
|
-rwxr-xr-x 1 root root 520992 Jan 20 2016 btrfs*
|
||||||
|
-rwxr-xr-x 1 root root 249464 Jan 20 2016 btrfs-calc-size*
|
||||||
|
lrwxrwxrwx 1 root root 5 Jul 31 16:19 btrfsck -> btrfs*
|
||||||
|
-rwxr-xr-x 1 root root 278376 Jan 20 2016 btrfs-convert*
|
||||||
|
-rwxr-xr-x 1 root root 249464 Jan 20 2016 btrfs-debug-tree*
|
||||||
|
-rwxr-xr-x 1 root root 245368 Jan 20 2016 btrfs-find-root*
|
||||||
|
-rwxr-xr-x 1 root root 270136 Jan 20 2016 btrfs-image*
|
||||||
|
-rwxr-xr-x 1 root root 249464 Jan 20 2016 btrfs-map-logical*
|
||||||
|
-rwxr-xr-x 1 root root 245368 Jan 20 2016 btrfs-select-super*
|
||||||
|
-rwxr-xr-x 1 root root 253816 Jan 20 2016 btrfs-show-super*
|
||||||
|
-rwxr-xr-x 1 root root 249464 Jan 20 2016 btrfstune*
|
||||||
|
-rwxr-xr-x 1 root root 245368 Jan 20 2016 btrfs-zero-log*
|
||||||
|
-rwxr-xr-x 1 root root 31288 May 20 2015 bunzip2*
|
||||||
|
-rwxr-xr-x 1 root root 1964536 Aug 19 2015 busybox*
|
||||||
|
-rwxr-xr-x 1 root root 31288 May 20 2015 bzcat*
|
||||||
|
lrwxrwxrwx 1 root root 6 Jul 31 16:19 bzcmp -> bzdiff*
|
||||||
|
-rwxr-xr-x 1 root root 2140 May 20 2015 bzdiff*
|
||||||
|
lrwxrwxrwx 1 root root 6 Jul 31 16:19 bzegrep -> bzgrep*
|
||||||
|
-rwxr-xr-x 1 root root 4877 May 20 2015 bzexe*
|
||||||
|
lrwxrwxrwx 1 root root 6 Jul 31 16:19 bzfgrep -> bzgrep*
|
||||||
|
-rwxr-xr-x 1 root root 3642 May 20 2015 bzgrep*
|
||||||
|
```
|
||||||
|
|
||||||
|
### 2. Linux 别名
|
||||||
|
|
||||||
|
这些是用户定义的命令,它们是通过 shell 内置命令 `alias` 创建的,其中包含其它一些带有选项和参数的 shell 命令。这个意图主要是使用新颖、简短的名字来替代冗长的命令。
|
||||||
|
|
||||||
|
创建一个别名的语法像下面这样:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ alias newcommand='command -options'
|
||||||
|
```
|
||||||
|
|
||||||
|
通过下面的命令,可以列举系统中的所有别名:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ alias -p
|
||||||
|
alias alert='notify-send --urgency=low -i "$([ $? = 0 ] && echo terminal || echo error)" "$(history|tail -n1|sed -e '\''s/^\s*[0-9]\+\s*//;s/[;&|]\s*alert$//'\'')"'
|
||||||
|
alias egrep='egrep --color=auto'
|
||||||
|
alias fgrep='fgrep --color=auto'
|
||||||
|
alias grep='grep --color=auto'
|
||||||
|
alias l='ls -CF'
|
||||||
|
alias la='ls -A'
|
||||||
|
alias ll='ls -alF'
|
||||||
|
alias ls='ls --color=auto'
|
||||||
|
```
|
||||||
|
|
||||||
|
要在 Linux 中创建一个新的别名,仔细阅读下面的例子。
|
||||||
|
|
||||||
|
```
|
||||||
|
$ alias update='sudo apt update'
|
||||||
|
$ alias upgrade='sudo apt dist-upgrade'
|
||||||
|
$ alias -p | grep 'up'
|
||||||
|
```
|
||||||
|
|
||||||
|
![](http://www.tecmint.com/wp-content/uploads/2016/08/Create-Aliase-in-Linux.png)
|
||||||
|
|
||||||
|
然而,上面这些我们创建的别名只能暂时的工作,当经过下一次系统启动后它们不再工作。你可以像下面展示的这样在 '.bashrc' 文件中设置永久别名。
|
||||||
|
|
||||||
|
![](http://www.tecmint.com/wp-content/uploads/2016/08/Set-Linux-Aliases-Permanent.png)
|
||||||
|
|
||||||
|
添加以后,运行下面的命令来激活:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ source ~/.bashrc
|
||||||
|
```
|
||||||
|
|
||||||
|
### 3. Linux Shell 保留字
|
||||||
|
|
||||||
|
在 shell 程序设计中,`if`、`then`、`fi`、`for`、`while`、`case`、`esac`、`else`、`until` 以及其他更多的字都是 shell 保留字。正如描述所暗示的,它们在 shell 中有特殊的含义。
|
||||||
|
|
||||||
|
你可以通过使用下面展示的 `type` 命令来列出所有的 shell 关键字:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ type if then fi for while case esac else until
|
||||||
|
if is a shell keyword
|
||||||
|
then is a shell keyword
|
||||||
|
fi is a shell keyword
|
||||||
|
for is a shell keyword
|
||||||
|
while is a shell keyword
|
||||||
|
case is a shell keyword
|
||||||
|
esac is a shell keyword
|
||||||
|
else is a shell keyword
|
||||||
|
until is a shell keyword
|
||||||
|
```
|
||||||
|
|
||||||
|
### 4. Linux shell 函数
|
||||||
|
|
||||||
|
一个 shell 函数是一组在当前 shell 内一起执行的命令。函数有利于在 shell 脚本中实现特殊任务。在 shell 脚本中写 shell 函数的传统形式是下面这样:
|
||||||
|
|
||||||
|
```
|
||||||
|
function_name() {
|
||||||
|
command1
|
||||||
|
command2
|
||||||
|
......
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
或者像这样:
|
||||||
|
|
||||||
|
```
|
||||||
|
function function_name {
|
||||||
|
command1
|
||||||
|
command2
|
||||||
|
......
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
让我们看一看如何在一个名为 shell_functions.sh 的脚本中写 shell 函数。
|
||||||
|
|
||||||
|
```
|
||||||
|
#!/bin/bash
|
||||||
|
#write a shell function to update and upgrade installed packages
|
||||||
|
upgrade_system(){
|
||||||
|
sudo apt update;
|
||||||
|
sudo apt dist-upgrade;
|
||||||
|
}
|
||||||
|
#execute function
|
||||||
|
upgrade_system
|
||||||
|
```
|
||||||
|
|
||||||
|
取代通过命令行执行两条命令:`sudo apt update` 和 `sudo apt dist-upgrade`,我们在脚本内写了一个像执行一条单一命令一样来执行两条命令的 shell 函数 upgrade_system。
|
||||||
|
|
||||||
|
保存文件,然后使脚本可执行。最后像下面这样运行 shell 函数:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ chmod +x shell_functions.sh
|
||||||
|
$ ./shell_functions.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
![](http://www.tecmint.com/wp-content/uploads/2016/08/Linux-Shell-Functions-Script.png)
|
||||||
|
|
||||||
|
### 5. Linux Shell 内置命令
|
||||||
|
|
||||||
|
这些是在 shell 中内置的 Linux 命令,所以你无法在文件系统中找到它们。这些命令包括 `pwd`、`cd`、`bg`、`alias`、`history`、`type`、`source`、`read`、`exit` 等。
|
||||||
|
|
||||||
|
你可以通过下面展示的 `type` 命令来列出或检查 Linux 内置命令:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ type pwd
|
||||||
|
pwd is a shell builtin
|
||||||
|
$ type cd
|
||||||
|
cd is a shell builtin
|
||||||
|
$ type bg
|
||||||
|
bg is a shell builtin
|
||||||
|
$ type alias
|
||||||
|
alias is a shell builtin
|
||||||
|
$ type history
|
||||||
|
history is a shell builtin
|
||||||
|
```
|
||||||
|
|
||||||
|
学习一些 Linux 内置命令用法:
|
||||||
|
|
||||||
|
- [Linux 下 15 个 pwd 命令例子][2]
|
||||||
|
- [Linux 下 15 个 cd 命令例子][3]
|
||||||
|
- [了解 Linux 下 history 命令的威力][4]
|
||||||
|
|
||||||
|
### 结论
|
||||||
|
|
||||||
|
作为一个 Linux 用户,知道你所运行的命令类型是很重要的。我相信,通过上面明确、简单并且易于理解的解释,包括一些相关的说明,你可能对 “[Linux 命令的不同种类][1]”有了很好的理解。
|
||||||
|
|
||||||
|
你也可以在下面的评论区提任何问题或补充意见,从而和我们取得联系。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.tecmint.com/understanding-different-linux-shell-commands-usage/
|
||||||
|
|
||||||
|
作者:[Aaron Kili][a]
|
||||||
|
译者:[ucasFL](https://github.com/ucasFL)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.tecmint.com/author/aaronkili/
|
||||||
|
[1]: http://www.tecmint.com/different-types-of-linux-shells/
|
||||||
|
[2]: http://www.tecmint.com/pwd-command-examples/
|
||||||
|
[3]: http://www.tecmint.com/cd-command-in-linux/
|
||||||
|
[4]: http://www.tecmint.com/history-command-examples/
|
||||||
|
[5]: http://www.tecmint.com/category/linux-commands/
|
76
published/201609/20160828 4 Best Linux Boot Loaders.md
Normal file
76
published/201609/20160828 4 Best Linux Boot Loaders.md
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
4 个最好的 Linux 引导程序
|
||||||
|
==================
|
||||||
|
|
||||||
|
当你打开你的机器,开机自检(POST)成功完成后,BIOS(基本输入输出系统)立即定位所配置的引导介质,并从 MBR(主引导记录)或 GUID(全局唯一标识符)分区表读取一些命令,这是引导介质的最前面 512 个字节内容。主引导记录(MBR)中包含两个重要的信息集合,第一个是引导程序,第二个是分区表。
|
||||||
|
|
||||||
|
### 什么是引导程序?
|
||||||
|
|
||||||
|
引导程序(Boot Loader)是存储在 MBR(主引导记录)或 GUID(全局唯一标识符)分区表中的一个小程序,用于帮助把操作系统装载到内存中。如果没有引导程序,那么你的操作系统将不能够装载到内存中。
|
||||||
|
|
||||||
|
有一些我们可以随同 Linux 安装到系统上的引导程序,在这篇文章里,我将简要地谈论几个最好的可以与 Linux 一同工作的 Linux 引导程序。
|
||||||
|
|
||||||
|
### 1. GNU GRUB
|
||||||
|
|
||||||
|
GNU GRUB 是一个非常受欢迎,也可能是用的最多的具有多重引导能力的 Linux 引导程序,它以原始的 Eirch Stefan Broleyn 发明的 GRUB(GRand Unified Bootlader)为基础。GNU GRUB 增强了原来的 GRUB,带来了一些改进、新的特性和漏洞修复。
|
||||||
|
|
||||||
|
重要的是,GRUB 2 现在已经取代了 GRUB。值得注意的是,GRUB 这个名字被重新命名为 GRUB Legacy,但没有活跃开发,不过,它可以用来引导老的系统,因为漏洞修复依然继续。
|
||||||
|
|
||||||
|
GRUB 具有下面一些显著的特性:
|
||||||
|
|
||||||
|
- 支持多重引导
|
||||||
|
- 支持多种硬件结构和操作系统,比如 Linux 和 Windows
|
||||||
|
- 提供一个类似 Bash 的交互式命令行界面,从而用户可以运行 GRUB 命令来和配置文件进行交互
|
||||||
|
- 允许访问 GRUB 编辑器
|
||||||
|
- 支持设置加密口令以确保安全
|
||||||
|
- 支持从网络进行引导,以及一些次要的特性
|
||||||
|
|
||||||
|
访问主页: <https://www.gnu.org/software/grub/>
|
||||||
|
|
||||||
|
### 2. LILO(Linux 引导程序(LInux LOader))
|
||||||
|
|
||||||
|
LILO 是一个简单但强大且非常稳定的 Linux 引导程序。由于 GRUB 有很大改善和增加了许多强大的特性,越来越受欢迎,因此 LILO 在 Linux 用户中已经不是很流行了。
|
||||||
|
|
||||||
|
当 LILO 引导的时候,单词“LILO”会出现在屏幕上,并且每一个字母会在一个特定的事件发生前后出现。然而,从 2015 年 12 月开始,LILO 的开发停止了,它有许多特性比如下面列举的:
|
||||||
|
|
||||||
|
- 不提供交互式命令行界面
|
||||||
|
- 支持一些错误代码
|
||||||
|
- 不支持网络引导(LCTT 译注:其变体 ELILO 支持 TFTP/DHCP 引导)
|
||||||
|
- 所有的文件存储在驱动的最开始 1024 个柱面上
|
||||||
|
- 面临 BTFS、GTP、RAID 等的限制
|
||||||
|
|
||||||
|
访问主页: <http://lilo.alioth.debian.org/>
|
||||||
|
|
||||||
|
### 3. BURG - 新的引导程序
|
||||||
|
|
||||||
|
基于 GRUB,BURG 是一个相对来说比较新的引导程序(LCTT 译注:已于 2011 年停止了开发)。由于 BURG 起源于 GRUB, 所以它带有一些 GRUB 主要特性。尽管如此, BURG 也提供了一些出色的特性,比如一种新的对象格式可以支持包括 Linux、Windows、Mac OS、 FreeBSD 等多种平台。
|
||||||
|
|
||||||
|
另外,BURG 支持可高度配置的文本和图标模式的引导菜单,计划增加的“流”支持未来可以不同的输入/输出设备一同工作。
|
||||||
|
|
||||||
|
访问主页: <https://launchpad.net/burg>
|
||||||
|
|
||||||
|
### 4. Syslinux
|
||||||
|
|
||||||
|
Syslinux 是一种能从光盘驱动器、网络等进行引导的轻型引导程序。Syslinux 支持诸如 MS-DOS 上的 FAT、 Linux 上的 ext2、ext3、ext4 等文件系统。Syslinux 也支持未压缩的单一设备上的 Btrfs。
|
||||||
|
|
||||||
|
注意由于 Syslinux 仅能访问自己分区上的文件,因此不具备多重文件系统引导能力。
|
||||||
|
|
||||||
|
访问主页: <http://www.syslinux.org/wiki/index.php?title=The_Syslinux_Project>
|
||||||
|
|
||||||
|
### 结论
|
||||||
|
|
||||||
|
一个引导程序允许你在你的机器上管理多个操作系统,并在某个的时间选择其中一个使用。没有引导程序,你的机器就不能够装载内核以及操作系统的剩余部分。
|
||||||
|
|
||||||
|
我们是否遗漏了任何一流的 Linux 引导程序?如果有请让我们知道,请在下面的评论表中填入值得推荐的 Linux 系统引导程序。
|
||||||
|
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.tecmint.com/best-linux-boot-loaders/
|
||||||
|
|
||||||
|
作者:[Aaron Kili][a]
|
||||||
|
译者:[ucasFL](https://github.com/ucasFL)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.tecmint.com/best-linux-boot-loaders/
|
@ -0,0 +1,265 @@
|
|||||||
|
Ohm:用两百行 JavaScript 创造你自己的编程语言
|
||||||
|
==========
|
||||||
|
|
||||||
|
![](https://www.pubnub.com/wp-content/uploads/2016/08/ohm-banner.jpg)
|
||||||
|
|
||||||
|
解析器是一种超级有用的软件库。从概念上简单的说,它们的实现很有挑战性,并且在计算机科学中经常被认为是黑魔法。在这个系列的博文中,我会向你们展示为什么你不需要成为哈利波特就能够精通解析器这种魔法。但是为了以防万一带上你的魔杖吧!
|
||||||
|
|
||||||
|
我们将探索一种叫做 Ohm 的新的开源库,它使得搭建解析器很简单并且易于重用。在这个系列里,我们使用 Ohm 去识别数字,构建一个计算器等等。在这个系列的最后你将已经用不到 200 行的代码发明了一种完整的编程语言。这个强大的工具将让你能够做到一些你可能过去认为不可能的事情。
|
||||||
|
|
||||||
|
###为什么解析器很困难?
|
||||||
|
|
||||||
|
解析器非常有用。在很多时候你可能需要一个解析器。或许有一种你需要处理的新的文件格式,但还没有人为它写了一个库;又或许你发现了一种古老格式的文件,但是已有的解析器不能在你的平台上构建。我已经看到这样的事发生无数次。 Code 在或者不在, Data 就在那里,不增不减。
|
||||||
|
|
||||||
|
从根本上来说,解析器很简单:只是把一个数据结构转化成另一个。所以你会不会觉得你要是邓布利多校长就好了?
|
||||||
|
|
||||||
|
解析器历来是出奇地难写,所面临的挑战是绝大多数现有的工具都很老,并且需要一定的晦涩难懂的计算机科学知识。如果你在大学里上过编译器课程,那么课本里也许还有从上世纪七十年传下来的技术。幸运的是,解析器技术从那时候起已经提高了很多。
|
||||||
|
|
||||||
|
典型的,解析器是通过使用一种叫作[形式语法(formal grammar)][1]的特殊语法来定义你想要解析的东西来创造的,然后你需要把它放入像 [Bison][2] 和 [Yacc][3] 的工具中,这些工具能够产生一堆 C 代码,这些代码你需要修改或者链接到你实际写入的编程语言中。另外的选择是用你更喜欢的语言亲自动手写一个解析器,这很慢且很容易出错,在你能够真正使用它之前还有许多额外的工作。
|
||||||
|
|
||||||
|
想像一下,是否你关于你想要解析的东西的语法描述也是解析器?如果你能够只是直接运行这些语法,然后仅在你需要的地方增加一些挂钩(hook)呢?那就是 Ohm 所可以做到的事。
|
||||||
|
|
||||||
|
### Ohm 简介
|
||||||
|
|
||||||
|
[Ohm][4]是一种新的解析系统。它类似于你可能已经在课本里面看到过的语法,但是它更强大,使用起来更简单。通过 Ohm, 你能够使用一种灵活的语法在一个 .ohm 文件中来写你自己的格式定义,然后使用你的宿主语言把语义加入到里面。在这篇博文里,我们将用 JavaScript 作为宿主语言。
|
||||||
|
|
||||||
|
Ohm 建立于一个为创造更简单、更灵活的解析器的多年研究基础之上。VPRI 的 [STEPS program (pdf)][5] 使用 Ohm 的前身 [Ometa][6] 为许多特殊的任务创造了专门的语言(比如一个有 400 行代码的平行制图描绘器)。
|
||||||
|
|
||||||
|
Ohm 有许多有趣的特点和符号,但是相比于全部解释它们,我认为我们只需要深入其中并构建一些东西就行了。
|
||||||
|
|
||||||
|
###解析整数
|
||||||
|
|
||||||
|
让我们来解析一些数字。这看起来会很简单,只需在一个文本串中寻找毗邻的数字,但是让我们尝试去处理所有形式的数字:整数和浮点数、十六进制数和八进制数、科学计数、负数。解析数字很简单,正确解析却很难。
|
||||||
|
|
||||||
|
亲自构建这个代码将会很困难,会有很多问题,会伴随有许多特殊的情况,比如有时会相互矛盾。正则表达式或许可以做的这一点,但是它会非常丑陋而难以维护。让我们用 Ohm 来试试。
|
||||||
|
|
||||||
|
用 Ohm 构建的解析器涉及三个部分:语法(grammar)、语义(semantics)和测试(tests)。我通常挑选问题的一部分为它写测试,然后构建足够的语法和语义来使测试通过。然后我再挑选问题的另一部分,增加更多的测试、更新语法和语义,从而确保所有的测试能够继续通过。即使我们有了新的强大的工具,写解析器从概念上来说依旧很复杂。测试是用一种合理的方式来构建解析器的唯一方法。现在,让我们开始工作。
|
||||||
|
|
||||||
|
我们将从整数开始。一个整数由一系列相互毗邻的数字组成。让我们把下面的内容放入一个叫做 grammar.ohm 的文件中:
|
||||||
|
|
||||||
|
```
|
||||||
|
CoolNums {
|
||||||
|
// just a basic integer
|
||||||
|
Number = digit+
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
这创造了一条匹配一个或多个数字(`digit`)叫作 `Number` 的单一规则。`+` 意味着一个或更多,就在正则表达式中一样。当有一个或更多的数字时,这个规则将会匹配它们,如果没有数字或者有一些不是数字的东西将不会匹配。“数字(`digit`)”的定义是从 0 到 9 其中的一个字符。`digit` 也是像 `Number` 一样的规则,但是它是 Ohm 的其中一条构建规则因此我们不需要去定义它。如果我们想的话可以推翻它,但在这时候这没有任何意义,毕竟我们不打算去发明一种新的数。
|
||||||
|
|
||||||
|
现在,我们可以读入这个语法并用 Ohm 库来运行它。
|
||||||
|
|
||||||
|
把它放入 test1.js:
|
||||||
|
|
||||||
|
```
|
||||||
|
var ohm = require('ohm-js');
|
||||||
|
var fs = require('fs');
|
||||||
|
var assert = require('assert');
|
||||||
|
var grammar = ohm.grammar(fs.readFileSync('src/blog_numbers/syntax1.ohm').toString());
|
||||||
|
```
|
||||||
|
|
||||||
|
`Ohm.grammar` 调用将读入该文件并解析成一个语法对象。现在我们可以增加一些语义。把下面内容增加到你的 JavaScript 文件中:
|
||||||
|
|
||||||
|
```
|
||||||
|
var sem = grammar.createSemantics().addOperation('toJS', {
|
||||||
|
Number: function(a) {
|
||||||
|
return parseInt(this.sourceString,10);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
这通过 `toJS` 操作创造了一个叫作 `sem` 的语法集。这些语义本质上是一些对应到语法中每个规则的函数。每个函数当与之相匹配的语法规则被解析时将会被调用。上面的 `Number` 函数将会在语法中的 `Number` 规则被解析时被调用。语法(grammar)定义了在语言中这些代码是什么,语义(semantics)定义了当这些代码被解析时应该做什么。
|
||||||
|
|
||||||
|
语义函数能够做我们想做的任何事,比如打印出故障信息、创建对象,或者在任何子节点上递归调用 `toJS`。此时我们仅仅想把匹配的文本转换成真正的 JavaScript 整数。
|
||||||
|
|
||||||
|
所有的语义函数有一个内含的 `this` 对象,带有一些有用的属性。其 `source` 属性代表了输入文本中和这个节点相匹配的部分。`this.sourceString` 是一个匹配输入的串,调用内置在 JavaScript 中的 `parseInt` 函数会把这个串转换成一个数。传给 `parseInt` 的 `10` 这个参数告诉 JavaScript 我们输入的是一个以 `10` 为基底(10 进制)的数。如果少了这个参数, JavaScript 也会假定以 10 为基底,但是我们把它包含在里面因为后面我们将支持以 16 为基底的数,所以使之明确比较好。
|
||||||
|
|
||||||
|
既然我们有一些语法,让我们来实际解析一些东西看一看我们的解析器是否能够工作。如何知道我们的解析器可以工作?通过测试,许多许多的测试,每一个可能的边缘情况都需要一个测试。
|
||||||
|
|
||||||
|
使用标准的断言 `assert` API,以下这个测试函数能够匹配一些输入并运用我们的语义把它转换成一个数,然后把这个数和我们期望的输入进行比较。
|
||||||
|
|
||||||
|
```
|
||||||
|
function test(input, answer) {
|
||||||
|
var match = grammar.match(input);
|
||||||
|
if(match.failed()) return console.log("input failed to match " + input + match.message);
|
||||||
|
var result = sem(match).toJS();
|
||||||
|
assert.deepEqual(result,answer);
|
||||||
|
console.log('success = ', result, answer);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
就是如此。现在我们能够为各种不同的数写一堆测试。如果匹配失败我们的脚本将会抛出一个例外。否则就打印成功信息。让我们尝试一下,把下面这些内容加入到脚本中:
|
||||||
|
|
||||||
|
```
|
||||||
|
test("123",123);
|
||||||
|
test("999",999);
|
||||||
|
test("abc",999);
|
||||||
|
```
|
||||||
|
|
||||||
|
然后用 `node test1.js` 运行脚本。
|
||||||
|
|
||||||
|
你的输出应该是这样:
|
||||||
|
|
||||||
|
```
|
||||||
|
success = 123 123
|
||||||
|
success = 999 999
|
||||||
|
input failed to match abcLine 1, col 1:
|
||||||
|
> 1 | abc
|
||||||
|
^
|
||||||
|
Expected a digit
|
||||||
|
```
|
||||||
|
|
||||||
|
真酷。正如预期的那样,前两个成功了,第三个失败了。更好的是,Ohm 自动给了我们一个很棒的错误信息指出匹配失败。
|
||||||
|
|
||||||
|
###浮点数
|
||||||
|
|
||||||
|
我们的解析器工作了,但是它做的工作不是很有趣。让我们把它扩展成既能解析整数又能解析浮点数。改变 grammar.ohm 文件使它看起来像下面这样:
|
||||||
|
|
||||||
|
```
|
||||||
|
CoolNums {
|
||||||
|
// just a basic integer
|
||||||
|
Number = float | int
|
||||||
|
int = digit+
|
||||||
|
float = digit+ "." digit+
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
这把 `Number` 规则改变成指向一个浮点数(`float`)或者一个整数(`int`)。这个 `|` 代表着“或”。我们把这个读成“一个 `Number` 由一个浮点数或者一个整数构成。”然后整数(`int`)定义成 `digit+`,浮点数(`float`)定义成 `digit+` 后面跟着一个句号然后再跟着另一个 `digit+`。这意味着在句号前和句号后都至少要有一个数字。如果一个数中没有一个句号那么它就不是一个浮点数,因此就是一个整数。
|
||||||
|
|
||||||
|
现在,让我们再次看一下我们的语义功能。由于我们现在有了新的规则所以我们需要新的功能函数:一个作为整数的,一个作为浮点数的。
|
||||||
|
|
||||||
|
```
|
||||||
|
var sem = grammar.createSemantics().addOperation('toJS', {
|
||||||
|
Number: function(a) {
|
||||||
|
return a.toJS();
|
||||||
|
},
|
||||||
|
int: function(a) {
|
||||||
|
console.log("doing int", this.sourceString);
|
||||||
|
return parseInt(this.sourceString,10);
|
||||||
|
},
|
||||||
|
float: function(a,b,c) {
|
||||||
|
console.log("doing float", this.sourceString);
|
||||||
|
return parseFloat(this.sourceString);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
这里有两件事情需要注意。首先,整数(`int`)、浮点数(`float`)和数(`Number`)都有相匹配的语法规则和函数。然而,针对 `Number` 的功能不再有任何意义。它接收子节点 `a` 然后返回该子节点的 `toJS` 结果。换句话说,`Number` 规则简单的返回相匹配的子规则。由于这是在 Ohm 中任何规则的默认行为,因此实际上我们不用去考虑 `Number` 的作用,Ohm 会替我们做好这件事。
|
||||||
|
|
||||||
|
其次,整数(`int`)有一个参数 `a`,然而浮点数有三个:`a`、`b` 和 `c`。这是由于规则的实参数量(arity)决定的。[实参数量(arity)][7] 意味着一个规则里面有多少参数。如果我们回过头去看语法,浮点数(`float`)的规则是:
|
||||||
|
|
||||||
|
```
|
||||||
|
float = digit+ "." digit+
|
||||||
|
```
|
||||||
|
|
||||||
|
浮点数规则通过三个部分来定义:第一个 `digit+`、`.`、以及第二个 `digit+`。这三个部分都会作为参数传递给浮点数的功能函数。因此浮点数必须有三个参数,否则 Ohm 库会给出一个错误。在这种情况下我们不用在意参数,因为我们仅仅直接攫取了输入串,但是我们仍然需要参数列在那里来避免编译器错误。后面我们将实际使用其中一些参数。
|
||||||
|
|
||||||
|
现在我们可以为新的浮点数支持添加更多的测试。
|
||||||
|
|
||||||
|
```
|
||||||
|
test("123",123);
|
||||||
|
test("999",999);
|
||||||
|
//test("abc",999);
|
||||||
|
test('123.456',123.456);
|
||||||
|
test('0.123',0.123);
|
||||||
|
test('.123',0.123);
|
||||||
|
```
|
||||||
|
|
||||||
|
注意最后一个测试将会失败。一个浮点数必须以一个数开始,即使它就是个 0,`.123` 不是有效的,实际上真正的 JavaScript 语言也有相同的规则。
|
||||||
|
|
||||||
|
###十六进制数
|
||||||
|
|
||||||
|
现在我们已经有了整数和浮点数,但是还有一些其它的数的语法最好可以支持:十六进制数和科学计数。十六进制数是以 16 为基底的整数。十六进制数的数字能从 0 到 9 和从 A 到 F。十六进制数经常用在计算机科学中,当用二进制数据工作时,你可以仅仅使用两个数字表示 0 到 255 的数。
|
||||||
|
|
||||||
|
在绝大多数源自 C 的编程语言(包括 JavaScript),十六进制数通过在前面加上 `0x` 来向编译器表明后面跟的是一个十六进制数。为了让我们的解析器支持十六进制数,我们只需要添加另一条规则。
|
||||||
|
|
||||||
|
```
|
||||||
|
Number = hex | float | int
|
||||||
|
int = digit+
|
||||||
|
float = digit+ "." digit+
|
||||||
|
hex = "0x" hexDigit+
|
||||||
|
hexDigit := "0".."9" | "a".."f" | "A".."F"
|
||||||
|
```
|
||||||
|
|
||||||
|
我实际上已经增加了两条规则。十六进制数(`hex`)表明它是一个 `0x` 后面一个或多个十六进制数字(`hexDigits`)的串。一个十六进制数字(`hexDigit`)是从 0 到 9,或从 a 到 f,或 A 到 F(包括大写和小写的情况)的一个字符。我也修改了 `Number` 规则来识别十六进制数作为另外一种可能的情况。现在我们只需要另一条针对十六进制数的功能规则。
|
||||||
|
|
||||||
|
```
|
||||||
|
hex: function(a,b) {
|
||||||
|
return parseInt(this.sourceString,16);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
注意到,在这种情况下,我们把 `16` 作为基底传递给 `parseInt`,因为我们希望 JavaScript 知道这是一个十六进制数。
|
||||||
|
|
||||||
|
我略过了一些很重要需要注意的事。`hexDigit` 的规则像下面这样:
|
||||||
|
|
||||||
|
```
|
||||||
|
hexDigit := "0".."9" | "a".."f" | "A".."F"
|
||||||
|
```
|
||||||
|
|
||||||
|
注意我使用的是 `:=` 而不是 `=`。在 Ohm 中,`:=` 是当你需要推翻一条规则的时候使用。这表明 Ohm 已经有了一条针对 `hexDigit` 的默认规则,就像 `digit`、`space` 等一堆其他的东西。如果我使用了 `=`, Ohm 将会报告一个错误。这是一个检查,从而避免我无意识的推翻一个规则。由于新的 `hexDigit` 规则和 Ohm 的构建规则一样,所以我们可以把它注释掉,然后让 Ohm 自己来实现它。我留下这个规则只是因为这样我们可以看到它实际上是如何进行的。
|
||||||
|
|
||||||
|
现在,我们可以添加更多的测试然后看到十六进制数真的能工作:
|
||||||
|
|
||||||
|
```
|
||||||
|
test('0x456',0x456);
|
||||||
|
test('0xFF',255);
|
||||||
|
```
|
||||||
|
|
||||||
|
###科学计数
|
||||||
|
|
||||||
|
最后,让我们来支持科学计数。科学计数是针对非常大或非常小的数的,比如 `1.8×10^3`。在大多数编程语言中,科学计数法表示的数会写成这样:1.8e3 表示 18000,或者 1.8e-3 表示 .018。让我们增加另外一对规则来支持这个指数表示:
|
||||||
|
|
||||||
|
```
|
||||||
|
float = digit+ "." digit+ exp?
|
||||||
|
exp = "e" "-"? digit+
|
||||||
|
```
|
||||||
|
|
||||||
|
上面在浮点数规则末尾增加了一个指数(`exp`)规则和一个 `?`。`?` 表示没有或有一个,所以指数(`exp`)是可选的,但是不能超过一个。增加指数(`exp`)规则也改变了浮点数规则的实参数量,所以我们需要为浮点数功能增加另一个参数,即使我们不使用它。
|
||||||
|
|
||||||
|
```
|
||||||
|
float: function(a,b,c,d) {
|
||||||
|
console.log("doing float", this.sourceString);
|
||||||
|
return parseFloat(this.sourceString);
|
||||||
|
},
|
||||||
|
```
|
||||||
|
|
||||||
|
现在我们的测试可以通过了:
|
||||||
|
|
||||||
|
```
|
||||||
|
test('4.8e10',4.8e10);
|
||||||
|
test('4.8e-10',4.8e-10);
|
||||||
|
```
|
||||||
|
|
||||||
|
###结论
|
||||||
|
|
||||||
|
Ohm 是构建解析器的一个很棒的工具,因为它易于上手,并且你可以递增的增加规则。Ohm 也还有其他我今天没有写到的很棒的特点,比如调试观察仪和子类化。
|
||||||
|
|
||||||
|
到目前为止,我们已经使用 Ohm 来把字符串翻译成 JavaScript 数,并且 Ohm 经常用于把一种表示方式转化成另外一种。然而,Ohm 还有更多的用途。通过放入不同的语义功能集,你可以使用 Ohm 来真正处理和计算东西。一个单独的语法可以被许多不同的语义使用,这是 Ohm 的魔法之一。
|
||||||
|
|
||||||
|
在这个系列的下一篇文章中,我将向你们展示如何像真正的计算机一样计算像 `(4.85 + 5 * (238 - 68)/2)` 这样的数学表达式,不仅仅是解析数。
|
||||||
|
|
||||||
|
额外的挑战:你能够扩展语法来支持八进制数吗?这些以 8 为基底的数能够只用 0 到 7 这几个数字来表示,前面加上一个数字 0 或者字母 `o`。看看针对下面这些测试情况是够正确。下次我将给出答案。
|
||||||
|
|
||||||
|
```
|
||||||
|
test('0o77',7*8+7);
|
||||||
|
test('0o23',0o23);
|
||||||
|
```
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://www.pubnub.com/blog/2016-08-30-javascript-parser-ohm-makes-creating-a-programming-language-easy/
|
||||||
|
|
||||||
|
作者:[Josh Marinacci][a]
|
||||||
|
译者:[ucasFL](https://github.com/ucasFL)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://www.pubnub.com/blog/author/josh/
|
||||||
|
[1]: https://en.wikipedia.org/wiki/Formal_grammar
|
||||||
|
[2]: https://en.wikipedia.org/wiki/GNU_bison
|
||||||
|
[3]: https://en.wikipedia.org/wiki/Yacc
|
||||||
|
[4]: https://github.com/cdglabs/ohm
|
||||||
|
[5]: http://www.vpri.org/pdf/tr2012001_steps.pdf
|
||||||
|
[6]: http://tinlizzie.org/ometa/
|
||||||
|
[7]: https://en.wikipedia.org/wiki/Arity
|
80
published/201609/20160830 Spark comparison - AWS vs. GCP.md
Normal file
80
published/201609/20160830 Spark comparison - AWS vs. GCP.md
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
AWS 和 GCP 的 Spark 技术哪家强?
|
||||||
|
==============
|
||||||
|
|
||||||
|
> Tianhui Michael Li 和Ariel M’ndange-Pfupfu将在今年10月10、12和14号组织一个在线经验分享课程:[Spark 分布式计算入门][4]。该课程的内容包括创建端到端的运行应用程序和精通 Spark 关键工具。
|
||||||
|
|
||||||
|
毋庸置疑,云计算将会在未来数据科学领域扮演至关重要的角色。弹性,可扩展性和按需分配的计算能力作为云计算的重要资源,直接导致云服务提供商集体火拼。其中最大的两股势力正是[亚马逊网络服务(AWS)][1] 和[谷歌云平台(GCP)][2]。
|
||||||
|
|
||||||
|
本文依据构建时间和运营成本对 AWS 和 GCP 的 Spark 工作负载作一个简短比较。实验由我们的学生在数据孵化器(The Data Incubator)进行,[数据孵化器(The Data Incubator)][3]是一个大数据培训组织,专门为公司招聘顶尖数据科学家并为公司职员培训最新的大数据科学技能。尽管 Spark 效率惊人,分布式工作负载的时间和成本亦然可以大到不能忽略不计。因此我们一直努力寻求更高效的技术,以便我们的学生能够学习到最好和最快的工具。
|
||||||
|
|
||||||
|
### 提交 Spark 任务到云
|
||||||
|
|
||||||
|
[Spark][5] 是一个类 MapReduce 但是比 MapReduce 更灵活、更抽象的并行计算框架。Spark 提供 Python 和 Java 编程接口,但它更愿意用户使用原生的 Scala 语言进行应用程序开发。Scala 可以把应用程序和依赖文件打包在一个 JAR 文件从而使 Spark 任务提交变得简单。
|
||||||
|
|
||||||
|
通常情况下,Sprark 结合 HDFS 应用于分布式数据存储,而与 YARN 协同工作则应用于集群管理;这种堪称完美的配合使得 Spark 非常适用于 AWS 的弹性 MapReduce (EMR)集群和 GCP 的 Dataproc 集群。这两种集群都已有 HDFS 和 YARN 预配置,不需要额外进行配置。
|
||||||
|
|
||||||
|
### 配置云服务
|
||||||
|
|
||||||
|
通过命令行比通过网页界面管理数据、集群和任务具有更高的可扩展性。对 AWS 而言,这意味着客户需要安装 [CLI][6]。客户必须获得证书并为每个 EC2 实例创建[独立的密钥对][7]。除此之外,客户还需要为 EMR 用户和 EMR 本身创建角色(基本权限),主要是准入许可规则,从而使 EMR 用户获得足够多的权限(通常在 CLI 运行 `aws emr create-default-roles` 就可以)。
|
||||||
|
|
||||||
|
相比而言,GCP 的处理流程更加直接。如果客户选择安装 [Google Cloud SDK][8] 并且使用其 Google 账号登录,那么客户即刻可以使用 GCP 的几乎所有功能而无需其他任何配置。唯一需要提醒的是不要忘记通过 API 管理器启用计算引擎、Dataproc 和云存储 JSON 的 API。
|
||||||
|
|
||||||
|
当你安装你的喜好设置好之后,有趣的事情就发生了!比如可以通过`aws s3 cp`或者`gsutil cp`命令拷贝客户的数据到云端。再比如客户可以创建自己的输入、输出或者任何其他需要的 bucket,如此,运行一个应用就像创建一个集群或者提交 JAR 文件一样简单。请确定日志存放的地方,毕竟在云环境下跟踪问题或者调试 bug 有点诡异。
|
||||||
|
|
||||||
|
### 一分钱一分货
|
||||||
|
|
||||||
|
谈及成本,Google 的服务在以下几个方面更有优势。首先,购买计算能力的原始成本更低。4 个 vCPU 和 15G RAM 的 Google 计算引擎服务(GCE)每小时只需 0.20 美元,如果运行 Dataproc,每小时也只需区区 0.24 美元。相比之下,同等的云配置,AWS EMR 则需要每小时 0.336 美元。
|
||||||
|
|
||||||
|
其次,计费方式。AWS 按小时计费,即使客户只使用 15 分钟也要付足 1 小时的费用。GCP 按分钟计费,最低计费 10 分钟。在诸多用户案例中,资费方式的不同直接导致成本的巨大差异。
|
||||||
|
|
||||||
|
两种云服务都有其他多种定价机制。客户可以使用 AWS 的 Sport Instance 或 GCP 的 Preemptible Instance 来竞价它们的空闲云计算能力。这些服务比专有的、按需服务便宜,缺点是不能保证随时有可用的云资源提供服务。在 GCP 上,如果客户长时间(每月的 25% 至 100%)使用服务,可以获取更多折扣。在 AWS 上预付费或者一次购买大批量服务可以节省不少费用。底线是,如果你是一个超级用户,并且使用云计算已经成为一种常态,那么最好深入研究云计算,自己算计好成本。
|
||||||
|
|
||||||
|
最后,新手在 GCP 上体验云服务的费用较低。新手只需 300 美元信用担保,就可以免费试用 60 天 GCP 提供的全部云服务。AWS 只免费提供特定服务的特定试用层级,如果运行 Spark 任务,需要付费。这意味着初次体验 Spark,GCP 具有更多选择,也少了精打细算和讨价还价的烦恼。
|
||||||
|
|
||||||
|
### 性能比拼
|
||||||
|
|
||||||
|
我们通过实验检测一个典型 Spark 工作负载的性能与开销。实验分别选择 AWS 的 m3.xlarg 和 GCP 的 n1-standard-4,它们都是由一个 Master 和 5 个核心实例组成的集群。除了规格略有差别,虚拟核心和费用都相同。实际上它们在 Spark 任务的执行时间上也表现的惊人相似。
|
||||||
|
|
||||||
|
测试 Spark 任务包括对数据的解析、过滤、合并和聚合,这些数据来自公开的[堆栈交换数据转储(Stack Exchange Data Dump)][9]。通过运行相同的 JAR,我们首先对大约 50M 的数据子集进行[交叉验证][10],然后将验证扩大到大约 9.5G 的数据集。
|
||||||
|
|
||||||
|
![](https://d3ansictanv2wj.cloudfront.net/1400_img_1_AWS_GCP-25ed6069029112a8439d89999796be18.jpg)
|
||||||
|
>Figure 1. Credit: Michael Li and Ariel M'ndange-Pfupfu.
|
||||||
|
|
||||||
|
![](https://d3ansictanv2wj.cloudfront.net/1400_img_2_AWS_GCP-448714718896b21e32f8b47d4657fc8c.jpg)
|
||||||
|
>Figure 2. Credit: Michael Li and Ariel M'ndange-Pfupfu.
|
||||||
|
|
||||||
|
结果表明,短任务在 GCP 上具有明显的成本优势,这是因为 GCP 以分钟计费,并最终扣除了 10 分钟的费用,而 AWS 则收取了 1 小时的费用。但是即使长任务,因为计费方式占优,GPS 仍然具有相当优势。同样值得注意的是存储成本并不包括在此次比较当中。
|
||||||
|
|
||||||
|
###结论
|
||||||
|
|
||||||
|
AWS 是云计算的先驱,这甚至体现在 API 中。AWS 拥有巨大的生态系统,但其许可模型已略显陈旧,配置管理也有些晦涩难解。相比之下,Google 是云计算领域的新星并且将云计算服务打造得更加圆润自如。但是 GCP 缺少一些便捷的功能,比如通过简单方法自动结束集群和详细的任务计费信息分解。另外,其 Python 编程接口也不像 [AWS 的 Boto][11] 那么全面。
|
||||||
|
|
||||||
|
如果你初次使用云计算,GCP 因其简单易用,别具魅力。即使你已在使用 AWS,你也许会发现迁移到 GCP 可能更划算,尽管真正从 AWS 迁移到 GCP 的代价可能得不偿失。
|
||||||
|
|
||||||
|
当然,现在对两种云服务作一个全面的总结还非常困难,因为它们都不是单一的实体,而是由多个实体整合而成的完整生态系统,并且各有利弊。真正的赢家是用户。一个例证就是在数据孵化器(The Data Incubator),我们的博士数据科学研究员在学习分布式负载的过程中真正体会到成本的下降。虽然我们的[大数据企业培训客户][12]可能对价格不那么敏感,他们更在意能够更快速地处理企业数据,同时保持价格不增加。数据科学家现在可以享受大量的可选服务,这些都是从竞争激烈的云计算市场得到的实惠。
|
||||||
|
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://www.oreilly.com/ideas/spark-comparison-aws-vs-gcp
|
||||||
|
|
||||||
|
作者:[Michael Li][a],[Ariel M'Ndange-Pfupfu][b]
|
||||||
|
译者:[firstadream](https://github.com/firstadream)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://www.oreilly.com/people/76a5b-michael-li
|
||||||
|
[b]: https://www.oreilly.com/people/Ariel-Mndange-Pfupfu
|
||||||
|
[1]: https://aws.amazon.com/
|
||||||
|
[2]: https://cloud.google.com/
|
||||||
|
[3]: https://www.thedataincubator.com/training.html?utm_source=OReilly&utm_medium=blog&utm_campaign=AWSvsGCP
|
||||||
|
[4]: http://www.oreilly.com/live-training/distributed-computing-with-spark.html?intcmp=il-data-olreg-lp-oltrain_20160828_new_site_spark_comparison_aws_gcp_post_top_note_training_link
|
||||||
|
[5]: http://spark.apache.org/
|
||||||
|
[6]: http://docs.aws.amazon.com/cli/latest/userguide/cli-chap-getting-set-up.html
|
||||||
|
[7]: http://docs.aws.amazon.com/ElasticMapReduce/latest/DeveloperGuide/EMR_SetUp_KeyPair.html
|
||||||
|
[8]: https://cloud.google.com/sdk/#Quick_Start
|
||||||
|
[9]: https://archive.org/details/stackexchange
|
||||||
|
[10]: http://stats.stackexchange.com/
|
||||||
|
[11]: https://github.com/boto/boto3
|
||||||
|
[12]: https://www.thedataincubator.com/training.html?utm_source=OReilly&utm_medium=blog&utm_campaign=AWSvsGCP
|
@ -0,0 +1,99 @@
|
|||||||
|
JavaScript 现状:方言篇
|
||||||
|
===========
|
||||||
|
|
||||||
|
JavaScript 和其他编程语言有一个很大的不同,它不像单纯的一个语言,而像一个由众多方言组成大家族。
|
||||||
|
|
||||||
|
从 2009 年 CoffeeScript 出现开始,近几年出现了大量基于 JavaScript 语言,或者叫方言,例如 ES6、TypeScript、Elm 等等。它们都有自己的优势,且都可以被完美编译成标准 JavaScript。
|
||||||
|
|
||||||
|
所以,继上周的前端框架篇,今天带来 JavaScript 现状之方言篇,看一下大家对于 JavaScript 的方言是怎么选择的。
|
||||||
|
|
||||||
|
> 声明:下面的部分结论来自部分数据,这是在我想要展示完整数据时找到的最好的办法,这便于我分享我的一些想法。
|
||||||
|
|
||||||
|
> 注意:如果你还没有参与[这个调查][3],现在就来参加吧,可以花十分钟完成调查然后再回来看这篇文章。
|
||||||
|
|
||||||
|
### 认知度
|
||||||
|
|
||||||
|
首先,我想看一下参与问卷调查的人是否**知道**下面六种语言:
|
||||||
|
|
||||||
|
- 经典的 JavaScript: 97%
|
||||||
|
- ES6: 98%
|
||||||
|
- CoffeeScript: 99%
|
||||||
|
- TypeScript: 98%
|
||||||
|
- Elm: 66%
|
||||||
|
- ClojureScript: 77%
|
||||||
|
|
||||||
|
你可能觉得 100% 的人都应该知道『经典的 JavaScript 』,我想是有人无法抵抗在一个 JavaScript 调查中投『我从来没有听说过 JavaScript 』这个选项的强大诱惑吧……
|
||||||
|
|
||||||
|
几乎所有人都知道 ES6、CoffeeScript 和 TypeScript 这三种语言,比较令我惊讶的是 TypeScript 竟然会稍微落后于 ES6 和 CoffeeScript。
|
||||||
|
|
||||||
|
另一方面,Elm 和 ClojureScript 得分就要低得多,当然这也有道理,因为它们跟自己的生态环境绑定的比较紧密,也很难在已有的 App 中进行使用。
|
||||||
|
|
||||||
|
### 兴趣度
|
||||||
|
|
||||||
|
接下来,让我们一起看一下,哪一种方言吸引新开发者的能力更强一些:
|
||||||
|
|
||||||
|
![](https://d3ugvbs94d921r.cloudfront.net/57c4dc599973d2525fee820a.png?t=3efc9491eba2ce2)
|
||||||
|
|
||||||
|
要注意,该表是统计该语言对从未使用过它们的用户的吸引度,因为只有很少人没有用过经典 JavaScript,所以『经典 JavaScript 』这一列的数值很低。
|
||||||
|
|
||||||
|
ES6的数值很有趣:已有很大比例的用户在使用 ES6 了,没有用过的人中的绝大部分(89%)也很想学习它。
|
||||||
|
|
||||||
|
TypeScript 和 Elm 的状态差不多:用过的人不多,但感兴趣的比例表现不错,分别是 53% 和 58%。
|
||||||
|
|
||||||
|
如果让我预测一下,那我觉得 TypeScript 和 Elm 都很难向普通的 JavaScript 开发者讲明自己的优势。毕竟如果开发者只懂 JavaScript 的话,你很难解释清楚静态类型的好处。
|
||||||
|
|
||||||
|
另外,只有很少开发者用过 CoffeeScript,而且很明显几乎没人想去学。我觉得我该写一本 12 卷厚的 CoffeeScript 百科全书了……
|
||||||
|
|
||||||
|
### 满意度
|
||||||
|
|
||||||
|
现在是最重要的问题的时间了:有多少开发者用过这些语言,有多少人还想继续使用这些方言呢?
|
||||||
|
|
||||||
|
![](https://d3ugvbs94d921r.cloudfront.net/57c4e5f79973d29461ee820a.png?t=1061d2ab8fc9838)
|
||||||
|
|
||||||
|
虽然经典 JavaScript 拥有最多的用户量,但就满意度来说 ES6 才是大赢家,而且我想现在已经能安全的说,ES6 可以作为开发 JavaScript App 默认的语言。
|
||||||
|
|
||||||
|
TypeScript 和 Elm 有相似的高满意度,都在 85% 上下。然后,只有可怜的 17% 的开发者会考虑继续使用 CoffeeScript。
|
||||||
|
|
||||||
|
### 快乐度
|
||||||
|
|
||||||
|
最后一个问题,我问大家在用现在的方式写 JavaScript 时是否感到快乐:
|
||||||
|
|
||||||
|
![](https://d3ugvbs94d921r.cloudfront.net/57c4cd8c9973d2d95bee820a.png?t=f53efb029ea4456)
|
||||||
|
|
||||||
|
这个问题的答案和上一个问题的满意度想匹配:平均分达到 3.96 分(1 - 5 分),大家在使用 JavaScript 时候确实是快乐的。
|
||||||
|
|
||||||
|
不过很难说高分是因为 JavaScript 最近的一些改进造成的呢,还是发现 JavaScript 可能(仅仅是可能)没有大家认为的那么讨厌。总之,JavaScript 令人满意。
|
||||||
|
|
||||||
|
### 总结
|
||||||
|
|
||||||
|
如果说上次的赢家是 React 和 Vue,那此次调查的冠军毫无争议是 ES6 了。 ES6 并带来没有开天辟地的变化,但整个社区都还是很认可当前 JavaScript 演进方向的。
|
||||||
|
|
||||||
|
我觉得一年之后我们再来一次这样的调查,结果会很有趣。同时也可以关注一下 TypeScript、Elm 还有ClojureScript 有没有继续进步。
|
||||||
|
|
||||||
|
个人认为,当前 JavaScript 大家庭百花齐放的现象还只是一个开始,或许几年之后 JavaScript 就会变得非常不同了。
|
||||||
|
|
||||||
|
### 结语 & 敬请期待
|
||||||
|
|
||||||
|
对于我这样的调查来说数据越多就意味着数据越准确!越多人参加这个调查,那就越能代表整个 JavaScript 社区。
|
||||||
|
|
||||||
|
所以,我十分希望你能帮忙分享这个调查问卷:
|
||||||
|
|
||||||
|
- [在 Twitter 上][1]
|
||||||
|
- [在 Facebook 上][2]
|
||||||
|
|
||||||
|
另外,如果你想收到我下一个调查结果分析,前往 [调查问卷主页][3] 并留下自己的邮箱吧。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://medium.com/@sachagreif/the-state-of-javascript-javascript-flavors-1e02b0bfefb6
|
||||||
|
|
||||||
|
作者:[Sacha Greif][a]
|
||||||
|
译者:[eriwoon](https://github.com/eriwoon)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://medium.com/@sachagreif
|
||||||
|
[1]: https://twitter.com/intent/tweet/?text=The%20State%20Of%20JavaScript%3A%20take%20a%20short%20survey%20about%20popular%20JavaScript%20technologies%20http%3A%2F%2Fstateofjs.com%20%23stateofjs
|
||||||
|
[2]: https://facebook.com/sharer/sharer.php?u=http%3A%2F%2Fstateofjs.com
|
||||||
|
[3]: http://stateofjs.com/
|
@ -0,0 +1,77 @@
|
|||||||
|
QOwnNotes:一款记录笔记和待办事项的应用,集成 ownCloud 云服务
|
||||||
|
===============
|
||||||
|
|
||||||
|
[QOwnNotes][1] 是一款自由而开源的笔记记录和待办事项的应用,可以运行在 Linux、Windows 和 mac 上。
|
||||||
|
|
||||||
|
这款程序将你的笔记保存为纯文本文件,它支持 Markdown 支持,并与 ownCloud 云服务紧密集成。
|
||||||
|
|
||||||
|
![](https://2.bp.blogspot.com/-a2vsrOG0zFk/V81gyHWlaaI/AAAAAAAAYZs/uzY16JtNcT8bnje1rTKJx1135WueY6V9gCLcB/s400/qownnotes.png)
|
||||||
|
|
||||||
|
QOwnNotes 的亮点就是它集成了 ownCloud 云服务(当然是可选的)。在 ownCloud 上用这款 APP,你就可以在网路上记录和搜索你的笔记,也可以在移动设备上使用(比如一款像 CloudNotes 的软件[2])。
|
||||||
|
|
||||||
|
不久以后,用你的 ownCloud 账户连接上 QOwnNotes,你就可以从你 ownCloud 服务器上分享笔记和查看或恢复之前版本记录的笔记(或者丢到垃圾箱的笔记)。
|
||||||
|
|
||||||
|
同样,QOwnNotes 也可以与 ownCloud 任务或者 Tasks Plus 应用程序相集成。
|
||||||
|
|
||||||
|
如果你不熟悉 [ownCloud][3] 的话,这是一款替代 Dropbox、Google Drive 和其他类似商业性的网络服务的自由软件,它可以安装在你自己的服务器上。它有一个网络界面,提供了文件管理、日历、照片、音乐、文档浏览等等功能。开发者同样提供桌面同步客户端以及移动 APP。
|
||||||
|
|
||||||
|
因为笔记被保存为纯文本,它们可以在不同的设备之间通过云存储服务进行同步,比如 Dropbox,Google Drive 等等,但是在这些应用中不能完全替代 ownCloud 的作用。
|
||||||
|
|
||||||
|
我提到的上述特点,比如恢复之前的笔记,只能在 ownCloud 下可用(尽管 Dropbox 和其他类似的也提供恢复以前的文件的服务,但是你不能在 QOwnnotes 中直接访问到)。
|
||||||
|
|
||||||
|
鉴于 QOwnNotes 有这么多优点,它支持 Markdown 语言(内置了 Markdown 预览模式),可以标记笔记,对标记和笔记进行搜索,在笔记中加入超链接,也可以插入图片:
|
||||||
|
|
||||||
|
![](https://4.bp.blogspot.com/-SuBhC43gzkY/V81oV7-zLBI/AAAAAAAAYZ8/l6nLQQSUv34Y7op_Xrma8XYm6EdWrhbIACLcB/s400/qownnotes_2.png)
|
||||||
|
|
||||||
|
标记嵌套和笔记文件夹同样支持。
|
||||||
|
|
||||||
|
代办事项管理功能比较基本还可以做一些改进,它现在打开在一个单独的窗口里,它也不用和笔记一样的编辑器,也不允许添加图片或者使用 Markdown 语言。
|
||||||
|
|
||||||
|
![](https://3.bp.blogspot.com/-AUeyZS3s_ck/V81opialKtI/AAAAAAAAYaA/xukIiZZUdNYBVZ92xgKEsEFew7q961CDwCLcB/s400/qownnotes-tasks.png)
|
||||||
|
|
||||||
|
它可以让你搜索你代办事项,设置事项优先级,添加提醒和显示完成的事项。此外,待办事项可以加入笔记中。
|
||||||
|
|
||||||
|
这款软件的界面是可定制的,允许你放大或缩小字体,切换窗格等等,也支持无干扰模式。
|
||||||
|
|
||||||
|
![](https://4.bp.blogspot.com/-Pnzw1wZde50/V81rrE6mTWI/AAAAAAAAYaM/0UZnH9ktbAgClkuAk1g6fgXK87kB_Bh0wCLcB/s400/qownnotes-distraction-free.png)
|
||||||
|
|
||||||
|
从程序的设置里,你可以开启黑夜模式(这里有个 bug,在 Ubuntu 16.04 里有些工具条图标消失了),改变状态条大小,字体和颜色方案(白天和黑夜):
|
||||||
|
|
||||||
|
![](https://1.bp.blogspot.com/-K1MGlXA8sxs/V81rv3fwL6I/AAAAAAAAYaQ/YDhhhnbJ9gY38B6Vz1Na_pHLCjLHhPWiwCLcB/s400/qownnotes-settings.png)
|
||||||
|
|
||||||
|
其他的特点有支持加密(笔记只能在 QOwnNotes 中加密),自定义键盘快捷键,输出笔记为 pdf 或者 Markdown,自定义笔记自动保存间隔等等。
|
||||||
|
|
||||||
|
访问 [QOwnNotes][11] 主页查看完整的特性。
|
||||||
|
|
||||||
|
### 下载 QOwnNotes
|
||||||
|
|
||||||
|
如何安装,请查看安装页(支持 Debian、Ubuntu、Linux Mint、openSUSE、Fedora、Arch Linux、KaOS、Gentoo、Slackware、CentOS 以及 Mac OSX 和 Windows)。
|
||||||
|
|
||||||
|
QOwnNotes 的 [snap][5] 包也是可用的,在 Ubuntu 16.04 或更新版本中,你可以通过 Ubuntu 的软件管理器直接安装它。
|
||||||
|
|
||||||
|
为了集成 QOwnNotes 到 ownCloud,你需要有 [ownCloud 服务器][6],同样也需要 [Notes][7]、[QOwnNotesAPI][8]、[Tasks][9]、[Tasks Plus][10] 等 ownColud 应用。这些可以从 ownCloud 的 Web 界面上安装,不需要手动下载。
|
||||||
|
|
||||||
|
请注意 QOenNotesAPI 和 Notes ownCloud 应用是实验性的,你需要“启用实验程序”来发现并安装他们,可以从 ownCloud 的 Web 界面上进行设置,在 Apps 菜单下,在左下角点击设置按钮。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.webupd8.org/2016/09/qownnotes-is-note-taking-and-todo-list.html
|
||||||
|
|
||||||
|
作者:[Andrew][a]
|
||||||
|
译者:[jiajia9linuxer](https://github.com/jiajia9linuxer)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.webupd8.org/p/about.html
|
||||||
|
[1]: http://www.qownnotes.org/
|
||||||
|
[2]: http://peterandlinda.com/cloudnotes/
|
||||||
|
[3]: https://owncloud.org/
|
||||||
|
[4]: http://www.qownnotes.org/installation
|
||||||
|
[5]: https://uappexplorer.com/app/qownnotes.pbek
|
||||||
|
[6]: https://download.owncloud.org/download/repositories/stable/owncloud/
|
||||||
|
[7]: https://github.com/owncloud/notes
|
||||||
|
[8]: https://github.com/pbek/qownnotesapi
|
||||||
|
[9]: https://apps.owncloud.com/content/show.php/Tasks?content=164356
|
||||||
|
[10]: https://apps.owncloud.com/content/show.php/Tasks+Plus?content=170561
|
||||||
|
[11]: http://www.qownnotes.org/
|
@ -0,0 +1,333 @@
|
|||||||
|
17 个 tar 命令实用示例
|
||||||
|
=====
|
||||||
|
|
||||||
|
Tar(Tape ARchive,磁带归档的缩写,LCTT 译注:最初设计用于将文件打包到磁带上,现在我们大都使用它来实现备份某个分区或者某些重要的目录)是类 Unix 系统中使用最广泛的命令,用于归档多个文件或目录到单个归档文件中,并且归档文件可以进一步使用 gzip 或者 bzip2 等技术进行压缩。换言之,tar 命令也可以用于备份:先是归档多个文件和目录到一个单独的 tar 文件或归档文件,然后在需要之时将 tar 文件中的文件和目录释放出来。
|
||||||
|
|
||||||
|
本文将介绍 tar 的 17 个实用示例。
|
||||||
|
|
||||||
|
tar 命令语法如下:
|
||||||
|
|
||||||
|
```
|
||||||
|
# tar <选项> <文件>
|
||||||
|
```
|
||||||
|
|
||||||
|
下面列举 tar 命令中一些常用的选项:
|
||||||
|
|
||||||
|
> --delete : 从归档文件 (而非磁带) 中删除
|
||||||
|
|
||||||
|
> -r, --append : 将文件追加到归档文件中
|
||||||
|
|
||||||
|
> -t, --list : 列出归档文件中包含的内容
|
||||||
|
|
||||||
|
> --test-label : 测试归档文件卷标并退出
|
||||||
|
|
||||||
|
> -u, --update : 将已更新的文件追加到归档文件中
|
||||||
|
|
||||||
|
> -x, --extract, --get : 释放归档文件中文件及目录
|
||||||
|
|
||||||
|
> -C, --directory=DIR : 执行归档动作前变更工作目录到 DIR
|
||||||
|
|
||||||
|
> -f, --file=ARCHIVE : 指定 (将要创建或已存在的) 归档文件名
|
||||||
|
|
||||||
|
> -j, --bip2 : 对归档文件使用 bzip2 压缩
|
||||||
|
|
||||||
|
> -J, --xz : 对归档文件使用 xz 压缩
|
||||||
|
|
||||||
|
> -p, --preserve-permissions : 保留原文件的访问权限
|
||||||
|
|
||||||
|
> -v, --verbose : 显示命令整个执行过程
|
||||||
|
|
||||||
|
> -z, gzip : 对归档文件使用 gzip 压缩
|
||||||
|
|
||||||
|
|
||||||
|
注 : 在 tar 命令选项中的连接符 `-` 是可选的(LCTT 译注:不用 `-` 也没事。这在 GNU 软件里面很罕见,大概是由于 tar 命令更多受到古老的 UNIX 风格影响)。
|
||||||
|
|
||||||
|
### 示例 1:创建一个 tar 归档文件
|
||||||
|
|
||||||
|
现在来创建一个 tar 文件,将 /etc/ 目录和 /root/anaconda-ks.cfg 文件打包进去。
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -cvf myarchive.tar /etc /root/anaconda-ks.cfg
|
||||||
|
```
|
||||||
|
|
||||||
|
以上命令会在当前目录创建一个名为 "myarchive" 的 tar 文件,内含 /etc/ 目录和 /root/anaconda-ks.cfg 文件。
|
||||||
|
|
||||||
|
其中,`-c` 选项表示要创建 tar 文件,`-v` 选项用于输出 tar 的详细过程到屏幕上,`-f` 选项则是指定归档文件名称。
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# ls -l myarchive.tar
|
||||||
|
-rw-r--r--. 1 root root 22947840 Sep 7 00:24 myarchive.tar
|
||||||
|
[root@linuxtechi ~]#
|
||||||
|
```
|
||||||
|
|
||||||
|
### 示例 2:列出归档文件中的内容
|
||||||
|
|
||||||
|
在 tar 命令中使用 `–t` 选项可以不用释放其中的文件就可以快速列出文件中包含的内容。
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -tvf myarchive.tar
|
||||||
|
```
|
||||||
|
|
||||||
|
列出 tar 文件中的指定的文件和目录。下列命令尝试查看 anaconda-ks.cfg 文件是否存在于 tar 文件中。
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -tvf myarchive.tar root/anaconda-ks.cfg
|
||||||
|
-rw------- root/root 953 2016-08-24 01:33 root/anaconda-ks.cfg
|
||||||
|
[root@linuxtechi ~]#
|
||||||
|
```
|
||||||
|
|
||||||
|
### 示例 3:追加文件到归档(tar)文件中
|
||||||
|
|
||||||
|
`-r` 选项用于向已有的 tar 文件中追加文件。下面来将 /etc/fstab 添加到 data.tar 中。
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -rvf data.tar /etc/fstab
|
||||||
|
```
|
||||||
|
|
||||||
|
注:在压缩过的 tar 文件中无法进行追加文件操作。
|
||||||
|
|
||||||
|
### 示例 4:从 tar 文件中释放文件以及目录
|
||||||
|
|
||||||
|
`-x` 选项用于释放出 tar 文件中的文件和目录。下面来释放上边创建的 tar 文件中的内容。
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -xvf myarchive.tar
|
||||||
|
```
|
||||||
|
|
||||||
|
这个命令会在当前目录中释放出 myarchive.tar 文件中的内容。
|
||||||
|
|
||||||
|
### 示例 5:释放 tar 文件到指定目录
|
||||||
|
|
||||||
|
假如你想要释放 tar 文件中的内容到指定的文件夹或者目录,使用 `-C` 选项后边加上指定的文件的路径。
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -xvf myarchive.tar -C /tmp/
|
||||||
|
```
|
||||||
|
|
||||||
|
### 示例 6:释放 tar 文件中的指定文件或目录
|
||||||
|
|
||||||
|
假设你只要释放 tar 文件中的 anaconda-ks.cfg 到 /tmp 目录。
|
||||||
|
|
||||||
|
语法如下:
|
||||||
|
|
||||||
|
```
|
||||||
|
# tar –xvf {tar-file } {file-to-be-extracted } -C {path-where-to-extract}
|
||||||
|
|
||||||
|
[root@linuxtechi tmp]# tar -xvf /root/myarchive.tar root/anaconda-ks.cfg -C /tmp/
|
||||||
|
root/anaconda-ks.cfg
|
||||||
|
[root@linuxtechi tmp]# ls -l /tmp/root/anaconda-ks.cfg
|
||||||
|
-rw-------. 1 root root 953 Aug 24 01:33 /tmp/root/anaconda-ks.cfg
|
||||||
|
[root@linuxtechi tmp]#
|
||||||
|
```
|
||||||
|
|
||||||
|
### 示例 7:创建并压缩归档文件(.tar.gz 或 .tgz)
|
||||||
|
|
||||||
|
假设我们需要打包 /etc 和 /opt 文件夹,并用 gzip 工具将其压缩。可以在 tar 命令中使用 `-z` 选项来实现。这种 tar 文件的扩展名可以是 .tar.gz 或者 .tgz。
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -zcpvf myarchive.tar.gz /etc/ /opt/
|
||||||
|
```
|
||||||
|
|
||||||
|
或
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -zcpvf myarchive.tgz /etc/ /opt/
|
||||||
|
```
|
||||||
|
|
||||||
|
### 示例 8:创建并压缩归档文件(.tar.bz2 或 .tbz2)
|
||||||
|
|
||||||
|
假设我们需要打包 /etc 和 /opt 文件夹,并使用 bzip2 压缩。可以在 tar 命令中使用 `-j` 选项来实现。这种 tar 文件的扩展名可以是 .tar.bz2 或者 .tbz。
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -jcpvf myarchive.tar.bz2 /etc/ /opt/
|
||||||
|
```
|
||||||
|
|
||||||
|
或
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -jcpvf myarchive.tbz2 /etc/ /opt/
|
||||||
|
```
|
||||||
|
|
||||||
|
### 示例 9:排除指定文件或类型后创建 tar 文件
|
||||||
|
|
||||||
|
创建 tar 文件时在 tar 命令中使用 `–exclude` 选项来排除指定文件或者类型。假设在创建压缩的 tar 文件时要排除 .html 文件。
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -zcpvf myarchive.tgz /etc/ /opt/ --exclude=*.html
|
||||||
|
```
|
||||||
|
|
||||||
|
### 示例 10:列出 .tar.gz 或 .tgz 文件中的内容
|
||||||
|
|
||||||
|
使用 `-t` 选项可以查看 .tar.gz 或 .tgz 文件中内容。如下:
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -tvf myarchive.tgz | more
|
||||||
|
.............................................
|
||||||
|
drwxr-xr-x root/root 0 2016-09-07 08:41 etc/
|
||||||
|
-rw-r--r-- root/root 541 2016-08-24 01:23 etc/fstab
|
||||||
|
-rw------- root/root 0 2016-08-24 01:23 etc/crypttab
|
||||||
|
lrwxrwxrwx root/root 0 2016-08-24 01:23 etc/mtab -> /proc/self/mounts
|
||||||
|
-rw-r--r-- root/root 149 2016-09-07 08:41 etc/resolv.conf
|
||||||
|
drwxr-xr-x root/root 0 2016-09-06 03:55 etc/pki/
|
||||||
|
drwxr-xr-x root/root 0 2016-09-06 03:15 etc/pki/rpm-gpg/
|
||||||
|
-rw-r--r-- root/root 1690 2015-12-09 04:59 etc/pki/rpm-gpg/RPM-GPG-KEY-CentOS-7
|
||||||
|
-rw-r--r-- root/root 1004 2015-12-09 04:59 etc/pki/rpm-gpg/RPM-GPG-KEY-CentOS-Debug-7
|
||||||
|
-rw-r--r-- root/root 1690 2015-12-09 04:59 etc/pki/rpm-gpg/RPM-GPG-KEY-CentOS-Testing-7
|
||||||
|
-rw-r--r-- root/root 3140 2015-09-15 06:53 etc/pki/rpm-gpg/RPM-GPG-KEY-foreman
|
||||||
|
..........................................................
|
||||||
|
```
|
||||||
|
|
||||||
|
### 示例 11:列出 .tar.bz2 或 .tbz2 文件中的内容
|
||||||
|
|
||||||
|
使用 `-t` 选项可以查看 .tar.bz2 或 .tbz2 文件中内容。如下:
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -tvf myarchive.tbz2 | more
|
||||||
|
........................................................
|
||||||
|
rwxr-xr-x root/root 0 2016-08-24 01:25 etc/pki/java/
|
||||||
|
lrwxrwxrwx root/root 0 2016-08-24 01:25 etc/pki/java/cacerts -> /etc/pki/ca-trust/extracted/java/cacerts
|
||||||
|
drwxr-xr-x root/root 0 2016-09-06 02:54 etc/pki/nssdb/
|
||||||
|
-rw-r--r-- root/root 65536 2010-01-12 15:09 etc/pki/nssdb/cert8.db
|
||||||
|
-rw-r--r-- root/root 9216 2016-09-06 02:54 etc/pki/nssdb/cert9.db
|
||||||
|
-rw-r--r-- root/root 16384 2010-01-12 16:21 etc/pki/nssdb/key3.db
|
||||||
|
-rw-r--r-- root/root 11264 2016-09-06 02:54 etc/pki/nssdb/key4.db
|
||||||
|
-rw-r--r-- root/root 451 2015-10-21 09:42 etc/pki/nssdb/pkcs11.txt
|
||||||
|
-rw-r--r-- root/root 16384 2010-01-12 15:45 etc/pki/nssdb/secmod.db
|
||||||
|
drwxr-xr-x root/root 0 2016-08-24 01:26 etc/pki/CA/
|
||||||
|
drwxr-xr-x root/root 0 2015-06-29 08:48 etc/pki/CA/certs/
|
||||||
|
drwxr-xr-x root/root 0 2015-06-29 08:48 etc/pki/CA/crl/
|
||||||
|
drwxr-xr-x root/root 0 2015-06-29 08:48 etc/pki/CA/newcerts/
|
||||||
|
drwx------ root/root 0 2015-06-29 08:48 etc/pki/CA/private/
|
||||||
|
drwx------ root/root 0 2015-11-20 06:34 etc/pki/rsyslog/
|
||||||
|
drwxr-xr-x root/root 0 2016-09-06 03:44 etc/pki/pulp/
|
||||||
|
..............................................................
|
||||||
|
```
|
||||||
|
|
||||||
|
### 示例 12:解压 .tar.gz 或 .tgz 文件
|
||||||
|
|
||||||
|
使用 `-x` 和 `-z` 选项来解压 .tar.gz 或 .tgz 文件。如下:
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -zxpvf myarchive.tgz -C /tmp/
|
||||||
|
```
|
||||||
|
|
||||||
|
以上命令将 tar 文件解压到 /tmp 目录。
|
||||||
|
|
||||||
|
注:现今的 tar 命令会在执行解压动作前自动检查文件的压缩类型,这意味着我们在使用 tar 命令是可以不用指定文件的压缩类型。如下:
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -xpvf myarchive.tgz -C /tmp/
|
||||||
|
```
|
||||||
|
|
||||||
|
### 示例 13:解压 .tar.bz2 或 .tbz2 文件
|
||||||
|
|
||||||
|
使用 `-j` 和 `-x` 选项来解压 .tar.bz2 或 .tbz2 文件。如下:
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -jxpvf myarchive.tbz2 -C /tmp/
|
||||||
|
```
|
||||||
|
|
||||||
|
或
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar xpvf myarchive.tbz2 -C /tmp/
|
||||||
|
```
|
||||||
|
|
||||||
|
### 示例 14:使用 tar 命令进行定时备份
|
||||||
|
|
||||||
|
总有一些实时场景需要我们对指定的文件和目录进行打包,已达到日常备份的目的。假设需要每天备份整个 /opt 目录,可以创建一个带 tar 命令的 cron 任务来完成。如下:
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -zcvf optbackup-$(date +%Y-%m-%d).tgz /opt/
|
||||||
|
```
|
||||||
|
|
||||||
|
为以上命令创建一个 cron 任务即可。
|
||||||
|
|
||||||
|
### 示例 15:使用 -T 及 -X 创建压缩归档文件
|
||||||
|
|
||||||
|
想像这样一个场景:把想要归档和压缩的文件及目录记录到到一个文件,然后把这个文件当做 tar 命令的传入参数来完成归档任务;而有时候则是需要排除上面提到的这个文件里面记录的特定路径后进行归档和压缩。
|
||||||
|
|
||||||
|
在 tar 命令中使用 `-T` 选项来指定该输入文件,使用 `-X` 选项来指定包含要排除的文件列表。
|
||||||
|
|
||||||
|
假设要归档 /etc、/opt、/home 目录,并排除 /etc/sysconfig/kdump 和 /etc/sysconfig/foreman 文件,可以创建 /root/tar-include 和 /root/tar-exclude 然后分别输入以下内容:
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# cat /root/tar-include
|
||||||
|
/etc
|
||||||
|
/opt
|
||||||
|
/home
|
||||||
|
[root@linuxtechi ~]#
|
||||||
|
[root@linuxtechi ~]# cat /root/tar-exclude
|
||||||
|
/etc/sysconfig/kdump
|
||||||
|
/etc/sysconfig/foreman
|
||||||
|
[root@linuxtechi ~]#
|
||||||
|
```
|
||||||
|
|
||||||
|
运行以下命令来创建一个压缩归档文件。
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar zcpvf mybackup-$(date +%Y-%m-%d).tgz -T /root/tar-include -X /root/tar-exclude
|
||||||
|
```
|
||||||
|
|
||||||
|
### 示例 16:查看 .tar、.tgz 和 .tbz2 文件的大小
|
||||||
|
|
||||||
|
使用如下命令来查看 (压缩) tar 文件的体积。
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# tar -czf - data.tar | wc -c
|
||||||
|
427
|
||||||
|
[root@linuxtechi ~]# tar -czf - mybackup-2016-09-09.tgz | wc -c
|
||||||
|
37956009
|
||||||
|
[root@linuxtechi ~]# tar -czf - myarchive.tbz2 | wc -c
|
||||||
|
30835317
|
||||||
|
[root@linuxtechi ~]#
|
||||||
|
```
|
||||||
|
|
||||||
|
### 示例 17:分割体积庞大的 tar 文件为多份小文件
|
||||||
|
|
||||||
|
类 Unix 系统中使用 split 命令来将大体积文件分割成小体积文件。大体积的 tar 当然也可以使用这个命令来进行分割。
|
||||||
|
|
||||||
|
假设需要将 "mybackup-2016-09-09.tgz" 分割成每份 6 MB 的小文件。
|
||||||
|
|
||||||
|
```
|
||||||
|
Syntax : split -b <Size-in-MB> <tar-file-name>.<extension> “prefix-name”
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# split -b 6M mybackup-2016-09-09.tgz mybackup-parts
|
||||||
|
```
|
||||||
|
|
||||||
|
以上命令会在当前目录分割 mybackup-2016-09-09.tgz 文件成为多个 6 MB 的小文件,文件名为 mybackup-partsaa ~ mybackup-partsag。如果在要在分割文件后以数字而非字母来区分,可以在以上的 split 命令使用 `-d` 选项。
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# ls -l mybackup-parts*
|
||||||
|
-rw-r--r--. 1 root root 6291456 Sep 10 03:05 mybackup-partsaa
|
||||||
|
-rw-r--r--. 1 root root 6291456 Sep 10 03:05 mybackup-partsab
|
||||||
|
-rw-r--r--. 1 root root 6291456 Sep 10 03:05 mybackup-partsac
|
||||||
|
-rw-r--r--. 1 root root 6291456 Sep 10 03:05 mybackup-partsad
|
||||||
|
-rw-r--r--. 1 root root 6291456 Sep 10 03:05 mybackup-partsae
|
||||||
|
-rw-r--r--. 1 root root 6291456 Sep 10 03:05 mybackup-partsaf
|
||||||
|
-rw-r--r--. 1 root root 637219 Sep 10 03:05 mybackup-partsag
|
||||||
|
[root@linuxtechi ~]#
|
||||||
|
```
|
||||||
|
|
||||||
|
然后通过网络将这些分割文件转移到其他服务器,就可以合并成为一个单独的 tar 文件了,如下:
|
||||||
|
|
||||||
|
```
|
||||||
|
[root@linuxtechi ~]# cat mybackup-partsa* > mybackup-2016-09-09.tgz
|
||||||
|
[root@linuxtechi ~]#
|
||||||
|
```
|
||||||
|
|
||||||
|
文毕,希望你喜欢 tar 命令的这几个不同的示例。随时评论并分享你的心得。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.linuxtechi.com/17-tar-command-examples-in-linux/
|
||||||
|
|
||||||
|
作者:[Pradeep Kumar][a]
|
||||||
|
译者:[GHLandy](https://github.com/GHLandy)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.linuxtechi.com/author/pradeep/
|
@ -0,0 +1,58 @@
|
|||||||
|
5 个值得了解的 Linux 服务器发行版
|
||||||
|
=========
|
||||||
|
|
||||||
|
> 你在 Distrowatch.com 上看到列出的将近 300 个 Linux 发行版本中,几乎任何一个发行版都可以被用来作为服务器系统。下面是一些相对于其他发行版而言比较突出的一些发行版。
|
||||||
|
|
||||||
|
![](http://windowsitpro.com/site-files/windowsitpro.com/files/imagecache/large_img/uploads/2016/09/cloudservers.jpg)
|
||||||
|
|
||||||
|
你在 Distrowatch.com 上看到列出的将近 300 个 Linux 发行版本中,几乎任何一个发行版都可以被用来作为服务器系统,在 Linux 发展的早期,给用户提供的一直是“全能”发行版,例如 Slackware、Debian 和 Gentoo 可以为家庭和企业作为服务器完成繁重的工作。那或许对业余爱好者是不错的,但是它对于专业人员来说也有好多不必要的地方。
|
||||||
|
|
||||||
|
首先,这里有一些发行版可以作为文件和应用服务器,给工作站提供常见外围设备的共享,提供网页服务和其它我们希望服务器做的任何工作,不管是在云端、在数据中心或者在服务器机架上,除此之外没有别的用途。
|
||||||
|
|
||||||
|
下面是 5 个最常用的 Linux 发行版的简单总结,而且每一个发行版都可以满足小型企业的需求。
|
||||||
|
|
||||||
|
### Red Hat Enterprise Linux(RHEL)
|
||||||
|
|
||||||
|
这或许是最有名的 Linux 服务器发行版了。RHEL 以它在高要求的至关重要的任务上坚如磐石的稳定性而出名,例如运行着纽约证券交易系统。红帽也提供了业内最佳的服务支持。
|
||||||
|
|
||||||
|
那么红帽 Linux 的缺点都有什么呢? 尽管红帽以提供首屈一指的客户服务和支持而出名,但是它的支持订阅费用并不便宜。有人可能会指出,这的确物有所值。确实有便宜的 RHEL 第三方服务,但是你或许应该在这么做之前做一些研究。
|
||||||
|
|
||||||
|
### CentOS
|
||||||
|
|
||||||
|
任何喜欢 RHEL,但是又不想给红帽付费来获得支持的人都应该了解一下 CentOS,它基本上是红帽企业版 Linux 的一个分支。尽管这个项目 2004 年左右才开始,但它在 2014 年得到了红帽的官方支持,而它现在雇佣可这个项目的大多数开发者,这意味着安全补丁和漏洞修复提交到红帽不久后就会在 CentOS 上可用。
|
||||||
|
|
||||||
|
如果你想要部署 CentOS,你将需要有 Linux 技能的员工,因为没有了技术支持,你基本上只能靠自己。有一个好消息是 CentOS 社区提供了十分丰富的资源,例如邮件列表、Web 论坛和聊天室,所以对那些寻找帮助的人来说,社区帮助还是有的。
|
||||||
|
|
||||||
|
### Ubuntu Server
|
||||||
|
|
||||||
|
当许多年前 Canonical 宣布它将要推出一个服务器版本的 Ubuntu 的时候,你可能会听到过嘲笑者的声音。然而嘲笑很快变成了惊奇,Ubuntu Server 迅速地站稳了脚跟。部分原因是因为其来自 Debian 派生的基因,Debian 长久以来就是一个备受喜爱的 Linux 服务器发行版,Ubuntu 通过提供一般人可以支付的起的技术支持费用、优秀的硬件支持、开发工具和很多亮点填补了这个市场空隙。
|
||||||
|
|
||||||
|
那么 Ubuntu Server 有多么受欢迎呢?最近的数据表明它正在成为在 OpenStack 和 Amazon Elastic Compute Cloud 上[部署最多的操作系统][1]。在那里 Ubuntu Server 超过了位居第二的 Amazon Linux 的 Amazon Machine Image 一大截,而且让第三位 Windows 处于尘封的地位。另外一个调查显示 Ubuntu Server 是[使用最多的 Linux web 服务器][2]。
|
||||||
|
|
||||||
|
### SUSE Linux Enterprise Server(SLES)
|
||||||
|
|
||||||
|
这个源自德国的发行版在欧洲有很大的用户群,而且在本世纪初由 Novell 公司引起的 PR 问题出现之前,它一直都是大西洋这边的排名第一服务器发行版。在那段漫长的时期之后,SUSE 在美国获得了发展,而且它的使用或许加速了惠普企业公司将它作为[ Linux 首选合作伙伴][3]。
|
||||||
|
|
||||||
|
SLES 稳定而且易于维护,这正是很久以来对于一个好的 Linux 发行版所期待的东西。付费的 7*24 小时快速响应技术支持可以供你选择,使得这发行版很适合关键任务的部署。
|
||||||
|
|
||||||
|
### ClearOS
|
||||||
|
|
||||||
|
基于 RHEL,之所以这里要包括 [ClearOS][4] 是因为它对于每个人来说都足够简单,甚至是没有专业知识的人都可以去配置它。它定位于服务中小型企业,它也可以被家庭用户用来作为娱乐服务器,为了简单易用我们可以基于 Web 界面进行管理,它是以“构建你的 IT 基础设施应该像在智能手机上下载 app 一样简单”为前提来定制的。
|
||||||
|
|
||||||
|
最新的 7.2 发行版本,包括了一些可能并不“轻量级”的功能,例如对微软 Hyper-V 技术的 VM 支持,支持 XFS 和 BTRFS 文件系统,也支持 LVM 和 IPv6。这些新特性在免费版本或者在并不算贵的带着各种支持选项的专业版中都是可用的。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://windowsitpro.com/industry/five-linux-server-distros-worth-checking-out
|
||||||
|
|
||||||
|
作者:[Christine Hall][a]
|
||||||
|
译者:[LinuxBars](https://github.com/LinuxBars)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://windowsitpro.com/industry/five-linux-server-distros-worth-checking-out
|
||||||
|
[1]: http://www.zdnet.com/article/ubuntu-linux-continues-to-dominate-openstack-and-other-clouds/
|
||||||
|
[2]: https://w3techs.com/technologies/details/os-linux/all/all
|
||||||
|
[3]: http://windowsitpro.com/industry/suse-now-hpes-preferred-partner-micro-focus-pact
|
||||||
|
[4]: https://www.clearos.com/
|
@ -0,0 +1,58 @@
|
|||||||
|
在 WordPress 下如何通过 Markdown 来提高工作效率
|
||||||
|
=================
|
||||||
|
|
||||||
|
![](https://maketecheasier-2d0f.kxcdn.com/assets/uploads/2016/09/markdown-wordpress-featured-2.jpg)
|
||||||
|
|
||||||
|
Markdown 是一种简单的标记语言,旨在帮助你花费更小的代价来格式化纯文本文档。在 WordPress 下你可以使用 HTML 或者可视化编辑器来格式化你的文档,但是使用 markdown 可以让格式化文档变得更加容易,而且你随时可以导出成很多种格式,包括(但不限于)HTML。
|
||||||
|
|
||||||
|
WordPress 没有原生的 markdown 的支持,但是,如果你希望的话,在你的网站上有多种插件可以添加这种功能。
|
||||||
|
|
||||||
|
在这个教程中,我将会演示如何使用流行的 WP-Markdown 插件为 WordPress 网站添加 markdown 支持。
|
||||||
|
|
||||||
|
### 安装
|
||||||
|
|
||||||
|
导航到 “Plugins -> Add New”,然后在提供的搜索框中输入 “[wp-markdown][1]” 就可以直接安装。插件应该会出现在列表中的第一个。单击 “Install Now” 进行安装。
|
||||||
|
|
||||||
|
![](https://maketecheasier-2d0f.kxcdn.com/assets/uploads/2016/08/markdown-wordpress-install-plugin-1.png)
|
||||||
|
|
||||||
|
### 配置
|
||||||
|
|
||||||
|
当你已经安装了这个插件并且激活它之后,导航到 “Settings -> Writing” 并向下滚动,直到 markdown 选项。
|
||||||
|
|
||||||
|
你可以启用文章、页面和评论中对于 markdown 的支持。如果你刚刚开始学习 markdown 语法的话,那么你也可以在文章编辑器或者评论的地方启用一个帮助栏,这可以使你更方便一些。
|
||||||
|
|
||||||
|
![](https://maketecheasier-2d0f.kxcdn.com/assets/uploads/2016/09/markdown-wordpress-configuration.png)
|
||||||
|
|
||||||
|
如果在你的博客文章中包括代码片段的话,那么启用 “Prettify syntax highlighter” 将会让你的代码片段自动语法高亮。
|
||||||
|
|
||||||
|
一旦对于你的选择感觉满意的话,那么就单击 “Save Changes” 来保存你的设置吧。
|
||||||
|
|
||||||
|
### 使用 Markdown 来写你的文章
|
||||||
|
|
||||||
|
当你在自己网站中启用了 markdown 的支持,你就可以立马开始使用了。
|
||||||
|
|
||||||
|
通过 “Posts -> Add New” 创建一篇新的文章。你将会注意到默认的可视化及纯文本编辑器已经被 markdown 编辑器所替代。
|
||||||
|
|
||||||
|
如果你在配置选项中没有启用 markdown 的帮助栏,你将不会看到 markdown 格式化后的实时预览。然而,只要你的语法是正确的,当你保存或者发布文章的时候,你的 markdown 就会转换成正确的 HTML。
|
||||||
|
|
||||||
|
然而,如果你是 markdown 的初学者的话,实时预览这一特征对你会很重要,只需要简单的回到刚才的设置中启用帮助栏选项,你就可以在你的文章底部看到一块漂亮的实时预览区域。另外,你也可以在顶部看到很多按钮,它们将帮助你在文章中快速的插入 markdown 格式。
|
||||||
|
|
||||||
|
![](https://maketecheasier-2d0f.kxcdn.com/assets/uploads/2016/08/markdown-wordpress-create-post.png)
|
||||||
|
|
||||||
|
### 结语
|
||||||
|
|
||||||
|
正如你所看到的那样,在 WordPress 网站上添加 markdown 支持确实容易,你将只需要花费几分钟的时间就可以了。如果对于 markdown 你全然不知的话,或许你也可以查看我们的 [markdown 备忘录][2],它将对于 markdown 语法提供一个全面的参考。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://www.maketecheasier.com/use-markdown-in-wordpress/
|
||||||
|
|
||||||
|
作者:[Ayo Isaiah][a]
|
||||||
|
译者:[yangmingming](https://github.com/yangmingming)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://www.maketecheasier.com/author/ayoisaiah/
|
||||||
|
[1]: https://wordpress.org/plugins/wp-markdown/
|
||||||
|
[2]: https://www.maketecheasier.com/productive-with-markdown-cheatsheet/
|
@ -0,0 +1,63 @@
|
|||||||
|
Linus Torvalds 透露他编程最喜欢使用的笔记本
|
||||||
|
=================
|
||||||
|
|
||||||
|
> 是戴尔 XPS 13 开发者版。下面就是原因。
|
||||||
|
|
||||||
|
我最近和一些 Linux 开发者讨论了对于严谨的程序员来说,最好的笔记本是什么样的问题。结果,我从这些程序员的观点中筛选出了多款笔记本电脑。那么依我之见赢家是谁呢?就是戴尔 XPS 13 开发者版。和我观点一样的大有人在。Linux的缔造者 Linus Torvalds 也认同这个观点。对于他来说,戴尔 XPS 13 开发者版大概是最好的笔记本电脑了。
|
||||||
|
|
||||||
|
![Linus Torvalds new favorite laptop is the Dell XPS 13 Developer Edition.](http://zdnet3.cbsistatic.com/hub/i/r/2016/07/18/702609c3-db38-4603-9f5f-4dcc3d71b140/resize/770xauto/50a8ba1c2acb1f0994aec2115d2e55ce/2016-dell-xps-13.jpg)
|
||||||
|
|
||||||
|
Torvalds 的需求可能和你的不同。
|
||||||
|
|
||||||
|
在 Google+ 上,Torvalds 解释道,“第一:[我从来不把笔记本当成台式机的替代品][1],并且,我每年旅游不了几次。所以对于我来说,笔记本是一个相当专用的东西,并不是每日(甚至每周)都要使用,因此,主要的标准不是那种“差不多每天都使用”的标准,而是非常适合于旅游时使用。”
|
||||||
|
|
||||||
|
因此,对于 Torvalds 来说,“我最后比较关心的一点是它是相当的小和轻,因为在会议上我可能一整天都需要带着它。我同样需要一个好的屏幕,因为到目前为止,我主要是在桌子上使用它,我希望文字的显示小而且清晰。”
|
||||||
|
|
||||||
|
戴尔的显示器是由 Intel's Iris 540 GPU 支持的。在我的印象中,它表现的非常的不错。
|
||||||
|
|
||||||
|
Iris 驱动了 13.3 英寸的 3200×1800 的显示屏。每英寸有 280 像素,比我喜欢的 [2015 年的 Chromebook Pixel][2] 多了 40 个像素,比 [Retina 屏的 MacBook Pro][3] 还要多 60 个像素。
|
||||||
|
|
||||||
|
然而,要让上面说的硬件配置在 [Gnome][4] 桌面上玩好也不容易。正如 Torvalds 在另一篇文章解释的那样,它“[和我的桌面电脑有一样的分辨率][5],但是,显然因为笔记本的显示屏更小,Gnome 桌面似乎自己做了个艰难的决定,认为我需要 2 倍的自动缩放比例,这简直愚蠢到炸裂(例如窗口显示,图标等)。”
|
||||||
|
|
||||||
|
解决方案?你不用想着在用户界面里面找了。你需要在 shell下运行:`gsettings set org.gnome.desktop.interface scaling-factor 1`。
|
||||||
|
|
||||||
|
Torvalds 或许使用 Gnome 桌面,但是他不是很喜欢 Gnome 3.x 系列。这一点上我跟他没有不同意见。这就是为什么我使用 [Cinnamon][7] 来代替。
|
||||||
|
|
||||||
|
他还希望“一个相当强大的 CPU,因为当我旅游的时候,我依旧需要编译 Linux 内核很多次。我并不会像在家那样每次 pull request 都进行一次完整的“make allmodconfig”编译,但是我希望可以比我以前的笔记本多编译几次,实际上,这(也包括屏幕)应该是我想升级的主要原因。”
|
||||||
|
|
||||||
|
Linus 没有描述他的 XPS 13 的细节,但是我测评过的那台是一个高端机型。它带有双核 2.2GHz 的第 6 代英特尔的酷睿 i7-6550U Skylake 处理器,16GBs DDR3 内存,以及一块半 TB (500GB)的 PCIe 固态硬盘(SSD)。我可以肯定,Torvalds 的系统至少是精良装备。”
|
||||||
|
|
||||||
|
一些你或许会关注的特征没有在 Torvalds 的清单中:
|
||||||
|
|
||||||
|
> “我不会关心的是触摸屏,因为我的手指相对于我所看到的文字是又大又笨拙(我也无法处理污渍:也许我的手指特别油腻,但是我真的不想碰那些屏幕)。
|
||||||
|
|
||||||
|
> 我并不十分关心那些“一整天的电池寿命”,因为坦率的讲,我不记得上次没有接入电源时什么时候了。我可能着急忙慌而忘记插电,但它不是一个天大的问题。现在电池的寿命“超过两小时”,我只是不那么在乎了。”
|
||||||
|
|
||||||
|
戴尔声称,XPS 13,搭配 56 瓦小时的 4 芯电池,拥有 12 小时的电池寿命。以我的使用经验它已经很好的用过了 10 个小时。我从没有尝试过把电量完全耗完是什么状态。
|
||||||
|
|
||||||
|
Torvalds 也没有遇到 Intel 的 Wi-Fi 设备问题。非开发者版使用 Broadcom 的芯片设备,已经被 Windows 和 Linux 用户发现了一些问题。戴尔的技术支持对于我来解决这些问题非常有帮助。
|
||||||
|
|
||||||
|
一些用户在使用 XPS 13 触摸板的时候,遇到了问题。Torvalds 和我都几乎没有什么困扰。Torvalds 写到,“XPS13 触摸板对于我来说运行的非常好。这可能只是个人喜好,但它操作起来比较流畅,响应比较快。”
|
||||||
|
|
||||||
|
不过,尽管 Torvalds 喜欢 XPS 13,他同时也钟情于最新版的联想 X1 Carbon、惠普 Spectre 13 x360,和去年的联想 Yoga 900。至于我?我喜欢 XPS 13 开发者版。至于价钱,我以前测评过的型号是 $1949.99,可能刷你的信用卡就够了。
|
||||||
|
|
||||||
|
因此,如果你希望像世界上顶级的程序员之一一样开发的话,Dell XPS 13 开发者版对得起它的价格。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.zdnet.com/article/linus-torvalds-reveals-his-favorite-programming-laptop/
|
||||||
|
|
||||||
|
作者:[Steven J. Vaughan-Nichols][a]
|
||||||
|
译者:[yangmingming](https://github.com/yangmingming)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.zdnet.com/meet-the-team/us/steven-j-vaughan-nichols/
|
||||||
|
[1]: https://plus.google.com/+LinusTorvalds/posts/VZj8vxXdtfe
|
||||||
|
[2]: http://www.zdnet.com/article/the-best-chromebook-ever-the-chromebook-pixel-2015/
|
||||||
|
[3]: http://www.zdnet.com/product/apple-15-inch-macbook-pro-with-retina-display-mid-2015/
|
||||||
|
[4]: https://www.gnome.org/
|
||||||
|
[5]: https://plus.google.com/+LinusTorvalds/posts/d7nfnWSXjfD
|
||||||
|
[6]: http://www.zdnet.com/article/linus-torvalds-finds-gnome-3-4-to-be-a-total-user-experience-design-failure/
|
||||||
|
[7]: http://www.zdnet.com/article/how-to-customise-your-linux-desktop-cinnamon/
|
@ -0,0 +1,54 @@
|
|||||||
|
Taskwarrior:Linux 下一个很棒的命令行 TODO 工具
|
||||||
|
==============
|
||||||
|
|
||||||
|
Taskwarrior 是 Ubuntu/Linux 下一个简单而直接的基于命令行的 TODO 工具。这个开源软件是我曾用过的最简单的[基于命令行的工具][4]之一。Taskwarrior 可以帮助你更好地组织你自己,而不用安装笨重的新工具——这有时丧失了 TODO 工具的目的。
|
||||||
|
|
||||||
|
![](https://2.bp.blogspot.com/-pQnRlOUNIxk/V9cuc3ytsBI/AAAAAAAAKHs/yYxyiAk4PwMIE0HTxlrm6arWOAPcBRRywCLcB/s1600/taskwarrior-todo-app.png)
|
||||||
|
|
||||||
|
### Taskwarrior:一个基于简单的基于命令行帮助完成任务的TODO工具
|
||||||
|
|
||||||
|
Taskwarrior是一个开源、跨平台、基于命令行的 TODO 工具,它帮你在终端中管理你的 to-do 列表。这个工具让你可以轻松地添加任务、展示列表、移除任务。而且,在你的默认仓库中就有,不用安装新的 PPA。在 Ubuntu 16.04 LTS 或者相似的发行版中。在终端中按照如下步骤安装 Taskwarrior。
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt-get install task
|
||||||
|
```
|
||||||
|
|
||||||
|
简单的用如下:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ task add Read a book
|
||||||
|
Created task 1.
|
||||||
|
$ task add priority:H Pay the bills
|
||||||
|
Created task 2.
|
||||||
|
```
|
||||||
|
|
||||||
|
我使用上面截图中的同样一个例子。是的,你可以设置优先级(H、L 或者 M)。并且你可以使用‘task’或者‘task next’命令来查看你最新创建的 to-do 列表。比如:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ task next
|
||||||
|
|
||||||
|
ID Age P Description Urg
|
||||||
|
-- --- - -------------------------------- ----
|
||||||
|
2 10s H Pay the bills 6
|
||||||
|
1 20s Read a book 0
|
||||||
|
```
|
||||||
|
|
||||||
|
完成之后,你可以使用 ‘task 1 done’ 或者 ‘task 2 done’ 来清除列表。[可以在这里][1]找到更加全面的命令和使用案例。同样,Taskwarrior 是跨平台的,这意味着不管怎样,你都可以找到一个[满足你需求][2]的版本。如果你需要的话,这里甚至有[一个安卓版][3]。祝您用得开心!
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.techdrivein.com/2016/09/taskwarrior-command-line-todo-app-linux.html
|
||||||
|
|
||||||
|
作者:[Manuel Jose][a]
|
||||||
|
译者:[geekpi](https://github.com/geekpi)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.techdrivein.com/2016/09/taskwarrior-command-line-todo-app-linux.html?utm_source=feedburner&utm_medium=feed&utm_campaign=Feed%3A+techdrivein+%28Tech+Drive-in%29
|
||||||
|
[1]: https://taskwarrior.org/docs/
|
||||||
|
[2]: https://taskwarrior.org/download/
|
||||||
|
[3]: https://taskwarrior.org/news/news.20160225.html
|
||||||
|
[4]: http://www.techdrivein.com/search/label/Terminal
|
||||||
|
|
||||||
|
|
@ -0,0 +1,78 @@
|
|||||||
|
如何用四个简单的步骤加速 LibreOffice
|
||||||
|
====
|
||||||
|
|
||||||
|
![](https://maketecheasier-2d0f.kxcdn.com/assets/uploads/2016/08/speed-up-libreoffice-featured-2.jpg)
|
||||||
|
|
||||||
|
对于许多许多开源软件的粉丝和支持者来说,LibreOffice 是 Microsoft Office 最好的替代品,在最近的一些发布版本中可以看到它明显有了巨大的改进。然而,初始启动的体验仍然距离期望有所距离。有一些方法可以缩短 LibreOffice 的启动时间并改善它的整体性能。
|
||||||
|
|
||||||
|
在下面的段落里,我将会展示一些实用性的步骤,你可以通过它们来改善 LibreOffice 的加载时间和响应能力。
|
||||||
|
|
||||||
|
### 1. 增加每个对象和图像缓存的内存占用
|
||||||
|
|
||||||
|
这将可以通过分配更多的内存资源给图像缓存和对象来加快程序的加载时间。
|
||||||
|
|
||||||
|
1. 启动 LibreOffice Writer (或者 Calc)。
|
||||||
|
2. 点击菜单栏上的 “工具 -> 选项” 或者按键盘上的快捷键“Alt + F12”。
|
||||||
|
3. 点击 LibreOffice 下面的“内存”然后增加“用于 LibreOffice” 到 128MB。
|
||||||
|
4. 同样的增加“每个对象的内存占用”到 20MB。
|
||||||
|
5. 点击确定来保存你的修改。
|
||||||
|
|
||||||
|
![](https://maketecheasier-2d0f.kxcdn.com/assets/uploads/2016/08/speed-up-libreoffice-step-1.png)
|
||||||
|
|
||||||
|
注意:你可以根据自己机器的性能把数值设置得比建议值的高一些或低一些。最好通过亲自体验来看看什么值能够让机器达到最佳性能。
|
||||||
|
|
||||||
|
### 2.启用 LibreOffice 的快速启动器(QuickStarter)
|
||||||
|
|
||||||
|
如果你的机器上有足够大的内存,比如 4GB 或者更大,你可以启用“系统托盘快速启动器”,从而让 LibreOffice 的一部分保持在内存中,在打开新文件时能够快速反应。
|
||||||
|
|
||||||
|
在启用这个选择以后,你会清楚的看到在打开新文件时它的性能有了很大的提高。
|
||||||
|
|
||||||
|
1. 通过点击“工具 -> 选项”来打开选项对话框。
|
||||||
|
2. 在 “LibreOffice” 下面的侧边栏选择“内存”。
|
||||||
|
3. 勾选“启用系统托盘快速启动器”复选框。
|
||||||
|
4. 点击“确定”来保存修改。
|
||||||
|
|
||||||
|
![](https://maketecheasier-2d0f.kxcdn.com/assets/uploads/2016/08/speed-up-libreoffice-2.png)
|
||||||
|
|
||||||
|
一旦这个选项启用以后,你将会在你的系统托盘看到 LibreOffice 图标,以及可以打开任何类型的文件的选项。
|
||||||
|
|
||||||
|
### 3. 禁用 Java 运行环境
|
||||||
|
|
||||||
|
另一个加快 LibreOffice 加载时间和响应能力的简单方法是禁用 Java。
|
||||||
|
|
||||||
|
1. 同时按下“Alt + F12”打开选项对话框。
|
||||||
|
2. 在侧边栏里,选择“Libreoffice”,然后选择“高级”。
|
||||||
|
3. 取消勾选“使用 Java 运行环境”选项。
|
||||||
|
4. 点击“确定”来关闭对话框。
|
||||||
|
|
||||||
|
![](https://maketecheasier-2d0f.kxcdn.com/assets/uploads/2016/08/speed-up-libreoffice-3.png)
|
||||||
|
|
||||||
|
如果你只使用 Writer 和 Calc,那么关闭 Java 不会影响你正常使用,但如果你需要使用 LibreOffice Base 和一些其他的特性,那么你可能需要重新启用它。在那种情况,将会弹出一个框询问你是否希望再次打开它。
|
||||||
|
|
||||||
|
### 4. 减少使用撤销步骤
|
||||||
|
|
||||||
|
默认情况下,LibreOffice 允许你撤销一个文件的多达 100 个改变。绝大多数用户不需要这么多,所以在内存中保留这么多撤销步骤是对资源的巨大浪费。
|
||||||
|
|
||||||
|
我建议减少撤销步骤到 20 次以下来为其他东西释放内存,但是这个部分需要根据你自己的需求来确定。
|
||||||
|
|
||||||
|
1. 通过点击 “工具 -> 选项”来打开选项对话框。
|
||||||
|
2. 在 “LibreOffice” 下面的侧边栏,选择“内存”。
|
||||||
|
3. 在“撤销”下面把步骤数目改成最适合你的值。
|
||||||
|
4. 点击“确定”来保存修改。
|
||||||
|
|
||||||
|
![](https://maketecheasier-2d0f.kxcdn.com/assets/uploads/2016/08/speed-up-libreoffice-5.png)
|
||||||
|
|
||||||
|
假如你这些技巧为加速你的 LibreOffice 套件的加载时间提供了帮助,请在评论里告诉我们。同样,请分享你知道的任何其他技巧来给其他人带来帮助。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://www.maketecheasier.com/speed-up-libreoffice/
|
||||||
|
|
||||||
|
|
||||||
|
作者:[Ayo Isaiah][a]
|
||||||
|
译者:[ucasFL](https://github.com/ucasFL)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://www.maketecheasier.com/author/ayoisaiah/
|
@ -0,0 +1,63 @@
|
|||||||
|
是时候合并 LibreOffice 和 OpenOffice 了
|
||||||
|
==========
|
||||||
|
|
||||||
|
![](http://tr2.cbsistatic.com/hub/i/2016/09/14/2e91089b-7ebd-4579-bf8f-74c34d1a94ce/e7e9c8dd481d8e068f2934c644788928/openofficedeathhero.jpg)
|
||||||
|
|
||||||
|
先说下 OpenOffice。可能你已经无数次地看到说 Apache OpenOffice 即将结束。上一个稳定版本是 4.1.2 (发布于 2015 年 10 月),而最近的一个严重安全漏洞用了一个月才打上补丁。编码人员的缺乏使得开发如爬行前进一般。然后,可能是最糟糕的消息了:这个项目建议用户切换到 [MS Office](https://products.office.com/)(或 [LibreOffice](https://www.libreoffice.org/download/))。
|
||||||
|
|
||||||
|
丧钟为谁而鸣?丧钟为你而鸣,OpenOffice。
|
||||||
|
|
||||||
|
我想说些可能会惹恼一些人的话。你准备好了吗?
|
||||||
|
|
||||||
|
OpenOffice 的终止对开源和用户来说都将是件好事。
|
||||||
|
|
||||||
|
让我解释一下。
|
||||||
|
|
||||||
|
### 一个分支统治所有
|
||||||
|
|
||||||
|
当 LibreOffice 从 OpenOffice 分支出来后,我们看到了另一个情况:分支不只在原始基础上进行改进,而且大幅超越了它。LibreOffice 一举成功。所有之前预装 OpenOffice 的 Linux 发行版都迁移到了这个新项目。LibreOffice 从起跑线突然冲出,并迅速迈出了一大步。更新以极快的速度发布,改善内容丰富而重要。
|
||||||
|
|
||||||
|
不久后,OpenOffice 就被开源社区丢在了脑后。当 2011 年 Oracle 决定终止这个项目并把代码捐赠给 Apache 项目时,这种情况自然更加恶化了。从此 OpenOffice 艰难前进,然后把我们就看到了现在这种局面:一个生机勃勃的 LibreOffice 和一个艰难的、缓慢的 OpenOffice。
|
||||||
|
|
||||||
|
但我认为在这个相当昏暗的隧道末尾有一丝曙光。
|
||||||
|
|
||||||
|
### 合并他们
|
||||||
|
|
||||||
|
这听起来可能很疯狂,但我认为是时候把 LibreOffice 和 OpenOffice 合二为一了。是的,我知道很可能有政治考虑和自尊意识,但我认为合并成一个会更好。合并的好处很多。我首先能想到的是:
|
||||||
|
|
||||||
|
- 把 MS Office 过滤器整合起来:OpenOffice 在更好地导入某些 MS Office 文件上功能很强(而众所周知 LibreOffice 正在改进,但时好时坏)
|
||||||
|
- LibreOffice 有更多开发者:尽管 OpenOffice 的开发者数量不多,但也无疑会增加到合并后的项目。
|
||||||
|
- 结束混乱:很多用户以为 OpenOffice 和 LibreOffice 是同一个东西。有些甚至不知道 LibreOffice 存在。这将终结那些混乱。
|
||||||
|
- 合并他们的用户量:OpenOffice 和 LibreOffice 各自拥有大量用户。联合后,他们将是个巨大的力量。
|
||||||
|
|
||||||
|
### 宝贵机遇
|
||||||
|
|
||||||
|
OpenOffice 的终止实际上会成为整个开源办公套件行业的一个宝贵机遇。为什么?我想表明有些东西我认为已经需要很久了。如果 OpenOffice 和 LibreOffice 集中他们的力量,比较他们的代码并合并,他们之后就可以做一些更必要的改进工作,不仅是整体的内部工作,也包括界面。
|
||||||
|
|
||||||
|
我们得面对现实,LibreOffice 和(相关的) OpenOffice 的用户界面都是过时的。当我安装 LibreOffice 5.2.1.2 时,工具栏绝对是个灾难(见下图)。
|
||||||
|
|
||||||
|
![](http://tr2.cbsistatic.com/hub/i/2016/09/14/cc5250df-48cd-40e3-a083-34250511ffab/c5ac8eb1e2cb12224690a6a3525999f0/openofficea.jpg)
|
||||||
|
|
||||||
|
*LibreOffice 默认工具栏显示*
|
||||||
|
|
||||||
|
尽管我支持和关心(并且日常使用)LibreOffice,但事实已经再清楚不过了,界面需要完全重写。我们正在使用的是 90 年代末/ 2000 年初的复古界面,它必须得改变了。当新用户第一次打开 LibreOffice 时,他们会被淹没在大量按钮、图标和工具栏中。Ubuntu Unity 的平视显示(Head up Display,简称 HUD)帮助解决了这个问题,但那并不适用于其它桌面和发行版。当然,有经验的用户知道在哪里找什么(甚至定制工具栏以满足特殊的需要),但对新用户或普通用户,那种界面是个噩梦。现在是做出改变的一个好时机。引入 OpenOffice 最后残留的开发者并让他们加入到改善界面的战斗中。借助于整合 OpenOffice 额外的导入过滤器和现代化的界面,LibreOffice 终能在家庭和办公桌面上都引起一些轰动。
|
||||||
|
|
||||||
|
### 这会真的发生吗?
|
||||||
|
|
||||||
|
这需要发生。但是会发生吗?我不知道。但即使掌权者决定用户界面并不需要重组(这会是个失误),合并 OpenOffice 仍是前进的一大步。合并两者将带来开发的更专注,更好的推广,公众更少的困惑。
|
||||||
|
|
||||||
|
我知道这可能看起来有悖于开源的核心精神,但合并 LibreOffice 和 OpenOffice 将能联合两者的力量,而且可能会摆脱弱点。
|
||||||
|
|
||||||
|
在我看来,这是双赢的。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.techrepublic.com/article/its-time-to-make-libreoffice-and-openoffice-one-again/
|
||||||
|
|
||||||
|
作者:[Jack Wallen][a]
|
||||||
|
译者:[bianjp](https://github.com/bianjp)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.techrepublic.com/search/?a=jack%2Bwallen
|
@ -0,0 +1,364 @@
|
|||||||
|
在 Ubuntu 16.04 上安装和使用服务器监控报警系统 Shinken
|
||||||
|
=====
|
||||||
|
|
||||||
|
Shinken 是一个用 Python 实现的开源的主机和网络监控框架,并与 Nagios like 兼容,它可以运行在所有支持 Python 程序的操作系统上,比如说 Linux、Unix 和 Windows,Shinken 是 Jean Gabes 为了验证一个新的 Nagios 架构思路而编写,但是这个想法被 Nagios 的作者拒绝后成为了一个独立的网络系统监视软件,并保持了与 Nagios 的兼容。
|
||||||
|
|
||||||
|
在这篇教程中,我将会描述如何从源代码编译安装 Shinken 和向监视系统中添加一台 Linux 主机。我将会以 Ubuntu 16.04 Xenial Xerus 操作系统来作为 Shinken 服务器和所监控的主机。
|
||||||
|
|
||||||
|
### 第一步 安装 Shinken 服务器
|
||||||
|
|
||||||
|
Shinken 是一个 Python 框架,我们可以通过 `pip` 安装或者从源码来安装它,在这一步中,我们将用源代码编译安装 Shinken。
|
||||||
|
|
||||||
|
在我们开始安装 Shinken 之前还需要完成几个步骤。
|
||||||
|
|
||||||
|
安装一些新的 Python 软件包并创建一个名为 `shinken` 的系统用户:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt-get install python-setuptools python-pip python-pycurl
|
||||||
|
useradd -m -s /bin/bash shinken
|
||||||
|
```
|
||||||
|
|
||||||
|
从 GitHub 仓库下载 Shinken 源代码:
|
||||||
|
|
||||||
|
```
|
||||||
|
git clone https://github.com/naparuba/shinken.git
|
||||||
|
cd shinken/
|
||||||
|
```
|
||||||
|
|
||||||
|
然后用以下命令安装 Shinken:
|
||||||
|
|
||||||
|
```
|
||||||
|
git checkout 2.4.3
|
||||||
|
python setup.py install
|
||||||
|
```
|
||||||
|
|
||||||
|
然后,为了得到更好的效果,我们还需要从 Ubuntu 软件库中安装 `python-cherrypy3` 软件包:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt-get install python-cherrypy3
|
||||||
|
```
|
||||||
|
|
||||||
|
到这里,Shinken 已经成功安装,接下来我们将 Shinken 添加到系统启动项并且启动它:
|
||||||
|
|
||||||
|
```
|
||||||
|
update-rc.d shinken defaults
|
||||||
|
systemctl start shinken
|
||||||
|
```
|
||||||
|
|
||||||
|
### 第二步 安装 Shinken Webui2
|
||||||
|
|
||||||
|
Webui2 是 Shinken 的 Web 界面(在 shinken.io 可以找到)。最简单的安装 Shinken webui2 的方法是使用shinken CLI 命令(必须作为 `shinken` 用户执行)。
|
||||||
|
|
||||||
|
切换到 shinken 用户:
|
||||||
|
|
||||||
|
```
|
||||||
|
su - shinken
|
||||||
|
```
|
||||||
|
|
||||||
|
初始化 shiken 配置文件,下面的命令将会创建一个新的配置文件 `.shinken.ini` :
|
||||||
|
|
||||||
|
```
|
||||||
|
shinken --init
|
||||||
|
```
|
||||||
|
|
||||||
|
接下来用 shinken CLI 命令来安装 `webui2`:
|
||||||
|
|
||||||
|
```
|
||||||
|
shinken install webui2
|
||||||
|
```
|
||||||
|
|
||||||
|
![](https://www.howtoforge.com/images/server-monitoring-with-shinken-on-ubuntu-16-04/6.png)
|
||||||
|
|
||||||
|
至此 webui2 已经安装好,但是我们还需要安装 MongoDB 和用 `pip` 来安装另一个 Python 软件包。在 root 下运行如下命令:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt-get install mongodb
|
||||||
|
pip install pymongo>=3.0.3 requests arrow bottle==0.12.8
|
||||||
|
```
|
||||||
|
|
||||||
|
接下来,切换到 shinken 目录下并且通过编辑 `broker-master.cfg` 文件来添加这个新的 webui2 模块:
|
||||||
|
|
||||||
|
```
|
||||||
|
cd /etc/shinken/brokers/
|
||||||
|
vim broker-master.cfg
|
||||||
|
```
|
||||||
|
|
||||||
|
在第 40 行添加一个模块选项:
|
||||||
|
|
||||||
|
```
|
||||||
|
modules webui2
|
||||||
|
```
|
||||||
|
|
||||||
|
保存文件并且退出编辑器。
|
||||||
|
|
||||||
|
现在进入 `contacts` 目录下编辑 `admin.cfg` 来进行管理配置。
|
||||||
|
|
||||||
|
```
|
||||||
|
cd /etc/shinken/contacts/
|
||||||
|
vim admin.cfg
|
||||||
|
```
|
||||||
|
|
||||||
|
按照如下修改:
|
||||||
|
|
||||||
|
```
|
||||||
|
contact_name admin # Username 'admin'
|
||||||
|
password yourpass # Pass 'mypass'
|
||||||
|
```
|
||||||
|
|
||||||
|
保存和退出。
|
||||||
|
|
||||||
|
### 第三步 安装 Nagios 插件和 Shinken 软件包
|
||||||
|
|
||||||
|
在这一步中,我们将安装 Nagios 插件和一些 Perl 模块。然后从 shinken.io 安装其他的软件包来实现监视。
|
||||||
|
|
||||||
|
安装 Nagios 插件和安装 Perl 模块所需要的 `cpanminus`:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt-get install nagios-plugins* cpanminus
|
||||||
|
```
|
||||||
|
|
||||||
|
用 `cpanm` 命令来安装 Perl 模块。
|
||||||
|
|
||||||
|
```
|
||||||
|
cpanm Net::SNMP
|
||||||
|
cpanm Time::HiRes
|
||||||
|
cpanm DBI
|
||||||
|
```
|
||||||
|
|
||||||
|
现在我们创建一个 `utils.pm` 文件的链接到 shinken 的目录,并且为 `Log_File_Health` 创建了一个新的日志目录 。
|
||||||
|
|
||||||
|
```
|
||||||
|
chmod u+s /usr/lib/nagios/plugins/check_icmp
|
||||||
|
ln -s /usr/lib/nagios/plugins/utils.pm /var/lib/shinken/libexec/
|
||||||
|
mkdir -p /var/log/rhosts/
|
||||||
|
touch /var/log/rhosts/remote-hosts.log
|
||||||
|
```
|
||||||
|
|
||||||
|
然后,从 shinken.io 安装 shinken 软件包 `ssh` 和 `linux-snmp` 来监视 SSH 和 SNMP :
|
||||||
|
|
||||||
|
```
|
||||||
|
su - shinken
|
||||||
|
shinken install ssh
|
||||||
|
shinken install linux-snmp
|
||||||
|
```
|
||||||
|
|
||||||
|
### 第四步 添加一个 Linux 主机 host-one
|
||||||
|
|
||||||
|
我们将添加一个新的将被监控的 Linux 主机,IP 地址为 192.168.1.121,主机名为 host-one 的 Ubuntu 16.04。
|
||||||
|
|
||||||
|
连接到 host-one 主机:
|
||||||
|
|
||||||
|
```
|
||||||
|
ssh host1@192.168.1.121
|
||||||
|
```
|
||||||
|
|
||||||
|
从 Ubuntu 软件库中安装 snmp 和snmpd 软件包:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt-get install snmp snmpd
|
||||||
|
```
|
||||||
|
|
||||||
|
然后,用 `vim` 编辑 `snmpd.conf` 配置文件:
|
||||||
|
|
||||||
|
```
|
||||||
|
vim /etc/snmp/snmpd.conf
|
||||||
|
```
|
||||||
|
|
||||||
|
注释掉第 15 行并取消注释第 17 行:
|
||||||
|
|
||||||
|
```
|
||||||
|
#agentAddress udp:127.0.0.1:161
|
||||||
|
agentAddress udp:161,udp6:[::1]:161
|
||||||
|
```
|
||||||
|
|
||||||
|
注释掉第 51 和 53 行,然后加一行新的配置,如下:
|
||||||
|
|
||||||
|
```
|
||||||
|
#rocommunity mypass default -V systemonly
|
||||||
|
#rocommunity6 mypass default -V systemonly
|
||||||
|
|
||||||
|
rocommunity mypass
|
||||||
|
```
|
||||||
|
|
||||||
|
保存并退出。
|
||||||
|
|
||||||
|
现在用 `systemctl` 命令来启动 `snmpd` 服务:
|
||||||
|
|
||||||
|
```
|
||||||
|
systemctl start snmpd
|
||||||
|
```
|
||||||
|
|
||||||
|
在 shinken 服务器上通过在 `hosts` 文件夹下创建新的文件来定义一个新的主机:
|
||||||
|
|
||||||
|
```
|
||||||
|
cd /etc/shinken/hosts/
|
||||||
|
vim host-one.cfg
|
||||||
|
```
|
||||||
|
|
||||||
|
粘贴如下配置信息:
|
||||||
|
|
||||||
|
```
|
||||||
|
define host{
|
||||||
|
use generic-host,linux-snmp,ssh
|
||||||
|
contact_groups admins
|
||||||
|
host_name host-one
|
||||||
|
address 192.168.1.121
|
||||||
|
_SNMPCOMMUNITY mypass # SNMP Pass Config on snmpd.conf
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
保存并退出。
|
||||||
|
|
||||||
|
在 shinken 服务器上编辑 SNMP 配置文件。
|
||||||
|
|
||||||
|
```
|
||||||
|
vim /etc/shinken/resource.d/snmp.cfg
|
||||||
|
```
|
||||||
|
|
||||||
|
将 `public` 改为 `mypass` -必须和你在客户端 `snmpd` 配置文件中使用的密码相同:
|
||||||
|
|
||||||
|
```
|
||||||
|
$SNMPCOMMUNITYREAD$=mypass
|
||||||
|
```
|
||||||
|
|
||||||
|
保存并退出。
|
||||||
|
|
||||||
|
现在将服务端和客户端都重启:
|
||||||
|
|
||||||
|
```
|
||||||
|
reboot
|
||||||
|
```
|
||||||
|
|
||||||
|
现在 Linux 主机已经被成功地添加到 shinken 服务器中了。
|
||||||
|
|
||||||
|
### 第五步 访问 Shinken Webui2
|
||||||
|
|
||||||
|
在端口 7677 访问 Shinken webui2 (将 URL 中的 IP 替换成你自己的 IP 地址):
|
||||||
|
|
||||||
|
```
|
||||||
|
http://192.168.1.120:7767
|
||||||
|
```
|
||||||
|
|
||||||
|
用管理员用户和密码登录(你在 admin.cfg 文件中设置的)
|
||||||
|
|
||||||
|
![](https://www.howtoforge.com/images/server-monitoring-with-shinken-on-ubuntu-16-04/1.png)
|
||||||
|
|
||||||
|
Webui2 中的 Shinken 面板:
|
||||||
|
|
||||||
|
![](https://www.howtoforge.com/images/server-monitoring-with-shinken-on-ubuntu-16-04/2.png)
|
||||||
|
|
||||||
|
我们的两个服务器正在被 Shinken 监控:
|
||||||
|
|
||||||
|
![](https://www.howtoforge.com/images/server-monitoring-with-shinken-on-ubuntu-16-04/3.png)
|
||||||
|
|
||||||
|
列出所有被 linux-snmp 监控的服务:
|
||||||
|
|
||||||
|
![](https://www.howtoforge.com/images/server-monitoring-with-shinken-on-ubuntu-16-04/4.png)
|
||||||
|
|
||||||
|
所有主机和服务的状态信息:
|
||||||
|
|
||||||
|
![](https://www.howtoforge.com/images/server-monitoring-with-shinken-on-ubuntu-16-04/5.png)
|
||||||
|
|
||||||
|
### 第6步 Shinken 的常见问题
|
||||||
|
|
||||||
|
#### NTP 服务器相关的问题
|
||||||
|
|
||||||
|
当你得到如下的 NTP 错误提示
|
||||||
|
|
||||||
|
```
|
||||||
|
TimeSync - CRITICAL ( NTP CRITICAL: No response from the NTP server)
|
||||||
|
TimeSync - CRITICAL ( NTP CRITICAL: Offset unknown )
|
||||||
|
```
|
||||||
|
|
||||||
|
为了解决这个问题,在所有 Linux 主机上安装 ntp。
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt-get install ntp ntpdate
|
||||||
|
```
|
||||||
|
|
||||||
|
编辑 ntp 配置文件:
|
||||||
|
|
||||||
|
```
|
||||||
|
vim /etc/ntp.conf
|
||||||
|
```
|
||||||
|
|
||||||
|
注释掉所有 pools 并替换为:
|
||||||
|
|
||||||
|
```
|
||||||
|
#pool 0.ubuntu.pool.ntp.org iburst
|
||||||
|
#pool 1.ubuntu.pool.ntp.org iburst
|
||||||
|
#pool 2.ubuntu.pool.ntp.org iburst
|
||||||
|
#pool 3.ubuntu.pool.ntp.org iburst
|
||||||
|
|
||||||
|
pool 0.id.pool.ntp.org
|
||||||
|
pool 1.asia.pool.ntp.org
|
||||||
|
pool 0.asia.pool.ntp.org
|
||||||
|
```
|
||||||
|
|
||||||
|
然后,在新的一行添加如下限制规则:
|
||||||
|
|
||||||
|
```
|
||||||
|
# Local users may interrogate the ntp server more closely.
|
||||||
|
restrict 127.0.0.1
|
||||||
|
restrict 192.168.1.120 #shinken server IP address
|
||||||
|
restrict ::1
|
||||||
|
NOTE: 192.168.1.120 is the Shinken server IP address.
|
||||||
|
```
|
||||||
|
|
||||||
|
保存并退出。
|
||||||
|
|
||||||
|
启动 ntp 并且检查 Shinken 面板。
|
||||||
|
|
||||||
|
```
|
||||||
|
ntpd
|
||||||
|
```
|
||||||
|
|
||||||
|
#### check_netint.pl Not Found 问题
|
||||||
|
|
||||||
|
从 github 仓库下载源代码到 shinken 的库目录下:
|
||||||
|
|
||||||
|
```
|
||||||
|
cd /var/lib/shinken/libexec/
|
||||||
|
wget https://raw.githubusercontent.com/Sysnove/shinken-plugins/master/check_netint.pl
|
||||||
|
chmod +x check_netint.pl
|
||||||
|
chown shinken:shinken check_netint.pl
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 网络占用的问题
|
||||||
|
|
||||||
|
这是错误信息:
|
||||||
|
|
||||||
|
```
|
||||||
|
ERROR : Unknown interface eth\d+
|
||||||
|
```
|
||||||
|
|
||||||
|
检查你的网络接口并且编辑 `linux-snmp` 模版。
|
||||||
|
|
||||||
|
在我的 Ununtu 服务器,网卡是 “enp0s8”,而不是 eth0,所以我遇到了这个错误。
|
||||||
|
|
||||||
|
`vim` 编辑 `linux-snmp` 模版:
|
||||||
|
|
||||||
|
```
|
||||||
|
vim /etc/shinken/packs/linux-snmp/templates.cfg
|
||||||
|
```
|
||||||
|
|
||||||
|
在第 24 行添加网络接口信息:
|
||||||
|
|
||||||
|
```
|
||||||
|
_NET_IFACES eth\d+|em\d+|enp0s8
|
||||||
|
```
|
||||||
|
|
||||||
|
保存并退出。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://www.howtoforge.com/tutorial/server-monitoring-with-shinken-on-ubuntu-16-04/
|
||||||
|
|
||||||
|
作者:[Muhammad Arul][a]
|
||||||
|
译者:[LinuxBars](https://github.com/LinuxBars)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://www.howtoforge.com/tutorial/server-monitoring-with-shinken-on-ubuntu-16-04/
|
||||||
|
Save and exit.
|
@ -0,0 +1,140 @@
|
|||||||
|
Googler:现在可以 Linux 终端下进行 Google 搜索了!
|
||||||
|
============================================
|
||||||
|
|
||||||
|
![](https://itsfoss.com/wp-content/uploads/2016/09/google-from-linux-terminal.jpg)
|
||||||
|
|
||||||
|
一个小问题:你每天做什么事?当然了,好多事情,但是我可以指出一件事,你几乎每天(如果不是每天)都会用 Google 搜索,我说的对吗?(LCTT 译注:Google 是啥?/cry )
|
||||||
|
|
||||||
|
现在,如果你是一位 Linux 用户(我猜你也是),这里有另外一个问题:如果你甚至不用离开终端就可以进行 Google 搜索那岂不是相当棒?甚至不用打开一个浏览器窗口?
|
||||||
|
|
||||||
|
如果你是一位类 [*nix][7] 系统的狂热者而且也是喜欢终端界面的人,我知道你的答案是肯定的,而且我认为,接下来你也将喜欢上我今天将要介绍的这个漂亮的小工具。它被称做 Googler。
|
||||||
|
|
||||||
|
### Googler:在你 linux 终端下的 google
|
||||||
|
|
||||||
|
Googler 是一个简单的命令行工具,它用于直接在命令行窗口中进行 google 搜索,Googler 主要支持三种类型的 Google 搜索:
|
||||||
|
|
||||||
|
- Google 搜索:简单的 Google 搜索,和在 Google 主页搜索是等效的。
|
||||||
|
- Google 新闻搜索:Google 新闻搜索,和在 Google News 中的搜索一样。
|
||||||
|
- Google 站点搜索:Google 从一个特定的网站搜索结果。
|
||||||
|
|
||||||
|
Googler 用标题、链接和网页摘要来显示搜索结果。搜索出来的结果可以仅通过两个按键就可以在浏览器里面直接打开。
|
||||||
|
|
||||||
|
![](https://itsfoss.com/wp-content/uploads/2016/09/googler-1.png)
|
||||||
|
|
||||||
|
### 在 Ubuntu 下安装 Googler
|
||||||
|
|
||||||
|
先让我们进行软件的安装。
|
||||||
|
|
||||||
|
首先确保你的 python 版本大于等于 3.3,可以用以下命令查看。
|
||||||
|
|
||||||
|
```
|
||||||
|
python3 --version
|
||||||
|
```
|
||||||
|
|
||||||
|
如果不是的话,就更新一下。Googler 要求 python 版本 3.3 及以上运行。
|
||||||
|
|
||||||
|
虽然 Googler 现在还不能在 Ununtu 的软件库中找到,我们可以很容易地从 GitHub 仓库中安装它。我们需要做的就是运行以下命令:
|
||||||
|
|
||||||
|
```
|
||||||
|
cd /tmp
|
||||||
|
git clone https://github.com/jarun/googler.git
|
||||||
|
cd googler
|
||||||
|
sudo make install
|
||||||
|
cd auto-completion/bash/
|
||||||
|
sudo cp googler-completion.bash /etc/bash_completion.d/
|
||||||
|
```
|
||||||
|
|
||||||
|
这样 Googler 就带着命令自动完成特性安装完毕了。
|
||||||
|
|
||||||
|
### 特点 & 基本用法
|
||||||
|
|
||||||
|
如果我们快速浏览它所有的特点,我们会发现 Googler 实际上是一个十分强大的工具,它的一些主要特点就是:
|
||||||
|
|
||||||
|
#### 交互界面
|
||||||
|
|
||||||
|
在终端下运行以下命令:
|
||||||
|
|
||||||
|
```
|
||||||
|
googler
|
||||||
|
```
|
||||||
|
|
||||||
|
交互界面就会被打开,Googler 的开发者 [Arun Prakash Jana][1] 称之为全向提示符(omniprompt),你可以输入 `?` 去寻找可用的命令参数:
|
||||||
|
|
||||||
|
![](https://itsfoss.com/wp-content/uploads/2016/09/googler-2.png)
|
||||||
|
|
||||||
|
在提示符处,输入任何搜索词汇关键字去开始搜索,然后你可以输入`n`或者`p`导航到搜索结果的后一页和前一页。
|
||||||
|
|
||||||
|
要在浏览器窗口中打开搜索结果,直接输入搜索结果的编号,或者你可以输入 `o` 命令来打开这个搜索网页。
|
||||||
|
|
||||||
|
#### 新闻搜索
|
||||||
|
|
||||||
|
如果你想去搜索新闻,直接以`N`参数启动 Googler:
|
||||||
|
|
||||||
|
```
|
||||||
|
googler -N
|
||||||
|
```
|
||||||
|
|
||||||
|
随后的搜索将会从 Google News 抓取结果。
|
||||||
|
|
||||||
|
#### 站点搜索
|
||||||
|
|
||||||
|
如果你想从某个特定的站点进行搜索,以`w 域名`参数启动 Googler:
|
||||||
|
|
||||||
|
```
|
||||||
|
googler -w itsfoss.com
|
||||||
|
```
|
||||||
|
|
||||||
|
随后的搜索会只从这个博客中抓取结果!
|
||||||
|
|
||||||
|
#### 手册页
|
||||||
|
|
||||||
|
运行以下命令去查看 Googler 的带着各种用例的手册页:
|
||||||
|
|
||||||
|
```
|
||||||
|
man googler
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 指定国家/地区的 Google 搜索引擎
|
||||||
|
|
||||||
|
```
|
||||||
|
googler -c in "hello world"
|
||||||
|
```
|
||||||
|
|
||||||
|
上面的示例命令将会开始从 Google 的印度域名搜索结果(in 代表印度)
|
||||||
|
|
||||||
|
还支持:
|
||||||
|
|
||||||
|
- 通过时间和语言偏好来过滤搜索结果
|
||||||
|
- 支持 Google 查询关键字,例如:`site:example.com` 或者 `filetype:pdf` 等等
|
||||||
|
- 支持 HTTPS 代理
|
||||||
|
- Shell 命令自动补全
|
||||||
|
- 禁用自动拼写纠正
|
||||||
|
|
||||||
|
这里还有更多特性。你可以用 Googler 去满足你的需要。
|
||||||
|
|
||||||
|
Googler 也可以和一些基于文本的浏览器整合在一起(例如:[elinks][2]、[links][3]、[lynx][4]、w3m 等),所以你甚至都不用离开终端去浏览网页。在 [Googler 的 GitHub 项目页][5]可以找到指导。
|
||||||
|
|
||||||
|
如果你想看一下 Googler 不同的特性的视频演示,方便的话你可以查看 GitHub 项目页附带的终端记录演示页: [jarun/googler v2.7 quick demo][6]。
|
||||||
|
|
||||||
|
### 对于 Googler 的看法?
|
||||||
|
|
||||||
|
尽管 googler 可能并不是对每个人都是必要和渴望的,对于一些不想打开浏览器进行 google 搜索或者就是想泡在终端窗口里面的人来说,这是一个很棒的工具。你认为呢?
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://itsfoss.com/review-googler-linux/
|
||||||
|
|
||||||
|
作者:[Munif Tanjim][a]
|
||||||
|
译者:[LinuxBars](https://github.com/LinuxBars)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://itsfoss.com/author/munif/
|
||||||
|
[1]: https://github.com/jarun
|
||||||
|
[2]: http://elinks.or.cz/
|
||||||
|
[3]: http://links.twibright.com/
|
||||||
|
[4]: http://lynx.browser.org/
|
||||||
|
[5]: https://github.com/jarun/googler#faq
|
||||||
|
[6]: https://asciinema.org/a/85019
|
||||||
|
[7]: https://en.wikipedia.org/wiki/Unix-like
|
@ -0,0 +1,90 @@
|
|||||||
|
你该选择 openSUSE 的五大理由
|
||||||
|
===============
|
||||||
|
|
||||||
|
[![Reasons why you should use OpenSUSE](https://itsfoss.com/wp-content/uploads/2016/09/why-opensuse-is-best.jpg)](https://itsfoss.com/wp-content/uploads/2016/09/why-opensuse-is-best.jpg)
|
||||||
|
|
||||||
|
多数的的桌面 Linux 用户都会选择三种发行版本:Debian/Ubuntu、Fedora 或者 Arch Linux。但是今天,我将给出你需要使用 openSUSE 的五大理由。
|
||||||
|
|
||||||
|
相比其他的 Linux 发行版,我总能在 [openSUSE](https://www.opensuse.org/) 上看到一些令人耳目一新的东西。我说不太好,但它总是如此的闪亮和个性鲜明。这绿色的大蜥蜴是看起来如此的令人惊叹!但这并不是 openSUSE **即便不是最好也是要比其它发行版更好**的原因!
|
||||||
|
|
||||||
|
请别误解我。我在各种场合用过许多不同的 Linux 发行版,同时也很敬佩在这些发行版背后默默工作的开发者,是他们让计算变成一件快乐的事情。但是 openSUSE 一直让人感觉,嗯,令人崇敬——你是不是也这样觉得?
|
||||||
|
|
||||||
|
### openSUSE 比其他 Linux 发行版要好的五大理由
|
||||||
|
|
||||||
|
你是不是认为我在说 openSUSE 是最好的 Linux 发行版?不,我并不是要表达这个意思。其实没有任何一个 Linux 发行版是最好的。它真的可以满足你寻找 “灵魂伴侣” 的需求。
|
||||||
|
|
||||||
|
但在这里,我准备给大家说说,openSUSE 比其他发行版做得要好的五件事。如下:
|
||||||
|
|
||||||
|
#### #1 社区规则
|
||||||
|
|
||||||
|
openSUSE 是一个典型的社区驱动型项目。我经常看到很多用户在升级后抱怨开发人员改变了他们喜欢的发行版。但在 openSUSE 不是这样,openSUSE 是纯社区驱动的项目,并且任何时候都朝着用户所希望的方向发展。
|
||||||
|
|
||||||
|
#### #2 系统的健壮性
|
||||||
|
|
||||||
|
另外一个是操作系统的集成程度。我可以在同一个 openSUSE 系统上安装所有的[最好的 Linux 桌面环境](https://itsfoss.com/best-linux-desktop-environments/),而在 Ubuntu 上则因为系统的稳定性,坚决不允许用户这样做。而这恰好体现了一个系统的健壮程度。因此,对于那些喜欢自己动手完成每一件事的用户,openSUSE 还是很诱人的。
|
||||||
|
|
||||||
|
#### #3 易于安装软件
|
||||||
|
|
||||||
|
在 Linux 的世界里有很多非常好用的包管理工具。从 Debian 的 apt-get 到 [Fedora](https://itsfoss.com/fedora-24-review/) 的 DNF,它们无不吸引着用户,而且在这些发行版成为流行版本的过程中扮演着重要角色。
|
||||||
|
|
||||||
|
openSUSE 同样有一个将软件传递到桌面的好方法。[software.opensuse.org](https://software.opensuse.org/421/en) 是一个 Web 界面,你可以用它从仓库中获取安装软件。你所需要做的就是打开这个链接 (当然,是在 openSUSE 系统上),在搜索框中输入你想要的软件,点击“直接安装”即可。就是这么简单,不是吗?
|
||||||
|
|
||||||
|
听起来就像是在使用 Google 商店一样,是吗?
|
||||||
|
|
||||||
|
#### #4 YAST
|
||||||
|
|
||||||
|
毫不夸张的说,[YaST](https://en.opensuse.org/Portal:YaST) (LCTT 译注: YaST 是 openSUSE 和 SUSE Linux 企业版的安装和配置工具) 绝对是世界上有史以来**操作系统**上最好的控制中心。并且毫无疑问地,你可以使用它来操控系统上的一切:网络、软件升级以及所有的基础设置等。无论是 openSUSE 的个人版或是 SUSE Linux 企业版,你都能在 YaST 的强力支撑下,轻松的完成安装。总之,一个工具,方便而无所不能。
|
||||||
|
|
||||||
|
#### #5 开箱即用的极致体验
|
||||||
|
|
||||||
|
SUSE 的团队是 Linux 内核中最大的贡献者团体之一。他们辛勤的努力也意味着,他们有足够的经验来应付不同的硬件条件。
|
||||||
|
|
||||||
|
有着良好的硬件支持,一定会有很棒的开箱即用的体验。
|
||||||
|
|
||||||
|
#### #6 他们做了一些搞笑视频
|
||||||
|
|
||||||
|
等等,不是说好了是五个理由吗?怎么多了一个!
|
||||||
|
|
||||||
|
但因为 [Abhishek](https://itsfoss.com/author/abhishek/) 逼着我加进来,因为他们做的 Linux 的搞笑视频才使 openSUSE 成为了最好的发行版。
|
||||||
|
|
||||||
|
开了个玩笑,不过还是看看 [Uptime Funk](https://www.youtube.com/watch?v=zbABy9ul11I),你将会知道[为什么 SUSE 是最酷的 Linux ](https://itsfoss.com/suse-coolest-linux-enterprise/)。
|
||||||
|
|
||||||
|
### LEAP 还是 TUMBLEWEED?该用哪一个!
|
||||||
|
|
||||||
|
如果你现在想要使用 openSUSE 了,让我来告诉你,这两个 openSUSE 版本:LEAP 和 TUMBLEWEED。哪一个更合适你一点。
|
||||||
|
|
||||||
|
![Choice](https://itsfoss.com/wp-content/uploads/2016/09/Untitled-design-2.jpg)
|
||||||
|
|
||||||
|
尽管两者都提供了相似的体验和相似的环境,但还是需要你自行决定安装那个版本到你硬盘上。
|
||||||
|
|
||||||
|
#### OPENSUSE : LEAP
|
||||||
|
|
||||||
|
[openSUSE Leap](https://en.opensuse.org/Portal:Leap) 是一个普通的大众版本,基本上每八个月更新一次。目前最新的版本是 openSUSE 42.1。它所有的软件都是稳定版,给用户提供最顺畅的体验。
|
||||||
|
|
||||||
|
对于家庭用户、办公和商业用户,是再合适不过的了。它合适那些想要一个稳定系统,而不必事事亲为,可以让他们安心工作的用户。只要进行正确的设置之后,你就不必担心其他事情,然后专心投入工作就好。同时,我也强烈建议图书馆和学校使用 Leap。
|
||||||
|
|
||||||
|
#### OPENSUSE: TUMBLEWEED
|
||||||
|
|
||||||
|
[Tumbleweed version of openSUSE](https://en.opensuse.org/Portal:Tumbleweed) 是滚动式更新的版本。它定期更新系统中所使用的软件集的最新版本。对于想要使用最新软件以及想向 openSUSE 做出贡献的开发者和高级用户来说,这个版本绝对值得一试。
|
||||||
|
|
||||||
|
需要指出的是,Tumbleweed 并不是 Leap 的 beta 和 测试版本 。它是最新锐的 Linux 稳定发行版。
|
||||||
|
|
||||||
|
Tumbleweed 给了你最快的更新,但是仅在开发者确定某个包的稳定性之后才行。
|
||||||
|
|
||||||
|
### 说说你的想法?
|
||||||
|
|
||||||
|
请在下方评论说出你对 openSUSE 的想法。如果你已经在考虑使用 openSUSE,你更想用 Leap 和 Tumbleweed 哪一个版本呢?
|
||||||
|
|
||||||
|
来,让我们开干!
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://itsfoss.com/why-use-opensuse/
|
||||||
|
|
||||||
|
作者:[Aquil Roshan][a]
|
||||||
|
译者:[GHLandy](https://github.com/GHLandy)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: https://itsfoss.com/author/aquil/
|
@ -0,0 +1,85 @@
|
|||||||
|
如何在 Ubuntu 16.04 和 Fedora 22-24 上安装最新的 XFCE 桌面?
|
||||||
|
==========================
|
||||||
|
|
||||||
|
Xfce 是一款针对 Linux 系统的现代化[轻型开源桌面环境][1],它在其他的类 Unix 系统上,比如 Mac OS X、 Solaries、 *BSD 以及其它几种上也能工作得很好。它非常快并以简洁而优雅的用户界面展现了用户友好性。
|
||||||
|
|
||||||
|
在服务器上安装一个桌面环境有时还是有用的,因为某些应用程序可能需要一个桌面界面,以便高效而可靠的管理。 Xfce 的一个卓越的特性是其内存消耗等系统资源占用率很低,因此,如果服务器需要一个桌面环境的话它会是首选。
|
||||||
|
|
||||||
|
### XFCE 桌面的功能特性
|
||||||
|
|
||||||
|
另外,它的一些值得注意的组件和功能特性列在下面:
|
||||||
|
|
||||||
|
- Xfwm 窗口管理器
|
||||||
|
- Thunar 文件管理器
|
||||||
|
- 用户会话管理器:用来处理用户登录、电源管理之类
|
||||||
|
- 桌面管理器:用来设置背景图片、桌面图标等等
|
||||||
|
- 应用管理器
|
||||||
|
- 它的高度可连接性以及一些其他次要功能特性
|
||||||
|
|
||||||
|
Xfce 的最新稳定发行版是 Xfce 4.12,它所有的功能特性和与旧版本的变化都列在了[这儿][2]。
|
||||||
|
|
||||||
|
#### 在 Ubuntu 16.04 上安装 Xfce 桌面
|
||||||
|
|
||||||
|
Linux 发行版,比如 Xubuntu、Manjaro、OpenSUSE、Fedora Xfce Spin、Zenwalk 以及许多其他发行版都提供它们自己的 Xfce 桌面安装包,但你也可以像下面这样安装最新的版本。
|
||||||
|
|
||||||
|
```
|
||||||
|
$ sudo apt update
|
||||||
|
$ sudo apt install xfce4
|
||||||
|
```
|
||||||
|
|
||||||
|
等待安装过程结束,然后退出当前会话,或者你也可以选择重启系统。在登录界面,选择 Xfce 桌面,然后登录,截图如下:
|
||||||
|
|
||||||
|
![](http://www.tecmint.com/wp-content/uploads/2016/09/Select-Xfce-Desktop-at-Login.png)
|
||||||
|
|
||||||
|
![](http://www.tecmint.com/wp-content/uploads/2016/09/XFCE-Desktop.png)
|
||||||
|
|
||||||
|
|
||||||
|
#### 在 Fedora 22-24 上安装 Xfce 桌面
|
||||||
|
|
||||||
|
如果你已经有一个安装好的 Linux 发行版 Fedora,想在上面安装 xfce 桌面,那么你可以使用如下所示的 yum 或 dnf 命令。
|
||||||
|
|
||||||
|
```
|
||||||
|
-------------------- 在 Fedora 22 上 --------------------
|
||||||
|
# yum install @xfce
|
||||||
|
-------------------- 在 Fedora 23-24 上 --------------------
|
||||||
|
# dnf install @xfce-desktop-environment
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
安装 Xfce 以后,你可以从会话菜单选择 Xfce 登录或者重启系统。
|
||||||
|
|
||||||
|
![](http://www.tecmint.com/wp-content/uploads/2016/09/Select-Xfce-Desktop-at-Fedora-Login.png)
|
||||||
|
|
||||||
|
![](http://www.tecmint.com/wp-content/uploads/2016/09/Install-Xfce-Desktop-in-Fedora.png)
|
||||||
|
|
||||||
|
|
||||||
|
如果你不再想要 Xfce 桌面留在你的系统上,那么可以使用下面的命令来卸载它:
|
||||||
|
|
||||||
|
```
|
||||||
|
-------------------- 在 Ubuntu 16.04 上 --------------------
|
||||||
|
$ sudo apt purge xfce4
|
||||||
|
$ sudo apt autoremove
|
||||||
|
-------------------- 在 Fedora 22 上 --------------------
|
||||||
|
# yum remove @xfce
|
||||||
|
-------------------- 在 Fedora 23-24 上 --------------------
|
||||||
|
# dnf remove @xfce-desktop-environment
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
在这个简单的入门指南中,我们讲解了如何安装最新版 Xfce 桌面的步骤,我相信这很容易掌握。如果一切进行良好,你可以享受一下使用 xfce —— 这个[ Linux 系统上最佳桌面环境][1]之一。
|
||||||
|
|
||||||
|
此外,如果你再次回来,你可以通过下面的反馈表单和我们始终保持联系。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: http://www.tecmint.com/install-xfce-desktop-in-ubuntu-fedora/
|
||||||
|
|
||||||
|
作者:[Aaron Kili][a]
|
||||||
|
译者:[ucasFL](https://github.com/ucasFL)
|
||||||
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]: http://www.tecmint.com/author/aaronkili/
|
||||||
|
[1]: http://www.tecmint.com/best-linux-desktop-environments/
|
||||||
|
[2]: https://www.xfce.org/about/news/?post=1425081600
|
@ -3,31 +3,30 @@
|
|||||||
|
|
||||||
![](https://opensource.com/sites/default/files/styles/image-full-size/public/images/life/wikipedia_checkuser_lead.jpg?itok=4lVDjSSM)
|
![](https://opensource.com/sites/default/files/styles/image-full-size/public/images/life/wikipedia_checkuser_lead.jpg?itok=4lVDjSSM)
|
||||||
|
|
||||||
|
我自认为自己是个奥迪亚的维基人。我通过写文章和纠正文章错误给很多维基项目贡献了[奥迪亚(Odia)][1]知识(这是在印度的[奥里萨邦][2]的主要语言),比如维基百科和维基文库,我也为用印地语和英语写的维基文章做贡献。
|
||||||
我自认为自己是个奥迪亚的维基人。我通过写文章和纠正错误的文章贡献[奥迪亚][1]知识(在印度的[奥里萨邦][2]的主要的语言 )给很多维基项目,像维基百科和维基文库,我也为用印地语和英语写的维基文章做贡献。
|
|
||||||
|
|
||||||
![](https://opensource.com/sites/default/files/resize/1st_day_at_odia_wikipedia_workshop_image_source_facebook-200x133.jpg)
|
![](https://opensource.com/sites/default/files/resize/1st_day_at_odia_wikipedia_workshop_image_source_facebook-200x133.jpg)
|
||||||
|
|
||||||
我对维基的爱从我第 10 次考试(像在美国的 10 年级学生的年级考试)之后看到的英文维基文章[孟加拉解放战争][3]开始。一不小心我打开了印度维基文章的链接,,并且开始阅读它. 在文章左边有用奥迪亚语写的东西, 所以我点击了一下, 打开了一篇在奥迪亚维基上的 [????/Bhārat][4] 文章. 发现了用母语写的维基让我很激动!
|
我对维基的爱从我第 10 次考试(像在美国的 10 年级学生的年级考试)之后看到的英文维基文章[孟加拉解放战争][3]开始。一不小心我打开了印度维基文章的链接,并且开始阅读它。在文章左边有用奥迪亚语写的东西,所以我点击了一下, 打开了一篇在奥迪亚维基上的 [ଭାରତ/Bhārat][4] 文章。发现了用母语写的维基让我很激动!
|
||||||
|
|
||||||
![](https://opensource.com/sites/default/files/resize/introducing_wikipedia_at_google_io_image_by_gdg_bhubaneswar-251x166.png)
|
![](https://opensource.com/sites/default/files/resize/introducing_wikipedia_at_google_io_image_by_gdg_bhubaneswar-251x166.png)
|
||||||
|
|
||||||
一个邀请读者参加 2014 年 4 月 1 日第二次布巴内斯瓦尔的研讨会的标语引起了我的好奇。我过去从来没有为维基做过贡献, 只用它搜索过, 我并不熟悉开源和社区贡献流程。加上,我只有 15 岁。我注册了。在研讨会上有很多语言爱好者,我是中间最年轻的一个。尽管我害怕我父亲还是鼓励我去参与。他起了非常重要的作用—他不是一个维基媒体人,和我不一样,但是他的鼓励给了我改变奥迪亚维基的动力和参加社区活动的勇气。
|
一个邀请读者参加 2014 年 4 月 1 日召开的第二届布巴内斯瓦尔研讨会的旗帜广告引起了我的好奇。我过去从来没有为维基做过贡献,只用它做过研究,我并不熟悉开源和社区贡献流程。再加上,当时我只有 15 岁。我注册了,在研讨会上有很多语言爱好者,他们全比我大。尽管我害怕,我父亲还是鼓励我去参与。他起了非常重要的作用—他不是一个维基媒体人,和我不一样,但是他的鼓励给了我改变奥迪亚维基的动力和参加社区活动的勇气。
|
||||||
|
|
||||||
我相信奥迪亚语言和文学需要改进很多错误的想法和知识缺口所以,我帮助组织关于奥迪亚维基的活动和和研讨会,我完成了如下列表:
|
我觉得关于奥迪亚语言和文学的知识很多需要改进,有很多错误的观念和知识缺口,所以,我帮助组织关于奥迪亚维基的活动和和研讨会,我完成了如下列表:
|
||||||
|
|
||||||
* 发起3次主要的 edit-a-thons 在奥迪亚维基:2015 年妇女节,2016年妇女节, abd [Nabakalebara edit-a-thon 2015][5]
|
* 在奥迪亚维基发起 3 次主要的 edit-a-thons :2015 年妇女节、2016年妇女节、abd [Nabakalebara edit-a-thon 2015][5]
|
||||||
* 在全印度发起了征集[檀车节][6]图片的比赛
|
* 在全印度发起了征集[檀车节][6]图片的比赛
|
||||||
* 在谷歌的两大事件([谷歌I/O大会扩展][7]和谷歌开发节)中代表奥迪亚维基
|
* 在谷歌的两大事件([谷歌I/O大会扩展][7]和谷歌开发节)中代表奥迪亚维基
|
||||||
* 在2015[Perception][8]和第一次[Open Access India][9]会议
|
* 在2015 [Perception][8] 和第一次 [Open Access India][9] 会议
|
||||||
|
|
||||||
![](https://opensource.com/sites/default/files/resize/bengali_wikipedia_10th_anniversary_cc-by-sa4.0_biswaroop_ganguly-251x166.jpg)
|
![](https://opensource.com/sites/default/files/resize/bengali_wikipedia_10th_anniversary_cc-by-sa4.0_biswaroop_ganguly-251x166.jpg)
|
||||||
|
|
||||||
我只编辑维基项目到了去年,在 2015 年一月,当我出席[孟加拉语维基百科的十周年会议][10]和[毗瑟挐][11]活动时,[互联网和社会中心][12]主任,邀请我参加[培训培训师][13] 计划。我的灵感始于扩展奥迪亚维基,为[华丽][14]的活动举办的聚会和培训新的维基人。这些经验告诉我作为一个贡献者该如何为社区工作。
|
我在维基项目当编辑直到去年( 2015 年 1 月)为止,当我出席[孟加拉语维基百科的十周年会议][10]和[毗瑟挐][11]活动时,[互联网和社会中心][12]主任,邀请我参加[培训师培训计划][13]。我的开始超越奥迪亚维基,为[GLAM][14]的活动举办聚会和培训新的维基人。这些经验告诉我作为一个贡献者该如何为社区工作。
|
||||||
|
|
||||||
[Ravi][15],在当时维基的主任,在我的旅程也发挥了重要作用。他非常相信我让我参与到了[Wiki Loves Food][16],维基共享中的公共摄影比赛,组织方是[2016 印度维基会议][17]。在2015的 Loves Food 活动期间,我的团队在维基共享中加入了 10,000+ 有 CC BY-SA 协议的图片。Ravi 进一步巩固了我的承诺,和我分享很多关于维基媒体运动的信息,和他自己在 [奥迪亚维基百科13周年][18]的经历。
|
[Ravi][15],在当时印度维基的总监,在我的旅程也发挥了重要作用。他非常相信我,让我参与到了 [Wiki Loves Food][16],维基共享中的公共摄影比赛,组织方是 [2016 印度维基会议][17]。在 2015 的 Loves Food 活动期间,我的团队在维基共享中加入了 10000+ 采用 CC BY-SA 协议的图片。Ravi 进一步坚定了我的信心,和我分享了很多关于维基媒体运动的信息,以及他自己在 [奥迪亚维基百科 13 周年][18]的经历。
|
||||||
|
|
||||||
不到一年后,在 2015 年十二月,我成为了网络与社会中心的[获取知识的程序][19]的项目助理( CIS-A2K 运动)。我自豪的时刻之一是在普里的研讨会,我们从印度带了 20 个新的维基人来编辑奥迪亚维基媒体社区。现在,我的指导者在一个普里非正式聚会被叫作[WikiTungi][20]。我和这个小组一起工作,把 wikiquotes 变成一个真实的计划项目。在奥迪亚维基我也致力于缩小性别差距。[八个女编辑][21]也正帮助组织聚会和研讨会,参加 [Women's History month edit-a-thon][22]。
|
不到一年后,在 2015 年十二月,我成为了网络与社会中心的[获取知识计划][19]的项目助理( CIS-A2K 运动)。我自豪的时刻之一是在印度普里的研讨会,我们给奥迪亚维基媒体社区带来了 20 个新的维基人。现在,我在一个名为 [WikiTungi][20] 普里的非正式聚会上指导着维基人。我和这个小组一起工作,把奥迪亚 Wikiquotes 变成一个真实的计划项目。在奥迪亚维基我也致力于缩小性别差距。[八个女编辑][21]也正帮助组织聚会和研讨会,参加 [Women's History month edit-a-thon][22]。
|
||||||
|
|
||||||
在我四年短暂而令人激动的旅行之中,我也参与到 [维基百科的教育项目][23],[通讯团队][24],两个全球的 edit-a-thons: [Art and Feminsim][25] 和 [Menu Challenge][26]。我期待着更多的到来!
|
在我四年短暂而令人激动的旅行之中,我也参与到 [维基百科的教育项目][23],[通讯团队][24],两个全球的 edit-a-thons: [Art and Feminsim][25] 和 [Menu Challenge][26]。我期待着更多的到来!
|
||||||
|
|
||||||
@ -38,8 +37,8 @@
|
|||||||
via: https://opensource.com/life/16/4/my-open-source-story-sailesh-patnaik
|
via: https://opensource.com/life/16/4/my-open-source-story-sailesh-patnaik
|
||||||
|
|
||||||
作者:[Sailesh Patnaik][a]
|
作者:[Sailesh Patnaik][a]
|
||||||
译者:[译者ID](https://github.com/hkurj)
|
译者:[hkurj](https://github.com/hkurj)
|
||||||
校对:[校对者ID](https://github.com/校对者ID)
|
校对:[wxy](https://github.com/wxy)
|
||||||
|
|
||||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创翻译,[Linux中国](https://linux.cn/) 荣誉推出
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创翻译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user