
在web开发中,我们经常需要从后端获取数据,然后动态地生成html元素并将其插入到页面中。这些动态生成的元素通常需要绑定javascript事件,例如点击按钮、提交表单等,以实现交互功能。一个常见的需求是,当用户点击一个动态生成的按钮时,触发表单提交但又不刷新页面,并调用特定的javascript函数来处理数据。
原始的实现方式可能是在每次生成HTML元素时,也在其内部嵌入一个<script>标签,用于为该元素绑定事件。例如,考虑以下动态生成的商品模板:
<!-- product -->
<div class="container" class="product" id="productId_${index}">
<form action="" class="product">
<img src="${url}"></img>
<p class="product_Description">${description}</p>
<input type="hidden" class="productId" value=${id}>
<input type="hidden" class="price" value="0.${price}">
<label for="quantity">Quantity:</label>
<input type="number" class="quantity" value="1" min="1">
<button class="submit">Add to cart</button>
</form>
</div>
<script>
// 为每个动态生成的商品表单绑定提交事件
document.addEventListener("DOMContentLoaded",() =>{
const product${index} = document.querySelector("#productId_${index}")
// 注意:原始代码中此处可能存在变量引用错误,应为product${index}
product${index}.addEventListener("submit",e=>{
e.preventDefault(); // 阻止表单默认提交
var formId = "productId_${index}";
productList(formId); // 调用业务逻辑函数
});
});
</script>
<!-- END product -->这种方法虽然能实现功能,但存在明显的问题:
为了解决上述问题,最佳实践是采用事件委托(Event Delegation)模式。事件委托的核心思想是:不为每个子元素单独绑定事件,而是将事件监听器附加到一个共同的祖先元素上。当子元素上的事件发生时,事件会沿着DOM树向上冒泡,直到被祖先元素上的监听器捕获。然后,监听器可以根据事件的target属性(即实际触发事件的元素)来判断并执行相应的处理逻辑。
在使用jQuery时,实现事件委托非常简单和强大,主要通过on()方法来完成。
立即学习“Java免费学习笔记(深入)”;
假设所有的动态商品表单都包含在一个静态的父容器#widgetContainer中,并且这些表单都带有product类。我们可以这样设置事件委托:
// 在页面加载后(或getWidgets函数调用后)执行一次
$(document).ready(function() { // 或者使用 $(function() { ... });
$('#widgetContainer').on('submit', 'form.product', function (e) {
e.preventDefault(); // 阻止表单默认提交行为,防止页面刷新
// $(this) 在事件委托中指向实际触发事件的元素,即当前的 'form.product'
// 获取触发事件的表单的父元素的ID (例如:productId_0, productId_1)
const formId = $(this).parent().attr('id');
// 调用业务逻辑函数,传入表单ID
productList(formId);
});
});代码解析:
结合事件委托,原始的getWidgets函数将变得更简洁、更高效。我们只需要在生成HTML模板时移除内联的<script>标签,并在getWidgets函数调用后(或者在文档加载完成后)添加一次事件委托代码。
// 假设 productList(formId) 是一个已经定义好的处理函数
function productList(formId) {
console.log("Form submitted for:", formId);
// 在这里处理表单数据,例如通过AJAX发送
// var formData = $('#' + formId + ' form.product').serialize();
// $.post('/api/submitProduct', formData, function(response){ ... });
}
function getWidgets(){
var listUrl = base_url + widgetsPath + url_auth;
console.log("Sending GET to " + listUrl);
function getSuccess(obj){
var dataWidget = obj.data;
var widgetContainer = $("#widgetContainer"); // 缓存jQuery对象以提高性能
for (let i=0; i< dataWidget.length; i++){
var id = dataWidget[i].id;
var description = dataWidget[i].description;
var price = dataWidget[i].pence_price;
var url = dataWidget[i].url;
var index = i;
console.log("index: ",i);
console.log(dataWidget[i].id);
console.log(dataWidget[i].description);
console.log(dataWidget[i].pence_price);
console.log(dataWidget[i].url);
console.log(" ");
// 移除内联的 <script> 标签
var template =`<!-- product -->
<div class="container" class="product" id="productId_${index}">
<form action="" class="product">
<img src="${url}"></img>
<p class="product_Description">${description}</p>
<input type="hidden" class="productId" value=${id}>
<input type="hidden" class="price" value="0.${price}">
<label for="quantity">Quantity:</label>
<input type="number" class="quantity" value="1" min="1">
<button class="submit">Add to cart</button>
</form>
</div>
<!-- END product -->`;
widgetContainer.append(template);
}
console.log("success");
console.log(dataWidget);
};
$.ajax(listUrl, {type: "GET", data: {},success: getSuccess });
};
// 确保在DOM加载完成后,并且 #widgetContainer 存在时绑定事件
$(document).ready(function() {
getWidgets(); // 调用函数生成动态内容
// 绑定事件委托,只需要执行一次
$('#widgetContainer').on('submit', 'form.product', function (e) {
e.preventDefault();
const formId = $(this).parent().attr('id');
productList(formId);
});
});在这个优化后的版本中,getWidgets函数只负责生成纯HTML内容并插入到DOM中,而事件处理逻辑则通过一个统一的事件委托机制来管理。
通过采用事件委托模式,我们能够优雅且高效地处理动态生成HTML元素的JavaScript事件。这种方法不仅解决了内联脚本带来的性能、维护和时机问题,还使得代码更加清晰、可扩展。在开发涉及大量动态内容的Web应用时,事件委托是绑定事件的首选策略,它能显著提升应用的性能和可维护性。
以上就是优化动态HTML中的JavaScript:告别内联脚本,拥抱事件委托的详细内容,更多请关注php中文网其它相关文章!
HTML怎么学习?HTML怎么入门?HTML在哪学?HTML怎么学才快?不用担心,这里为大家提供了HTML速学教程(入门课程),有需要的小伙伴保存下载就能学习啦!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号