Ajax:表单 模板引擎

Ajax:表单 & 模板引擎

    • form 表单
      • form 属性
    • Ajax操控表单
      • 事件监听
      • 阻止默认行为
      • 收集表单数据
    • 模板引擎
      • art-template
        • {{}}语法
        • 原文输出
        • 条件输出
        • 循环输出
        • 过滤器
      • 原理


form 表单

HTML中,可以通过<form>创建一个表单,收集用户信息。而采集到的信息想要发送给后端,此时就要与Ajax进行配合。

form 属性

在原生HMTL表单中,包含四个基本属性:

属性功能
action向何处提交表单内的数据url
method提交表单的方式getpost
enctype如何对表单数据编码
target打开action url的为止_blank_self
  • action

action的属性值往往是后端提供的一个url地址,这个地址接收并处理来自前端的数据。如果<form>没有指定该值,那么默认为当前页面的url

示例:

<form action="/login"><input type="account" name="account" /><input type="password" name="password" /><button type="submit">提交</button>
</form>

这是一个表单,指定action="/login"

输出结果:

请添加图片描述

可以看到,点击提交后,地址变成了/login?account=123456&password=54321,其中/login就是表单提交到的地址,而?后面是表单的参数。

这里有两个细节:

  1. 目标地址默认是在当前页面打开
  2. 表单的参数追加到了地址的末尾,说明默认是get请求
  • taget

taget指定在何处打开地址,刚刚打开地址是,是默认在当前页面打开,通过指定target就可以改变打开目标地址的方式。

  1. _blank:在新窗口打开
  2. _self:在当前窗口打开,默认值
  • method

刚刚发现,表单的提交方式默认为get,如果想要切换表单提交方式,可以通过method属性。

  1. get:以get形式发送请求,默认值
  2. post:以post形式发送请求

示例:

<body><form action="/login" method="post" ><input type="account" name="account" /><input type="password" name="password" /><button type="submit">提交</button></form>
</body>

此处把表单的提交方式改为post

提交表单后,查看控制台:

在这里插入图片描述

这次提交数据后,地址就是/login,表单的数据不再以?的形式追加到地址末尾。而在Payload窗口,可以看到表单数据accountpassword已经放在了请求的内部。

  • enctype

enctype用于指定表单数据的编码方式,其有三个可选值:

  1. application/x-www-form-urlencoded:在发送数据前编码所以的字符,默认值
  2. multipart/form-data:不对字符编码,如果表单内上传文件时,必须使用该选项,防止错误编码
  3. text/plain:空格转换为'+'字符,但是不对特殊字符编码,几乎不用

原生的HTML表单有很多缺点:

  1. 表单提交后,会发生页面跳转
  2. 表单提交后,会清空表单原有数据

想要处理这些问题,那就要引入JavaScriptAjaxJavaScript可以阻止默认行为,禁止页面进行跳转。但是这样也会阻止数据的提交,因此使用Ajax来进行表单数据提交,不再让表单自己提交数据。而且Ajax提交数据,不会清空表单的内容。


Ajax操控表单

事件监听

如果要在用户提交表单时,让Ajax立刻响应,代替表单的提交任务,此时就要对表单进行提交事件的事件监听。

只需要调用原先的jQuery接口即可:

$('form').submit(function(){})$('form').on('submit', function(){})

以上两种方式,都可以实现对表单的提交事件的监听。


阻止默认行为

想要阻止默认提交行为,也很简单:

$('form').on('submit', function (e) {  e.preventDefault()
})

在回调函数中,添加是事件参数e,随后调用preventDefault方法阻止默认行为,这可以同时阻止提交行为和页面跳转。


收集表单数据

现在已经阻止了提交行为,接下来就要通过Ajax代替原先的提交行为了。在提交数据之前,就要先获取到表单的数据,这里jQuery封装了接口,可以一次性获取所有表单内容。

语法:

$('form').serialize()

注意:使用这个方法时,要确保表单的每个栏目都有name属性,否则无法获取到对应的栏目。

示例:

<form action="/login" id="f1"><input type="account" name="account" /><input type="password" name="password" /><button type="submit">提交</button>
</form><script>$(function () {$('#f1').on('submit', function (e) {e.preventDefault()var data = $(this).serialize()console.log(data)})})
</script>

输出结果:

在这里插入图片描述

