lt;!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>vue组件的模板分离写法</title>
<style type="text/css">
.hezi{
width: 300px;
height: 250px;
border: #000000 solid 1px;
}
</style>
</head>
<body>
<div id="app">
<cpn></cpn>
</div>
<script src="../js/vue.js"></script>
<!--1.script标签, 注意:类型必须是text/x-template-->
<!-- <script type="text/x-template" id="app-hezi">
<div class="hezi">
<h1>我是h1标题</h1>
<p>我是p标签</p>
</div>
</script> -->
<!--2.template标签-->
<template id="app-hezi">
<div class="hezi">
<h1>我是h1标题</h1>
<p>我是p标签</p>
</div>
</template>
<script type="text/javascript">
Vue.component('cpn',{
template: '#app-hezi'
})
const app=new Vue({
el: '#app',
data:{
}
})
</script>
</body>
</html>
、明确前后端分离和前后端不分离的概念:
我的理解:前后端不分离的概念是后端要控制前端的数据显示和模板渲染(django),它有一个缺点就是可复用性不强,也就是它的后端程序只适用于一种前端类型,比如返回的是网页模板,则它只能用于网页端,移动端要用只能重新渲染一个移动端的模板。
而前后端分离则解决了这一问题,它的可复用性极强,一个后端可对接多个类型的前端,因为它不使用模板,而是通过向前端传递json数据的方式,将页面渲染和显示数据交给前端去做。这样写出来的后端可以适用于任何类型的前端。
二、项目的数据库设计:
三、模型类设计
class BaseModel(object):
"""模型基类,为每个模型补充创建时间与更新时间"""
create_time=db.Column(db.DateTime, default=datetime.now) # 记录的创建时间
update_time=db.Column(db.DateTime, default=datetime.now, onupdate=datetime.now) # 记录的更新时间
class User(BaseModel, db.Model):
"""用户"""
__tablename__= "ih_user_profile"
id=db.Column(db.Integer, primary_key=True) # 用户编号
name=db.Column(db.String(32), unique=True, nullable=False) # 用户暱称
password_hash=db.Column(db.String(128), nullable=False) # 加密的密码
mobile=db.Column(db.String(11), unique=True, nullable=False) # 手机号
real_name=db.Column(db.String(32)) # 真实姓名
id_card=db.Column(db.String(20)) # 身份证号
avatar_url=db.Column(db.String(128)) # 用户头像路径
houses=db.relationship("House", backref="user") # 用户发布的房屋
orders=db.relationship("Order", backref="user") # 用户下的订单
class Area(BaseModel, db.Model):
"""城区"""
__tablename__= "ih_area_info"
id=db.Column(db.Integer, primary_key=True) # 区域编号
name=db.Column(db.String(32), nullable=False) # 区域名字
houses=db.relationship("House", backref="area") # 区域的房屋
# 房屋设施表,建立房屋与设施的多对多关系
house_facility=db.Table(
"ih_house_facility",
db.Column("house_id", db.Integer, db.ForeignKey("ih_house_info.id"), primary_key=True), # 房屋编号
db.Column("facility_id", db.Integer, db.ForeignKey("ih_facility_info.id"), primary_key=True) # 设施编号
)
class House(BaseModel, db.Model):
"""房屋信息"""
__tablename__= "ih_house_info"
id=db.Column(db.Integer, primary_key=True) # 房屋编号
user_id=db.Column(db.Integer, db.ForeignKey("ih_user_profile.id"), nullable=False) # 房屋主人的用户编号
area_id=db.Column(db.Integer, db.ForeignKey("ih_area_info.id"), nullable=False) # 归属地的区域编号
title=db.Column(db.String(64), nullable=False) # 标题
price=db.Column(db.Integer, default=0) # 单价,单位:分
address=db.Column(db.String(512), default="") # 地址
room_count=db.Column(db.Integer, default=1) # 房间数目
acreage=db.Column(db.Integer, default=0) # 房屋面积
unit=db.Column(db.String(32), default="") # 房屋单元, 如几室几厅
capacity=db.Column(db.Integer, default=1) # 房屋容纳的人数
beds=db.Column(db.String(64), default="") # 房屋床铺的配置
deposit=db.Column(db.Integer, default=0) # 房屋押金
min_days=db.Column(db.Integer, default=1) # 最少入住天数
max_days=db.Column(db.Integer, default=0) # 最多入住天数,0表示不限制
order_count=db.Column(db.Integer, default=0) # 预订完成的该房屋的订单数
index_image_url=db.Column(db.String(256), default="") # 房屋主图片的路径
facilities=db.relationship("Facility", secondary=house_facility) # 房屋的设施
images=db.relationship("HouseImage") # 房屋的图片
orders=db.relationship("Order", backref="house") # 房屋的订单
class Facility(BaseModel, db.Model):
"""设施信息"""
__tablename__= "ih_facility_info"
id=db.Column(db.Integer, primary_key=True) # 设施编号
name=db.Column(db.String(32), nullable=False) # 设施名字
class HouseImage(BaseModel, db.Model):
"""房屋图片"""
__tablename__= "ih_house_image"
id=db.Column(db.Integer, primary_key=True)
house_id=db.Column(db.Integer, db.ForeignKey("ih_house_info.id"), nullable=False) # 房屋编号
url=db.Column(db.String(256), nullable=False) # 图片的路径
class Order(BaseModel, db.Model):
"""订单"""
__tablename__= "ih_order_info"
id=db.Column(db.Integer, primary_key=True) # 订单编号
user_id=db.Column(db.Integer, db.ForeignKey("ih_user_profile.id"), nullable=False) # 下订单的用户编号
house_id=db.Column(db.Integer, db.ForeignKey("ih_house_info.id"), nullable=False) # 预订的房间编号
begin_date=db.Column(db.DateTime, nullable=False) # 预订的起始时间
end_date=db.Column(db.DateTime, nullable=False) # 预订的结束时间
days=db.Column(db.Integer, nullable=False) # 预订的总天数
house_price=db.Column(db.Integer, nullable=False) # 房屋的单价
amount=db.Column(db.Integer, nullable=False) # 订单的总金额
status=db.Column( # 订单的状态
db.Enum(
"WAIT_ACCEPT", # 待接单,
"WAIT_PAYMENT", # 待支付
"PAID", # 已支付
"WAIT_COMMENT", # 待评价
"COMPLETE", # 已完成
"CANCELED", # 已取消
"REJECTED" # 已拒单
),
default="WAIT_ACCEPT", index=True)
comment=db.Column(db.Text) # 订单的评论信息或者拒单原因
四、数据库迁移
python manage.py db init
python manage.py db migrate -m 'init tables'
五、项目架构
5.1项目的整体目录如下图所示:
5.2 ihome的目录结构:
六、项目整体架构配置:
6.1主目录下config.py文件主要是配置数据库和redis缓存
6.2 ihome目录下的__init__.py文件主要是创建app对象:
6.3主目录下的manage.py用来运行项目代码:
6.4 ihome目录下的web_html.py文件用来配置静态文件的蓝图
6.5在ihome/utils/commons.py文件中定义正则转换器:
现在我们来看看为什么在Vue中推荐注册组件是使用驼峰写法, 在了解这个之前,详细大家应该都能明白为什么在Vue中, 局部组件的使用频率高于全局组件.
推荐使用驼峰写法也是和局部组件有关系
我们先看一个示例
<div id="app">
<!-- 3. 在注册了局部组件的实例中使用局部组件 -->
<my-component></my-component>
</div>
<script>
// 1. 创建局部组件的选项对象
let MyComponent={
template: `
<div>
<h2>局部组件</h2>
</div>
`,
}
const vm=new Vue({
el:"#app",
// 2. 将选项对象注册为局部组件
components: {
"my-component": MyComponent
}
})
</script>
通过前面的学习,这个例子应该已经熟悉了,
示例中,不使用驼峰命名组件,依然可以正常运行,那么为什么组件名还要推荐使用驼峰命名
那先看下面几点:
好此时大家想一想,如果组件名和选项对象的变量名一样会怎么样
是不是就会变成如下的写法
const vm=new Vue({
el:"#app",
components: {
MyComponent: MyComponent
}
})
在思考一下,我们之前在学习ES6的时候讲过在定义对象的是有一种简便写法. 当属性跟值长得一样时,就可以简写
因此这里我们就可以简写为
const vm=new Vue({
el:"#app",
components: {
MyComponent
}
})
所以,为什么推荐使用驼峰写法,这里就应该可以看出端倪了, 如果我们定义组件的组件名使用驼峰写法,也就是和需要被注册为组件的选项对象一致是,我们注册组件将变得简单
如果此时需要注册n个组件,就可以如下写法
const vm=new Vue({
el:"#app",
components: {
one,two,three,four.....
}
})
是不是感觉很优雅,
总结:
尽管语法糖简化了组件注册,但在template选项中拼接HTML元素比较麻烦,这也导致了HTML和JavaScript的高耦合性。
Vue.js提供了两种方式将定义在JavaScript中的HTML模板分离出来。
在使用script标签将template模板分离出来时,要注意script标签的type类型选择,
<div id="app">
<!-- 3. 使用组件 -->
<my-component></my-component>
</div>
<!-- 组件模板 -->
<!-- 注意: 如果不添加type属性,可能会显示效果,但是会报错, -->
<script id="myComponent" type="text/x-template">
<div>
<h2>我想被创建为局部组件</h2>
</div>
</script>
<script>
// 1. 创建组件选项对象
let MyComponent={
// 此时的模板template的值就是一个选择器
template: "#myComponent",
}
const vm=new Vue({
el:"#app",
// 2. 注册组件
components: {
"my-component": MyComponent
}
})
</script>
template选项现在不再是HTML元素,而是一个id,Vue.js根据这个id查找对应的元素,然后将这个元素内的HTML作为模板进行编译。
注意:
使用<script> 标签时,type指定为text/x-template,意在告诉浏览器这不是一段js脚本,浏览器在解析HTML文档时会忽略<script>标签内定义的内容。
如果使用<template>标签,则不需要指定type属性。
<div id="app">
<!-- 3. 使用组件 -->
<my-component></my-component>
</div>
<!-- 组件模板 -->
<!-- template标签不需要指定type,标签的本意就是告诉浏览器这是模板 -->
<template id="myComponent">
<div>
<h2>我想被创建为局部组件</h2>
</div>
</template>
<script>
// 1. 创建组件选项对象
let MyComponent={
// 此时的模板template的值就是一个选择器
template: "#myComponent",
}
const vm=new Vue({
el:"#app",
// 2. 注册组件
components: {
"my-component": MyComponent
}
})
</script>
在理解了组件的创建和注册过程后,我建议使用<script>或<template>标签来定义组件的HTML模板。
这使得HTML代码和JavaScript代码是分离的,便于阅读和维护。
另外,在Vue.js中,可创建.vue后缀的文件,.vue文件就是一个组件,成为单文件组件,这个内容我会在后面的文章介绍。
组件中的选项基本与实例选项对象一致, 但是有两个选项是特例,分别为el 和 data 属性
这个说的特例是指组件的选项对象和实例选项对象使用的不同
el属性的作用我们都了解了, 就是在实例中使用,决定Vue需要接管的DOM元素. 组件是在实例中使用,所以不需要el属性,
如果一个选项对象被注册为组件,添加el属性就会造成报错,
// 1. 创建局部组件的选项对象
let MyComponent={
el:"",
template: "#myComponent",
}
const vm=new Vue({
el:"#app",
// 2. 将选项对象注册为局部组件
components: {
"my-component": MyComponent
}
})
// 会报错, 告诉你el属性只能在实例中使用
可以通过报错信息了解到, el选项只能使用在new创建的vue实例上
组件中的data选项必须是一个函数,返回一个数据对象.
为什么需要是一个函数,而不可以像Vue实例的选项对象一样是一个对象呢
不能使用对象的原因:
这样就会出现问题,看代码:
示例代码如下:
<div id="app">
<!-- 3. 使用组件 -->
<my-component></my-component>
<my-component></my-component>
<my-component></my-component>
</div>
<template id="myComponent">
<div>
<h2>组件{{num}}</h2>
<button @click="handleClick">点击+1</button>
</div>
</template>
<script>
// 组件共享数据
let data={
num : 10
}
// 1. 选项对象
let MyComponent={
template: "#myComponent",
data: function(){
return data
},
methods: {
handleClick(){
this.num++
}
}
}
const vm=new Vue({
el:"#app",
// 2. 注册组件
components: {
"my-component": MyComponent
}
})
</script>
示例结果:
实例说明:
通过案例我们就会发现如果所有的组件都共享数据,当有一个组件中的数据发生了变化,所有的组件显示的数据都会发生变化, 这不是我们想要的,
所以组件的data数据属性才会需要函数,每次初始化化的时候都会执行函数,将返回的结果作为组件的数据,
这样每次执行函数都会给每个组件创建一个独立的数据
<div id="app">
<!-- 3. 使用组件 -->
<my-component></my-component>
<my-component></my-component>
<my-component></my-component>
</div>
<template id="myComponent">
<div>
<h2>组件{{num}}</h2>
<button @click="handleClick">点击+1</button>
</div>
</template>
<script>
// 1. 选项对象
let MyComponent={
template: "#myComponent",
data: function(){
// 每次函数执行都会返回新的数据,非共享数据
return {
num : 10
}
},
methods: {
handleClick(){
this.num++
}
}
}
const vm=new Vue({
el:"#app",
// 2. 注册组件
components: {
"my-component": MyComponent
}
})
</script>
显示结果:
通过示例就会发现, 当一个组件数据发生改变的时候, 其他组件的数据不会变化,因为每个组件都有自己独立的数据
当注册组件 (或者 prop) 时,可以使用 kebab-case (短横线分隔命名)、camelCase (驼峰式命名) 或 PascalCase (单词首字母大写命名)。
// 在组件定义中
components: {
// 使用 kebab-case 注册
'kebab-cased-component': { /* ... */ },
// 使用 camelCase 注册 俗称小驼峰
'camelCasedComponent': { /* ... */ },
// 使用 PascalCase 注册 俗称大驼峰
'PascalCasedComponent': { /* ... */ }
}
在 HTML 模板中,请使用 kebab-case:
<!-- 在 HTML 模板中始终使用 kebab-case -->
<kebab-cased-component></kebab-cased-component>
<camel-cased-component></camel-cased-component>
<pascal-cased-component></pascal-cased-component>
这个问题已经在前面认真探讨过了,这里不再详细阐述
通过上面上的例子我们已经了解了组件的使用,就是把组件名当做自定义标签使用,但是这种使用方法有的时候也会出现问题,
通过下面的实例了解组件标签解析是发生的问题.
例如:实例代码如下
<div id="app">
<!-- 3. 使用组件 -->
<table>
<tbody>
<row></row>
<row></row>
<row></row>
</tbody>
</table>
</div>
<!-- 组件模板 -->
<template id="myComponent">
<tr>
<td>内容</td>
<td>123</td>
</tr>
</template>
<script>
// 1. 组件选项对象
let MyComponent={
template: "#myComponent",
}
// 2. 注册组件
const vm=new Vue({
el:"#app",
components: {
"row": MyComponent
}
})
</script>
显示结果:
实例中的写法,在查看代码结构时,发现子组件的tr标签并不在tbody里,
原因在与浏览器规范中tbody标签里面必须放tr标签,但是我们放的是row自定义标签,所以在解析的时候就会出问题
诸如此类的还有ul,ol标签里只能放li标签, select标签中只能放option,这些都是需要注意的事项
那么我们怎么解决这类问题呢?
vue允许我们使用is属性来使用组件, is属性的值是组件名
所以可以采用is属性来制定组件
示例代码如下:
<div id="app">
<!-- 3. 使用组件 -->
<table>
<tbody>
<tr is="row"></tr>
<tr is="row"/>
<tr is="row"/>
</tbody>
</table>
</div>
显示结果:
此时我们就会发现不仅结果没问题, 编译后标签的嵌套也没有什么问题,
因为我们是按照标准在tbody标签中使用的是tr标签,只不过通过is属性将tr标签替换为了组件row的模板标签.
此时tr标签中没有嵌套内容,所有使用单标签,双标签都可以
*请认真填写需求信息,我们会在24小时内与您取得联系。