老古董Lisp实用主义入门教程(8):挠痒痒先生建网站记

news/2025/1/15 19:50:43/

logo

是时候来个真正的应用

几位奇形怪状, 百无聊赖的先生, 用Common Lisp 搞东搞西一阵子, 总觉得没有干什么正经事.
一般而言, 学习编程语言总是应该先搞点计算, 让CPU燥起来.
但是Lisp搞计算总感觉有点不太对劲, 虽然颠倒先生已经尝试把数学公式改成中序以增强动力, 但是不行. 隔壁Matlab太好用, Python太低阻, 而且我通常用C++搞计算.

那么干点什么呢?

挠痒痒先生强行刷存在感: web, web, web.

好吧, 那就来个web应用吧.

选择工具

最重要就是什么来着, 原生部署, 无框架, 无库, 无依赖. 这些都是Lisp早就玩过的啊.

只需要安装一个, 服务器, 一个网页框架, 随便选一个 hunchentoot来做服务器, 一个cl-who就能解决问题. 部署的事情, 粗鲁先生都已经解决.

安装

lisp">(qllisp-property property">:quickload lisp-property property">:hunchentoot)
(qllisp-property property">:quickload lisp-property property">:cl-who)

Hunchentoot

Hunchentoot is a web server written in Common Lisp and at the same time a toolkit for building dynamic websites. As a stand-alone web server, Hunchentoot is capable of HTTP/1.1 chunking (both directions), persistent connections (keep-alive), and SSL.

挠痒痒先生中文:

Hunchentoot是用Common Lisp编写的web服务器, 同时也是动态网站构建工具包. 作为一个独立web服务器, Hunchentoot支持HTTP/1.1 chunking(双向), 持久连接(keep-alive), 和SSL.

Hunchentoot provides facilities like automatic session handling (with and without cookies), logging, customizable error handling, and easy access to GET and POST parameters sent by the client. It does not include functionality to programmatically generate HTML output. For this task you can use any library you like, e.g. (shameless self-plug) CL-WHO or HTML-TEMPLATE.

挠痒痒先生中文:

Hunchentoot提供大量工具, 类似于自动会话处理(有和无cookie), 日志, 可定制的错误处理, 以及方便的访问客户端发送的GET和POST参数. 它不含HTML生成功能. 可以使用相应库来完成, 比如CL-WHO(同一作者)或HTML-TEMPLATE.

Hunchentoot talks with its front-end or with the client over TCP/IP sockets and optionally uses multiprocessing to handle several requests at the same time. Therefore, it cannot be implemented completely in portable Common Lisp. It currently works with LispWorks and all Lisps which are supported by the compatibility layers usocket and Bordeaux Threads.

挠痒痒先生中文:

Hunchentoot通过TCP/IP套接字与前端或客户端通信, 可选择使用多进程同时处理多请求. 因此, 它无法完全在可移植的Common Lisp中实现. 它目前可以在LispWorks和所有由兼容层usocket和Bordeaux Threads支持的Lisp中工作.^1

Hunchentoot comes with a BSD-style license so you can basically do with it whatever you want.

挠痒痒先生中文:

Hunchentoot使用类BSD许可证, 非常自由.

CL-WHO

Common Lisp是产生结构文档的最佳语言, 没有之一. 当然这是完美先生的个人意见, 他简直被CL操作符号的能力震撼得不知所措.

CL-WHO是一个Common Lisp库, 用于生成HTML, XML, 和其他结构文档. 它是一个简单的DSL, 用于生成文档. 它的目标是生成可读性高, 可维护性强的文档. 它的输出是一个字符串, 可以直接发送给客户端. 同样CL-WHO也是BSD许可证,相当自由.

与Hunchentoot不同, 所有的Common Lisp实现都可以运行CL-WHO.

古早游戏投票系统

挠痒痒先生自己还没有学会用Common Lisp写游戏, 他擅长爬网站, 因为手长.

这不, 他找到大神的著作Lisp for Web, 这里有一个古早游戏投票系统的例子.

这个例子也很简单, 为经典的游戏投票, 在线投票的票数会实时更新. 也能增加自己喜欢但是没有列出的游戏.

完美先生觉得这个例子很适合, 他也喜欢古早游戏, 他也想要一个投票系统.

开发过程