此时data就已经拿到了表单的内容,最后就可以通过Ajax进行发送了。


模板引擎

当从表单中拿到了大量的数据,此时可能就需要把这些数据重新渲染到页面上,如果对于原生的JavaScript或者jQuery,处理这个都是比较麻烦的,需要进行字符串拼接等操作,把数据拼接到标签的内部。

假设现在要渲染以下内容:

var data = {title: '<h3>用户信息</h3>',name: '张三',age: 18,isVIP: true,regTime: new Date(),hobby: ['HTML', 'HTTP', 'Linux']
}

渲染为如下结构:

在这里插入图片描述

如果只使用htmljQuery,那么就要从对象提取信息,然后一个一个通过.html方法把信息添加到页面中,对于最后一个hobby数组,还需要通过.each方法进行遍历,追加到页面中。

art-template

这很麻烦,模板引擎可以解决这个问题,以模板的形式,快速渲染大量的信息到网页。此处选用art-template模板引擎,可以到以下网址下载:

http://aui.github.io/art-template/zh-cn/docs/installation.html

只需要把template-web.js(gzip: 6kb)这个文件下载下来,并通过JavaScript的形式引入即可使用。

<script src="./template-web.js"></script>

导入art-template后,全局就会多出一个template方法,这就是模板引擎的核心方法。

示例:

首先,模板引擎的内容要定义在<script>标签内部:

<script type="text/html" id="tpl"><p>姓名: {{name}}</p><p>年龄: {{age}}</p>
</script>

此处要制定type属性和id属性,默认情况下type="text/JavaScript",表示以JavaScript形式解析<script>内部的内容,此处改为text/html,表示以html形式解析内容。

随后就可以在内部编写html的代码了,在编写代码时,可以使用{变量名}}的形式来引入外部变量,这个变量会直接取对象中的值进行填入。

模板语法:

template('id', 对象)

此处的id是刚才<script>内部的属性id="tpl",而对象是要填充的对象,最后会返回格式化后的html字符串。

示例:

<script>var data = { name: '张三', age: 20}var htmlStr = template('tpl-user', data)$('#container').html(htmlStr)
</script>