Lisp的开发过程懒惰先生整理得很清楚, 这里就不再赘述.

lisp">(qllisp-property property">:quickload '(lisp-property property">:hunchentoot lisp-property property">:cl-who  lisp-property property">:trivial-open-browser))(defpackage lisp-property property">:retro-games(lisp-property property">:nicknames lisp-property property">:rg)(lisp-property property">:use lisp-property property">:cl lisp-property property">:hunchentoot lisp-property property">:cl-who)(lisp-property property">:export lisp-property property">:main))(in-package lisp-property property">:retro-games)

这样, 就可以在文件中增加代码, 在代码中直接使用cl, hunchentoot, 和cl-who的函数. 当然, 所有的符号都会定义在retro-games包中, 这个包的别名叫rg. 这里还导出了一个函数main, 用于启动web服务器.

首先是后台部分

lisp">(defclass game ()((name lisp-property property">:initarg lisp-property property">:namelisp-property property">:reader name)(votes lisp-property property">:initform 0lisp-property property">:accessor votes)))

这里定义了一个game类, 有两个slot, 一个是name, 一个是votes. name是游戏的名字, votes是投票数.

lisp">
(defmethod vote-for (user-selected-game)(incf (votes user-selected-game)))

这里定义了一个方法, 用于投票. 传入一个游戏对象, 投票数加1.