在对象data中,包含nameage两个属性,这刚好和之前指定的{{name}}{{age}}相匹配。随后调用template('tpl, data),表示用data这个对象值,去填充模板tpl的内容,此时就会返回一个html格式的字符串,直接把字符串填入目标容器即可。

<div id="container"></div>
<script type="text/html" id="tpl-user"><p>姓名: {{name}}</p><p>年龄: {{age}}</p>
</script><script>var data = { name: '张三', age: 20}var htmlStr = template('tpl-user', data)$('#container').html(htmlStr)
</script>

输出结果:

在这里插入图片描述

这样就实现了快速渲染对象的内容到html中。


{{}}语法

刚才已经见过{{}}的基本功能了,其实就是把对象中命名相匹配的内容,直接填充到html中,其实它还支持很多种语法格式:

{{value}}
{{obj.key}}    // 获取对象值
{{obj['key']}} // 获取对象值
{{a ? b : c}}  // 三元表达式
{{a || b}}     // 逻辑判断
{{a + b}}      // 算数运算

原文输出

如果输出的内容是一个字符串,并且字符串中包含html标签,如果希望这个标签被渲染,需要使用原文输出

{{@ value}}

在值前面加一个@符号即可。

示例:

<div id="container"></div>
<script type="text/html" id="tpl"><p>姓名: {{@ name}}</p><p>年龄: {{age}}</p>
</script><script>var data = { name: '<li>张三</li>', age: 20}var htmlStr = template('tpl', data)$('#container').html(htmlStr)
</script>

输出结果:

在这里插入图片描述

通过{{@ name}}<li>标签就被正常解析了。


条件输出

有时候,可能需要在输出时进行条件判断,比如说如果某个用户是VIP,要额外输出VIP的剩余时间。而如果他不是VIP,这个内容就不用输出。对于这种情况,可以使用条件输出来完成。

语法:

{{if value}} 输出内容 {{/if}}{{if value}} 输出内容1 {{else}} 输出内容2 {{/if}}{{if value1}} 输出内容1 {{else if value2}} 输出内容2 {{/if}}

示例:

<script type="text/html" id="tpl"><p>姓名: {{name}}</p><p>年龄: {{age}}</p>{{if isVip}} <p> VIP剩余时间: {{time}} </p>{{else}}<p> 不是VIP </p>{{/if}}<hr>
</script>

在模板中,通过{{if isVip}}来判断该用户是不是VIP,并选择不同的输出结果。

填入以下两个对象:

<script>var zs = { name: '张三',age: 20,isVip: true,time: 100}var ls = { name: '李四',age: 25,isVip: false,time: null}var htmlStr1 = template('tpl', zs)var htmlStr2 = template('tpl', ls)$('#container').html(htmlStr1 + htmlStr2)
</script>

示例中,张三VIP,而李四不是,最后这两者渲染出来的结果就不一样:

在这里插入图片描述


循环输出

如果输出的目标是一个数组,可以通过循环输出来遍历所有的数组元素。

语法:

{{each arr}}{{$index}} {{$value}}
{{/each}}

其中arr是被遍历的数组,而$index是下标,$value是元素值。

示例:

<script type="text/html" id="tpl"><p>姓名: {{name}}</p><p>年龄: {{age}}</p><ul>{{each hobby}} <li> {{$value}} </li>{{/each}}</ul>
</script>

<ul>内部,通过循环遍历所有的hobby,并通过{{$value}}拿到元素值,渲染到li中。

给张三加上hobby属性:

<script>var zs = { name: '张三',age: 20,hobby: ['HTML', 'HTTP', 'Linux']}var htmlStr = template('tpl', zs)$('#container').html(htmlStr)
</script>

输出结果:

在这里插入图片描述

这样就可以快速遍历所有的数组成员了。


过滤器

如果说,对象中的元素并不是直接填写到页面中的,而是要经过一定的处理之后在填入。比如说商品价格,可能要乘以某个值后,以打折的价格显示在页面中,这就需要引入过滤器

{{value | filterName }}

其中value是要处理的值,这个值会交给filterName 函数处理,函数返回值会最为最终结果填入模板。

过滤器函数并不是直接就可以调用原生的函数,而是要导入到模板中:

template.defaults.imports.filterName = function(value){ 函数体 }

现在张三对象内容如下:

var zs = { name: '张三',age: 20,hobby: ['HTML', 'HTTP', 'Linux'],startYear: 2023
}

此处的startYear表示入学年份,最终渲染到页面中时,希望输出学年: 入学年份 ~ 毕业年份,此时可以交给一个过滤器处理:

template.defaults.imports.graduate = function(value){return value + " ~ " + (value + 4)  }

过滤器graduate 会进行一个字符串拼接,虽然value是数字类型,但是中间的" ~ "是一个字符串,最终整个返回值就是一个字符串,毕业年份是入学年份的四年后。

在模板中,把startYear 输入到过滤器:

<script type="text/html" id="tpl"><p>姓名: {{name}}</p><p>年龄: {{age}}</p><ul>{{each hobby}} <li> {{$value}} </li>{{/each}}</ul><p>学年: {{ startYear | graduate }}</p>
</script>

此处{{ startYear | graduate }}就是在处理数据。

输出结果:

在这里插入图片描述


原理

其实模板引擎的原理非常简单,就是正则表达式匹配,随后替换元素值即可。

art-template为例,其所有占位符格式都是{{}},那么正则就可以写为:

let pattern = /{{([a-zA-Z]+)}}/

首先通过最外层的{{}}进行占位符得到匹配,内层一个()进行分组匹配,提取出具体的信息,[a-zA-Z]+表示提取任意多个字母。

比如对于字符串<div>{{name}}</div>,通过以上正则就可以得到name这个值。

var str = '<div>{{name}}</div>'
var pattern = /{{([a-zA-Z]+)}}/var result = pattern.exec(str)
console.log(result)

输出结果:

在这里插入图片描述

可以看到,最后正则返回一个数组,第一个元素是匹配成功的字符串,第二个元素是提取出来的分组值。

当提取到变量名后,就要去对象中拿到对于的变量,并替换原先的占位符,替换通过replace方法完成:

'string'.replace('子串', '替换值')

比如说"helloworld".replace("world", "linux"),就会返回字符串"hellolinux"

let obj = {name: "张三"
}var str = '<div>{{name}}</div>'
var pattern = /{{([a-zA-Z]+)}}/var patternResult = pattern.exec(str)
str = str.replace(patternResult[0], obj[patternResult[1]])
console.log(str)

patternResult[0]是被匹配的字符串,也就是{{name}}patternResult[1]是提取到的分组,也就是name。只需要把原字符串str{{name}}替换为obj['name']即可,也就是str.replace(patternResult[0], obj[patternResult[1]])

输出结果:

在这里插入图片描述

这样就完成了元素的替换。

当整个字符串中有多个{{xxx}},可以使用一个循环进行匹配:

let patternResult = null;
while (patternResult = pattern.exec(str)) {str = str.replace(patternResult[0], obj[patternResult[1]])
}

pattern.exec匹配不到{{}}时,就会返回一个null,循环就会结束,整个模板字符串被完全替换为对象的值,最后得到一个可以直接渲染的html字符串。


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/459948.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【水下生物数据集】 水下生物识别 深度学习 目标检测 机器视觉 yolo(含数据集)

一、背景意义 随着全球海洋生态环境的日益变化&#xff0c;水下生物的监测和保护变得愈发重要。水下生物种类繁多&#xff0c;包括螃蟹、鱼类、水母、虾、小鱼和海星等&#xff0c;它们在海洋生态系统中扮演着关键角色。传统的水下生物监测方法通常依赖于人工观察&#xff0c;效…

[vulnhub]Kioptrix: Level 1.2 (#3)

https://www.vulnhub.com/entry/kioptrix-level-12-3,24/ 主机发现端口扫描 使用nmap扫描网段类存活主机 因为靶机是我最后添加的&#xff0c;所以靶机IP是169 nmap -sP 192.168.75.0/24 Starting Nmap 7.94SVN ( https://nmap.org ) at 2024-10-29 13:16 CST …

TVM前端研究--Relay

文章目录 深度学习IR梳理1. IR属性2. DL前端发展3. DL编译器4. DL编程语言Relay的主要内容一、Expression in Relay1. Dataflow and Control Fragments2. 变量3. 函数3.1 闭包3.2 多态和类型关系3.3. Call4. 算子5. ADT Constructors6. Moudle和Global Function7. 常量和元组8.…

Ubuntu UFW防火墙规则与命令示例大全

在服务器安全领域&#xff0c;防火墙是守护网络安全的坚实盾牌。UFW&#xff08;Uncomplicated Firewall&#xff09;&#xff0c;即“不复杂的防火墙”&#xff0c;是一个运行在iptables之上的防火墙配置工具&#xff0c;它为Ubuntu系统默认提供了一个简洁的命令行界面&#x…

Linux高阶——1026—验证内存映射mmap函数使用

1、验证共享映射后修改文件内容&#xff0c;是否能够同步 先创建一个映射文件&#xff0c;写入数据 分为四个步骤 1、打开映射文件 设文件描述符&#xff0c;使用open函数 int fd; if((fdopen("mapfile",O_RDWR))-1) { perror("open failed");exit…

从零开始的 vue项目部署到服务器详细步骤(vue项目build打包+nginx部署+配置ssl证书)

从零开始的 vue项目部署到服务器详细步骤&#xff08;vue项目build打包nginx部署配置ssl证书&#xff09; 文章目录 从零开始的 vue项目部署到服务器详细步骤&#xff08;vue项目build打包nginx部署配置ssl证书&#xff09;一、前言二、vue项目部署前配置1、vite.config.js 增加…

快速遍历包含合并单元格的Word表格

Word中的合并表格如下&#xff0c;现在需要根据子类&#xff08;例如&#xff1a;果汁&#xff09;查找对应的品类&#xff0c;如果这是Excel表格&#xff0c;那么即使包含合并单元格&#xff0c;也很容易处理&#xff0c;但是使用Word VBA进行查找&#xff0c;就需要一些技巧。…

智慧园区 | 数智引领,让智慧触手可及

随着科技的飞速发展&#xff0c;智慧园区正成为现代城市发展的重要方向之一。在智慧园区中&#xff0c;各种高科技手段被应用于园区的管理和服务&#xff0c;为园区的运营和居民的生活带来无限可能。 智慧园区管理平台是智慧园区建设的核心。它集聚了大数据、物联网、云计算等技…

基于uniapp微信小程序的旅游系统

作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、Vue项目源码、SSM项目源码、微信小程序源码 精品专栏&#xff1a;…

【分布式知识】分布式对象存储组件-Minio

文章目录 什么是minio核心特点&#xff1a;使用场景&#xff1a;开发者工具&#xff1a;社区和支持&#xff1a; 核心概念什么是对象存储&#xff1f;MinIO 如何确定对对象的访问权限&#xff1f;我可以在存储桶内按文件夹结构组织对象吗&#xff1f;如何备份和恢复 MinIO 上的…

iQOO手机怎样将屏幕投射到MacBook?可以同步音频吗?

众所周知&#xff0c;苹果品牌的设备自己有AirPlay的投屏功能&#xff0c;iPhone要投屏到MacBook只要连接同一网络&#xff0c;然后开启AirPlay就可以投屏。但其他品牌的手机没有AirPlay&#xff0c;怎么将手机屏幕投射到MacBook呢&#xff1f; 安卓系统的手机可以使用无线投屏…

2. 从服务器的主接口入手

Webserver 的主函数 main.cpp&#xff0c;完成了哪些功能&#xff1f; #include "config.h"int main(int argc, char *argv[]) {string user "";string passwd "";string databasename "";Config config;config.parse_arg(argc, a…

四、Prompt工程——简单应用

Prompt工程——简单应用 一、提示工程&#xff08;Prompt Engineering&#xff09;二、Prompt基本法则三、Prompt 调优四、简单的例子文本总结文本判断文本提取文本转化——翻译文本转化——语气 更多结语 一、提示工程&#xff08;Prompt Engineering&#xff09; 提示工程也…

5G RedCap工业路由器赋能电力物联网应用

随着5G轻量化技术应用的推进&#xff0c;5G RedCap旨在提供低功耗、低成本、广覆盖等功能特点赋能电力智能化升级。特别适用于工业物联网、低空经济、车联网、消费电子和轻量级5G的需求。 5G RedCap工业路由器的特点 低功耗&#xff1a;5G RedCap工业路由器通过节能技术&#…

Flume采集Kafka数据到Hive

版本&#xff1a; Kafka&#xff1a;2.4.1 Flume&#xff1a;1.9.0 Hive&#xff1a;3.1.0 Kafka主题准备&#xff1a; Hive表准备&#xff1a;确保hive表为&#xff1a;分区分桶、orc存储、开启事务 Flume准备&#xff1a; 配置flume文件&#xff1a; /opt/datasophon/flume-1…

react18中react-thunk实现公共数据仓库的异步操作

redux及react-redux都只能实现数据的同步修改更新&#xff0c;有点类似于vue中的mutation&#xff0c;只能做同步操作&#xff0c;异步的话不用actions来实现。由于在项目始终不可避免要实现的异步数据的更新&#xff0c;这明显不够用了。是时候引入我们的异步中间件redux-thun…

开源一款前后端分离的企业级网站内容管理系统,支持站群管理、多平台静态化,多语言、全文检索的源码

大家好&#xff0c;我是一颗甜苞谷&#xff0c;今天分享一款前后端分离的企业级网站内容管理系统&#xff0c;支持站群管理、多平台静态化&#xff0c;多语言、全文检索的源码。 前言 在当今的数字化时代&#xff0c;企业网站和个人博客已成为信息传播和品牌建设的重要渠道。…

Docker-常用命令大全(附命令详解)

文章目录 Docker 基础命令查看docker 运行状态关闭docker启动docker重启dockerdocker设置随服务启动而自启动查看docker 版本号信息docker 帮助命令 docker 镜像命令查看自己服务器中docker 镜像列表搜索镜像拉取镜像运行镜像保存镜像删除镜像加载镜像镜像标签 Docker 容器命令…

【ComfyUI】手动安装部署ComfyUI的运行环境

如果不喜欢已有的一键启动包&#xff0c;我们可以手动的安装和部署ComfyUI的运行环境&#xff0c;相比一键安装包&#xff0c;自己部署ComfyUI 环境具有相当大的灵活性&#xff0c;其实部署ComfyUI 环境非常简单&#xff0c;不像网上说的那么复杂。下面我们就按照顺序给大家分享…

Golang | Leetcode Golang题解之第520题检测大写字母

题目&#xff1a; 题解&#xff1a; func detectCapitalUse(word string) bool {// 若第 1 个字母为小写&#xff0c;则需额外判断第 2 个字母是否为小写if len(word) > 2 && unicode.IsLower(rune(word[0])) && unicode.IsUpper(rune(word[1])) {return f…