lisp">
(defmethod print-object ((object game) stream)(print-unreadable-object (object stream lisp-property property">:type t)(with-slots (name votes) object(format stream "name: ~s with ~d votes" name votes))))

这里定义了一个方法, 用于打印对象. 传入一个游戏对象, 打印出游戏名字和投票数. 这个方法, 会被系统的print函数调用.

lisp">
(defvar *games* '())(defun game-from-name (name)(find name *games* lisp-property property">:key #'name lisp-property property">:test #'string-equal))

这个函数用于根据游戏名字, 返回游戏对象.

lisp">
(defun game-stored-p (name)(game-from-name name))

这个函数判断游戏是否已经存在.

lisp">
(defun games ()(sort (copy-list *games*) #'> lisp-property property">:key #'votes))

这个函数返回所有游戏, 按照投票数排序.

lisp">
(defun add-game (name)(unless (game-stored-p name)(push (make-instance 'game lisp-property property">:name name) *games*)))

最后是增加游戏的函数.

完美先生, 鲁莽先生, 粗鲁先生, 懒惰先生, 他们都是暴白, 所以, 你懂的.

lisp">(mapcar #'add-game '("魔兽世界" "魔兽争霸" "魔兽争霸2" "魔兽争霸3" "风暴英雄"))

风暴英雄今年必火!

lisp">(game-from-name "魔兽世界")
#<GAME name: "魔兽世界" with 0 votes>

这里返回的游戏对象, 其打印形式就是由print-object方法定义的.

其次是前台部分

lisp">(setf (html-mode) lisp-property property">:html5)(defmacro standard-page ((lisp-marker">&key title) lisp-marker">&body body)`(with-html-output-to-string(*standard-output* nil lisp-property property">:prologue t lisp-property property">:indent t)(lisp-property property">:html lisp-property property">:lang "en"(lisp-property property">:head(lisp-property property">:meta lisp-property property">:charset "utf-8")(lisp-property property">:title ,title)(lisp-property property">:link lisp-property property">:type "text/css"lisp-property property">:rel "stylesheet"lisp-property property">:href "/retro.css"))(lisp-property property">:body(lisp-property property">:div lisp-property property">:id "header" ; Retro games header(lisp-property property">:img lisp-property property">:src "/logo.png"lisp-property property">:alt "Comodore 64"lisp-property property">:class "logo")(lisp-property property">:span lisp-property property">:class "strapline""Vote on your favarourite retro games")),@body))))

前台的部分特别简单, 定义了一个宏standard-page, 用于生成标准页面. 页面包括了html, head, body, div, img, span等标签.

最终页面的本体, 也就是body部分, 由调用者传入.

完美先生认为挠痒痒先生就不用纠结, 用就行了.

lisp">(standard-page(lisp-property property">:title "Page 1"))

这样就生成了一个页面, 但是没有内容.

<!DOCTYPE html><html lang='en'><head><meta charset='utf-8'><title>Page 1</title><link type='text/css' rel='stylesheet' href='/retro.css'></head><body><div id='header'><img src='/logo.png' alt='Comodore 64' class='logo'><span class='strapline'>Vote on your favarourite retro games</span></div></body>
</html>

增加内容, 也就是body部分.

lisp">(standard-page(lisp-property property">:title "Page 1")(lisp-property property">:h1 "Hello, World!"))

这样就生成了一个页面, 有一个标题.

<!DOCTYPE html><html lang='en'><head><meta charset='utf-8'><title>Page 2</title><link type='text/css' rel='stylesheet' href='/retro.css'></head><body><div id='header'><img src='/logo.png' alt='Comodore 64' class='logo'><span class='strapline'>Vote on your favarourite retro games</span></div><h1>Hello World</h1></body>
</html>

大概的情况就是这样, 完美先生编(抄)完这个完美的宏,感觉自己实在太完美.

最后是web服务器

Hunchentoot的结构很简单, 有一个对象ACCEPTOR, 用于接受请求.

lisp">(describe 'hunchentootlisp-property property">:acceptor)
HUNCHENTOOTlisp-property property">:ACCEPTOR[symbol]
ACCEPTOR names the standard-class #<STANDARD-CLASS HUNCHENTOOTlisp-property property">:ACCEPTOR>:Documentation:To create a Hunchentoot webserver, you make aninstance of this class and use the generic function START to start it(and STOP to stop it).  Use the lisp-property property">:PORT initarg if you don't want tolisten on the default http port 80.  There are other initargs most ofwhich you probably won't need very often.  They are explained indetail in the docstrings of the slot definitions for this class.Unless you are in a Lisp without MP capabilities, you can have severalactive instances of ACCEPTOR (listening on different ports) at thesame time.Direct superclasses: STANDARD-OBJECTDirect subclasses: HUNCHENTOOTlisp-property property">:EASY-ACCEPTOR, HUNCHENTOOTlisp-property property">:SSL-ACCEPTORNot yet finalized.Direct slots:HUNCHENTOOT:lisp-property property">:PORTInitargs: lisp-property property">:PORTReaders: HUNCHENTOOTlisp-property property">:ACCEPTOR-PORTDocumentation:The port the acceptor is listening on.  Thedefault is 80.  Note that depending on your operating system you mightneed special privileges to listen on port 80.  When 0, the port will bechosen by the system the first time the acceptor is started.
;; 太长不列...

这个类的基本结构如上所示, 主要的slotport, address, name, request-class, reply-class, taskmaster, output-chunking-p, input-chunking-p, persistent-connections-p, read-timeout, write-timeout, listen-socket, listen-backlog, acceptor-shutdown-p, requests-in-progress, shutdown-queue, shutdown-lock, access-log-destination, message-log-destination, error-template-directory, document-root, 等等.

两个主要的子类是easy-acceptorssl-acceptor.

classDiagramclass ACCEPTOR {+port+address+name+document-root}ACCEPTOR <|-- EASY-ACCEPTORACCEPTOR <|-- SSL-ACCEPTOR

这里的例子, 就主要需要用到port, document-root这两个slot.

lisp">(defvar *hunchentoot-directory*(pathname (directory-namestring #.(or *compile-file-pathname* *load-truename*))))(defvar *server*(make-instance 'easy-acceptorlisp-property property">:port 8080lisp-property property">:document-root *hunchentoot-directory*))(defun start-server ()(start *server*))(defun stop-server ()(stop *server* lisp-property property">:soft nil))

这个服务器,已经可以运行, 并且把当前目录作为document-root, 也就是说, 服务器会把当前目录下的文件作为静态文件提供.只有这样, 前面/retro.css/logo.png才能被访问到.

那么,其他的功能呢?在Common Lisp中,可以一边运行着服务器,一边增加功能,并且不需要重启服务器.这就是Common Lisp的强大之处.

首先是主页面, 用"/retro-games"作为URI, 这就是Hunchentoot提供的路由方式, 每定义一个路由, 就是一个easy-handler.

lisp">(define-easy-handler (retro-games lisp-property property">:uri "/retro-games") ()(standard-page(lisp-property property">:title "Top Tetro Games")(lisp-property property">:h1 "Vote on your all time favorite retro games")(lisp-property property">:p "Missiong a game? Make it available for votes"(lisp-property property">:a lisp-property property">:href "new-game" "here"))(lisp-property property">:h2 "Current stand")(lisp-property property">:div lisp-property property">:id "chart"(lisp-property property">:ol(dolist (game (games))(htm(lisp-property property">:li (lisp-property property">:a lisp-property property">:href (format nil "vote?name=~a" (name game)) "Vote!")(fmt "~A with ~d votes" (escape-string (name game))(votes game)))))))))

除了显示主页面, 我们还需要增加一个给游戏投票的页面, 用"/vote?name=xxx"作为URI.

lisp">(define-easy-handler (vote lisp-property property">:uri "/vote") (name)(when (game-stored-p name)(vote-for (game-from-name name)))(redirect "/retro-games"))

这个路由,增加游戏的投票数, 然后重定向到主页面.

最后, 增加一个页面, 用于增加游戏.

lisp">(define-easy-handler (new-game lisp-property property">:uri "/new-game") ()(standard-page(lisp-property property">:title "Add a new game")(lisp-property property">:h1 "Add a new game to the chart")(lisp-property property">:form lisp-property property">:action "game-added" lisp-property property">:method "post" lisp-property property">:id "addform"(lisp-property property">:p "What is the name of the game?" (lisp-property property">:br))(lisp-property property">:input lisp-property property">:type "text" lisp-property property">:name "name" lisp-property property">:class "txt")(lisp-property property">:p(lisp-property property">:input lisp-property property">:type "submit" lisp-property property">:value "Add" lisp-property property">:class "btn")))))

这个路由显示一个表单, 注意表单的定义, 也是CL-WHO提供的宏.表单的action是"/game-added", 'method’是"post", 这个表单提交后, 会调用"/game-added"这个路由.

lisp">(define-easy-handler (game-added lisp-property property">:uri "/game-added") (name)(unless (or (null name) (zerop (length name)))(add-game name))(redirect "/retro-games"))

这个路由, 用于增加游戏,如果是空白字符串, 什么也不做, 否则调用前面定义的add-game函数, 然后重定向到主页面.

最后是部署

我们还是使用粗鲁先生的方法, 用sbcl启动REPL, 然后加载文件, 最后调用main函数.

这里提供一个打开测试页面的函数, 用于在启动服务器后, 打开浏览器.

lisp">; run open browser by ignoring the error
(defun open-browser (url)(ignore-errors(trivial-open-browserlisp-property property">:open-browser url)))(defun wait-till-quit ()(sb-threadlisp-property property">:join-thread (find-if(lambda (th)(search "hunchentoot-listener-" (sb-threadlisp-property property">:thread-name th)))(sb-threadlisp-property property">:list-all-threads))))(defun kill-server-thread ()(sb-threadlisp-property property">:terminate-thread (find-if(lambda (th)(search "hunchentoot-listener-" (sb-threadlisp-property property">:thread-name th)))(sb-threadlisp-property property">:list-all-threads))))(defun main ()(start-server)(open-browser "http://localhost:8080/retro-games");; handle break signal(handler-case (wait-till-quit)(sb-syslisp-property property">:interactive-interrupt ()(format t "Try Stop the Server.~%")(stop-server)(when (started-p *server*)(kill-server-thread)(format t "Server Stopped.~%")))))

后面增加大量的用于关闭服务器的代码, 用于处理中断信号, 以及关闭服务器线程. 写了一大堆, 在windows上都不行……据说在linux上可以用。

你要问Windows上怎么关闭? 挠痒痒先生可以在楼下把手伸进来按100次Ctrl+C

编译成可执行文件以及丑化页面

lisp">;;;; build-retro-games.lisp
(load "retro-games.lisp")(sb-extlisp-property property">:save-lisp-and-die #p"retro-games.exe"lisp-property property">:toplevel #'rglisp-property property">:mainlisp-property property">:executable t)
$ sbcl --script build-retro-games.lisp

此外, 这里还用了一些css文件, 用于美化页面, 以及一个png文件, 用于图标.

/* set logo image */
.logo {width: auto;height: 30px;
}.strapline {font-family: 'Retro', sans-serif;font-size: 2em;color: #fff;text-shadow: 2px 2px 2px #000;
}/* set list with id chart */
/* list item style */
#chart {list-style-type: none;margin: 0;padding: 0;.li {margin: 0;padding: 0;border-bottom: 1px solid #fff;background-color: #000;color: #fff;font-family: 'Retro', sans-serif;font-size: 1.5em;text-shadow: 2px 2px 2px #000;}
}

大家都看不懂CSS, 也不知道是什么意思, 因为先生们都是美痴.

如果不信, 请看所有人投票选出来的Logo:

在这里插入图片描述

总结

最终的结果也很可喜, 一个简单的exe文件, 实现了超帅的web应用[^1]: 译者注: 本文中的代码, 以及部分内容, 参考了这里的内容.

完整代码文件:

  1. retro-games.lisp
  2. build-retro-games.lisp
  3. retro.css

在这里插入图片描述


http://www.ppmy.cn/news/1525785.html

相关文章

鸿蒙 ArkUI组件一

ArkUI组件 布局 布局指用特定的组件或者属性来管理用户页面所放置UI组件的大小和位置。在实际的开发过程中&#xff0c;需要遵守以下流程保证整体的布局效果&#xff1a; 确定页面的布局结构。分析页面中的元素构成。选用适合的布局容器组件或属性控制页面中各个元素的位置和大…

MySQL 数据库:原理、应用与发展

摘要&#xff1a;本文深入探讨了 MySQL 数据库相关内容。首先介绍了 MySQL 作为开源关系型数据库管理系统的显著特点&#xff0c;包括易用性、跨平台性、高性能、可扩展性、开源免费以及数据安全性等方面。接着详细阐述了其安装与配置过程&#xff0c;涵盖在不同操作系统上的安…

[晕事]今天做了件晕事44 wireshark 首选项IPv4:Reassemble Fragented IPv4 datagrams

不知不觉&#xff0c;已经来到了晕事系列的第四十四个晕事。今天办的晕事和Wireshark查看网络包相关。说&#xff0c;在Wireshark的编辑-首选项协议里的IPv4协议&#xff0c;有一个参数设置是&#xff1a;Reassemble Fragented IPv4 datagrams。 这个参数的含义是指定Wireshar…

Linux通配符*、man 、cp、mv、echo、cat、more、less、head、tail、等指令、管道 | 、指令的本质 等的介绍

文章目录 前言一、Linux通配符*二、man 指令三、 cp 指令四、mv指令五、 echo 指令六、cat 指令七、more 指令八、 less 指令九、 head 指令十、 tail指令十一、 管道 |十二、指令的本质总结 前言 Linux通配符*、man 、cp、mv、echo、cat、more、less、head、tail、等指令、管…

vue2,3生命周期

Vue.js 的生命周期在 Vue 2 和 Vue 3 中有所不同&#xff0c;但基本的概念是相似的。Vue 的生命周期是指 Vue 实例从创建到销毁的整个过程&#xff0c;这个过程中 Vue 实例会触发一系列的事件&#xff0c;我们称之为生命周期钩子&#xff08;Lifecycle Hooks&#xff09;。开发…

在Milvus中创建集合并在集合中插入数据,然后attu管理工具可以查看

日志打印出来的是这个&#xff0c;现在attu为什么看不到插入的数据信息&#xff0c;集合信息已经可以看到&#xff0c;为什么看不到数据呢/home/anaconda3/envs/bi-txt-sql/bin/python -X pycache_prefix/home/.cache/JetBrains/PyCharm2023.2/cpython-cache /home/tools/pycha…

前端——标签二(超链接)

标签二 超链接标签&#xff1a;a 超链接&#xff0c;实现页面间的跳转和数据传输 a标签的属性 href&#xff1a;跳转路径&#xff08;url&#xff09;必须具备&#xff0c;表示点击后会跳转到哪个页面 target&#xff1a;页面打开方式。默认是 _self 如果是 _blank则用新的…

CSDN玩法攻略(维护中)

以下均为测试过的条件 隐形条件和官方描写可能不准确更新不及时 勋章 签到勋章(已下架) 勤写标兵 每周三篇原创等级1 max10 创作能手 lv1 每周1-3 lv2 每周4-6 lv3 每周7-8 lv4 每周>9 持续创作 授予每个自然月内发布4篇或4篇以上原创或翻译IT博文的用户 五一创作勋章 每…