5.1 iHRM人力资源 - 员工管理

iHRM人力资源 - 员工管理 - 左右侧主体展示

一、页面主体结构

最终效果图如下所示

典型的,左边是树,右边是表的页面结构

image-20240331223459878

结构如下图所示

代码

image-20240331225859741

下面的代码其实就是将页面分成了左右两个部分

<template>
  <div class="container">
    <div class="app-container">
      <div class="left">
        <el-input style="margin-bottom:10px" type="text" prefix-icon="el-icon-search" size="small" placeholder="输入员工姓名全员搜索" />
        <!-- 树形组件 -->
      </div>
      <div class="right">
        <el-row class="opeate-tools" type="flex" justify="end">
          <el-button size="mini" type="primary">添加员工</el-button>
          <el-button size="mini">excel导入</el-button>
          <el-button size="mini">excel导出</el-button>
        </el-row>
        <!-- 表格组件 -->
        <!-- 分页 -->
      </div>
    </div>
  </div>
</template>

<script>
export default {
  name: 'Employee'
}
</script>

<style lang="scss" scoped>
.app-container {
  background: #fff;
  display: flex;
  .left {
    width: 280px;
    padding: 20px;
    border-right: 1px solid #eaeef4;
  }
  .right {
    flex: 1;
    padding: 20px;
    .opeate-tools {
      margin:10px ;
    }
    .username {
      height: 30px;
      width: 30px;
      line-height: 30px;
      text-align: center;
      border-radius: 50%;
      color: #fff;
      background: #04C9BE;
      font-size: 12px;
      display:inline-block;
    }
  }
}

</style>

效果图如下所示

image-20240331224220623

二、页面左树加载

效果图如下所示

image-20240331224608213

流程如下图所示

image-20240331225226378

2.1 加载左树数据

我们之前写过获取组织架构列表的接口

image-20240331225805057

/**
 *  获取组织架构数据
 */
export function getDepartment() {
  // request发送登录请求,会得到一个promise结果并将其返回
  return request({
    // 请求地址
    url: '/company/department',
    // 请求方式
    method: 'GET'
    // 请求参数,但是这里没有请求参数
    // data: Data
    // 在ES6中上面data: Data可以简写为 data
  })
}

左侧树页面

      <div class="left">
        <el-input style="margin-bottom:10px" type="text" prefix-icon="el-icon-search" size="small"
                  placeholder="输入员工姓名全员搜索"
        />
        <!-- 树形组件 -->
        <!-- :default-expand-all="true" 展开所有节点-->
        <!-- highlight-current高亮所选节点-->
        <el-tree :data="depts" :props="defaultProps" :default-expand-all="true" :highlight-current="true"></el-tree>
      </div>

数据

data() {
  return {
    // 组织架构数据
    depts: [],
    // 树形结构中的数据
    defaultProps: {
      label: 'name',
      children: 'children'
    }
  }
}

方法

created() {
  this.getDepartment()
},
methods: {
  // 获取组织架构数据
  async getDepartment() {
    // 递归方法将列表转换成树形结构
    this.depts = transListToTreeData(await getDepartment(), 0)
  }
}

工具类方法

/**
 * 列表数据转树形数据
 * rootValue: 其实就是pid(父id)
 */
export function transListToTreeData(list, rootValue) {
  const arr = []
  list.forEach(item => {
    if (item.pid === rootValue) {
      // 找到了匹配的节点
      arr.push(item)
      // 当前节点的id和当前节点的字节点的pid相等
      // 下面的方法其实就是找当前节点的子节点
      const children = transListToTreeData(list, item.id) // 找到的节点的子节点
      item.children = children // 将子节点赋值给当前节点
      // 我们先push再赋值childern也没关系,因为是一个对象,地址是一样的
    }
  })
  return arr
}

2.2 记录树的切换节点

默认选中“传智教育”节点,并且会在右侧展示“传智教育”下面所有员工的数据

当我们切换左侧节点的时候,右侧展示的员工数据也会发生改变

这就是左树右表的一个联动效果

image-20240401202750944

实现思路

其实也很简单

当我们获取获取数据的时候,找到“传智教育”节点,并且把这个节点记录下来,并且选中这个节点,然后查询“传智教育”下的员工数据

image-20240401203243392

  1. 获取首个节点并记录
data() {
  return {
    // 组织架构数据
    depts: [],
    // 存储查询参数
    queryParams: {
      // 当前选中节点id
      departmentId: null
    }
      ........
  }
},
methods: {
  // 获取组织架构数据
  async getDepartment() {
    // 递归方法将列表转换成树形结构
    this.depts = transListToTreeData(await getDepartment(), 0)
    // 将树形结构的首节点id赋值给departmentId
    this.queryParams.departmentId = this.depts[0].id
  }
}
  1. 选中首节点

树结构,多了 ref=“deptTree” node-key="id"方法

我们需要调用el-tree组件中的setCurrentKey方法来选中某个节点

<!-- 树形组件 -->
<!-- :default-expand-all="true" 展开所有节点-->
<!-- highlight-current高亮所选节点-->
<!--  node-key树形,每个树节点用来作为唯一标识的属性,是唯一的-->
<el-tree :data="depts" :props="defaultProps"
         :default-expand-all="true" :highlight-current="true"
         ref="deptTree" node-key="id"
></el-tree>

方法

// 获取组织架构数据
async getDepartment() {
  // 递归方法将列表转换成树形结构
  this.depts = transListToTreeData(await getDepartment(), 0)
  // 将树形结构的首节点id赋值给departmentId
  this.queryParams.departmentId = this.depts[0].id
  // 选中首个节点
  // this.$refs.deptTree.setCurrentKey(this.queryParams.departmentId) 在这里执行是不合适的
  // 这个地方涉及到一个同步异步的问题
  // 树组件的渲染是异步的,我们需要等渲染完毕后再选中首个节点
  this.$nextTick(() => {
    // 此时意味着树的渲染已经完毕了
    this.$refs.deptTree.setCurrentKey(this.queryParams.departmentId)
  })
}
  1. 选中节点的时候记录节点

我们切换节点的时候怎么记录节点呢?

可以看一下element-ui,看看树形组件是怎么去监听节点切换事件的

下面两个事件都是当前节点变化时触发,用哪个都行

image-20240401210253188

image-20240401210309374

事件

<!--树形组件-->
<!--:default-expand-all="true" 展开所有节点-->
<!--highlight-current高亮所选节点-->
<!--node-key树形,每个树节点用来作为唯一标识的属性,是唯一的-->
<!--@current-change监听,当前节点变化时触发此事件-->
<el-tree
  ref="deptTree"
  node-key="id"
  :data="depts"
  :props="defaultProps"
  :default-expand-all="true"
  :highlight-current="true"
  @current-change="selectNode"
></el-tree>

方法

// 参数1:当前节点数据;
selectNode(node) {
  // 当前选中节点的id
  this.queryParams.departmentId = node.id
}

三、右侧员工列表

实现右侧员工列表

如下图所示

image-20240401211032997

3.1 列表结构

      <div class="right">
        <el-row class="opeate-tools" type="flex" justify="end">
          <el-button size="mini" type="primary">添加员工</el-button>
          <el-button size="mini">excel导入</el-button>
          <el-button size="mini">excel导出</el-button>
        </el-row>
        <!-- 表格组件 -->
        <el-table>
          <!--表格中放列-->
          <!--将头像居中操作-->
          <el-table-column label="头像" align="center"></el-table-column>
          <el-table-column label="姓名"></el-table-column>
          <!--sortable表示这一列可以排序来查看-->
          <el-table-column label="手机号" sortable></el-table-column>
          <el-table-column label="工号" sortable></el-table-column>
          <el-table-column label="聘用形式"></el-table-column>
          <el-table-column label="部门"></el-table-column>
          <el-table-column label="入职时间" sortable></el-table-column>
          <el-table-column label="操作" width="280px">
            <template>
              <!--type="text"表示按钮是链接的形式-->
              <el-button size="mini" type="text">查看</el-button>
              <el-button size="mini" type="text">角色</el-button>
              <el-button size="mini" type="text">删除</el-button>
            </template>
          </el-table-column>
        </el-table>
        <!--type="flex" justify="end"分页设置在尾部-->
        <!--align="middle"表示水平居中-->
        <el-row type="flex" justify="end" align="middle" style="height:60px">
          <!--分页-->
           <el-pagination layout="total,prev,pager,next" :total="1000"></el-pagination>
        </el-row>
      </div>

页面效果

image-20240401224754532

3.2 员工数据获取

首次加载的时候获取员工数据,切换部门的时候加载此部门中员工数据

image-20240401224943565

api请求

/**
 * 获取员工列表
 */
export function getEmployeeList(params) {
  return request({
    url: '/sys/user',
    params: params // 地址参数,查询参数
  })
}

页面

<!-- 表格组件 -->
<!--使用data绑定数据-->
<el-table :data="list">
  <!--表格中放列-->
  <!--将头像居中操作-->
  <el-table-column prop="staffPhoto" label="头像" align="center"></el-table-column>
  <el-table-column prop="username" label="姓名"></el-table-column>
  <!--sortable表示这一列可以排序来查看-->
  <el-table-column prop="mobile" label="手机号" sortable></el-table-column>
  <el-table-column prop="workNumber" label="工号" sortable></el-table-column>
  <el-table-column prop="formOfEmployment" label="聘用形式"></el-table-column>
  <el-table-column prop="departmentName" label="部门"></el-table-column>
  <el-table-column prop="timeOfEntry" label="入职时间" sortable></el-table-column>
  <el-table-column label="操作" width="280px">
    <template>
      <!--type="text"表示按钮是链接的形式-->
      <el-button size="mini" type="text">查看</el-button>
      <el-button size="mini" type="text">角色</el-button>
      <el-button size="mini" type="text">删除</el-button>
    </template>
  </el-table-column>
</el-table>

方法

methods: {
  // 获取组织架构数据
  async getDepartment() {
    // 递归方法将列表转换成树形结构
    this.depts = transListToTreeData(await getDepartment(), 0)
    // 将树形结构的首节点id赋值给departmentId
    this.queryParams.departmentId = this.depts[0].id
    // 选中首个节点
    // this.$refs.deptTree.setCurrentKey(this.queryParams.departmentId) 在这里执行是不合适的
    // 这个地方涉及到一个同步异步的问题
    // 树组件的渲染是异步的,我们需要等渲染完毕后再选中首个节点
    this.$nextTick(() => {
      // 此时意味着树的渲染已经完毕了
      this.$refs.deptTree.setCurrentKey(this.queryParams.departmentId)
    })
    // 这个时候已经记录了id,根据部门的id获取部门人员
    this.getEmployeeList()
  },
  // 参数1:当前节点数据
  selectNode(node) {
    // 当前选中节点的id
    this.queryParams.departmentId = node.id
    // 这个时候已经记录了id,根据部门的id获取部门人员
    this.getEmployeeList()
  },
  async getEmployeeList() {
    // 解构出row数据
    const { rows } = await getEmployeeList(this.queryParams)
    this.list = rows
  }
}

效果

但是我们的头像列和聘用形式列的值不对劲,之后改一下

image-20240401231141419

3.3 头像和聘用形式

要实现下图的效果

image-20240401231323720

内容

image-20240401231422658

  1. 处理头像
<el-table-column prop="staffPhoto" label="头像" align="center">
  <!--自定义插槽-->
  <template v-slot="{row}">
    <!--头像组件-->
    <!--当此用户有头像的时候才显示此组件-->
    <el-avatar v-if="row.staffPhoto" :src="row.staffPhoto" :size="30"></el-avatar>
    <!--当没有头像的时候,就是用户名中的第一次字当头像-->
    <span v-else class="username">{{ row.username.charAt(0) }}</span>
  </template>
</el-table-column>

样式

<style lang="scss" scoped>
.app-container {
  background: #fff;
  display: flex;

  .left {
    width: 280px;
    padding: 20px;
    border-right: 1px solid #eaeef4;
  }

  .right {
    flex: 1;
    padding: 20px;

    .opeate-tools {
      margin: 10px;
    }

    .username {
      height: 30px;
      width: 30px;
      line-height: 30px;
      text-align: center;
      border-radius: 50%;
      color: #fff;
      background: #04C9BE;
      font-size: 12px;
      display: inline-block;
    }
  }
}

</style>

效果

image-20240401232101533

  1. 处理聘用形式
<el-table-column prop="formOfEmployment" label="聘用形式">
  <template v-slot="{row}">
    <span v-if="row.formOfEmployment===1">正式</span>
    <span v-else-if="row.formOfEmployment===2">非正式</span>
    <span v-else></span>
  </template>
</el-table-column>

效果

image-20240401232358599

3.4 数据分页

实现员工数据分页

image-20240407231503513

image-20240407232216522

  1. 分页总数
data() {
  return {
      .....
    // 记录当前查询的员工总数
    total: 0
  }
}

初始化总数

async getEmployeeList() {
  // 解构出row数据
  const { rows, total } = await getEmployeeList(this.queryParams)
  this.list = rows
  this.total = total
}

分页组件

<!--type="flex" justify="end"分页设置在尾部-->
<!--align="middle"表示水平居中-->
<el-row type="flex" justify="end" align="middle" style="height:60px">
  <!--分页-->
  <el-pagination layout="total,prev,pager,next" :total="total">
  </el-pagination>
</el-row>
  1. 配置每页的条数和当前页码

查询参数

// 存储查询参数
queryParams: {
  // 当前选中节点id
  departmentId: null,
  // 当前页码
  page: 1,
  // 每页有多少数据
  pagesize: 10
},

绑定到分页组件上

<!--align="middle"表示水平居中-->
<el-row type="flex" justify="end" align="middle" style="height:60px">
  <!--分页-->
  <el-pagination
    layout="total,prev,pager,next"
    :total="total"
    :current-page="queryParams.page"
    :page-size="queryParams.pagesize"
  >
  </el-pagination>
</el-row>
  1. 监听切换页码的事件
<!--分页-->
<el-pagination
  layout="total,prev,pager,next"
  :total="total"
  :current-page="queryParams.page"
  :page-size="queryParams.pagesize"
  @current-change="changePage"
>

实现方法

async getEmployeeList() {
  // 解构出row数据
  const { rows, total } = await getEmployeeList(this.queryParams)
  this.list = rows
  this.total = total
},
// 切换页码的事件
// 参数回调1:当前页,参数
changePage(newPage) {
  this.queryParams.page = newPage
  // 重新查询员工
  this.getEmployeeList()
}
  1. 切换部门的时候,要把queryParams里面的page及pagesize充值
// 参数1:当前节点数据
selectNode(node) {
  // 将页码设置为第一页
  this.queryParams.page = 1
  // 当前选中节点的id
  this.queryParams.departmentId = node.id
  // 这个时候已经记录了id,根据部门的id获取部门人员
  this.getEmployeeList()
},
  1. 将"Total"英文换成中文

image-20240407234035509

首先要打开入口文件,将下面第二个参数去掉即可

image-20240407234156408

如下图所示

// 全局注册Element-ui组件
Vue.use(ElementUI)
// 如果想要中文版 element-ui,按如下方式声明
// Vue.use(ElementUI)

3.5 员工模糊搜索

image-20240408193351462

我们要监听输入框

并且搜索的时候,我们应该把我们分页设置为第一页,因为只要是查询就要从第一页开始看,之后查询员工的数据即可

并且在搜索的此处做了一个优化,在用户输入完成一段时间后再进行搜索(防抖处理)

image-20240408193704635

  1. 绑定查询参数
<el-input
  v-model="queryParams.keyword"
  style="margin-bottom:10px"
  type="text"
  prefix-icon="el-icon-search"
  size="small"
  placeholder="输入员工姓名全员搜索"
/>
// 存储查询参数
queryParams: {
  // 当前选中节点id
  departmentId: null,
  // 当前页码
  page: 1,
  // 每页有多少数据
  pagesize: 10,
  // 模糊搜索查询
  keyword: ''
},
  1. 监听input绑定的值的改变

由于值的改变触发了一个事件

但是下面这个不合适,因为我们向输入完成之后就触发

image-20240408194157021

我们选择如下所示的方法,当值发生变化的时候就触发

image-20240408194239763

<el-input
  v-model="queryParams.keyword"
  style="margin-bottom:10px"
  type="text"
  prefix-icon="el-icon-search"
  size="small"
  placeholder="输入员工姓名全员搜索"
  @input="changeValue"
/>

对应的方法如下所示

async getEmployeeList() {
  // 解构出row数据
  const { rows, total } = await getEmployeeList(this.queryParams)
  this.list = rows
  this.total = total
},
// 当input框中输入的内容改变时触发此方法
changeValue() {
   // 修改查询第一页
   this.queryParams.page = 1
   // 调用封装好的方法
   this.getEmployeeList()
}
  1. 优化:完成防抖处理

加了一个定时器

// 当input框中输入的内容改变时触发此方法
changeValue() {
  // 单位时间内只执行最后一次
  // this的实例上赋值了一个timer的属性(此时timer并不是响应式数据,和视图没关系)
  clearTimeout(this.timer) // 清理上一次的定时器
  this.timer = setTimeout(() => {
    // 修改查询第一页
    this.queryParams.page = 1
    // 调用封装好的方法
    this.getEmployeeList()
  }, 700)
}
<think>嗯,用户想知道如何用LoadRunner对iHRM人力资源管理系统进行性能测试。首先,我需要回忆一下LoadRunner的基本使用流程,包括脚本录制、场景设计、执行测试和结果分析这些步骤。然后结合iHRM系统的特点,比如可能有登录、员工管理、考勤、薪资这些模块,需要考虑这些关键接口的性能测试。 用户之前的问题提到过iHRM项目实战,可能他们已经在做接口测试,现在需要扩展到性能测试。引用[1]和[2]里提到了JMeter的相关内容,但用户明确要求使用LoadRunner,所以需要对比两者的差异,比如LoadRunner的协议选择、脚本录制方式等。 接下来需要分步骤说明,首先是测试需求分析,确定并发用户数、响应时间等指标。然后协议选择,因为iHRM是Web系统,可能用HTTP/HTTPS协议。脚本录制部分要详细说明如何操作VuGen,添加事务和检查点,参数化登录数据,关联动态值,比如会话ID。 场景设计部分,要提到Controller的设置,比如逐步增加负载,持续时间,监控资源。执行测试时要注意观察服务器指标,比如CPU和内存使用率。最后的结果分析需要生成报告,关注关键指标是否达标。 可能还需要考虑关联动态参数,比如登录后的token,这在引用[2]里提到的接口概念和关联使用相关。此外,引用[3]中的性能测试示例提到并发用户和响应时间,这些指标需要包含在测试场景中。 用户可能遇到的问题包括脚本中的动态参数处理,比如如何关联会话ID,或者参数化多个用户账号。需要给出具体的解决方案,比如使用web_reg_save_param函数。另外,测试场景配置可能需要设置不同的负载模式,比如峰值测试或压力测试。 最后,生成相关问题时要围绕性能测试的深入分析、优化方法、工具对比和应用场景,帮助用户进一步扩展知识。确保引用正确,比如在提到参数化时引用[1]中的外部数据文件,或者关联技术引用[2]和[3]中的内容。</think>以下是如何使用LoadRunner对iHRM人力资源管理系统进行性能测试的详细配置方法,结合性能测试最佳实践和iHRM系统特点: ### 一、测试需求分析(基于iHRM特性) 1. **关键接口识别**:登录认证、员工档案管理、考勤统计、薪资计算等核心模块 2. **性能指标设定**: - 并发用户数:根据企业规模设定(如$C=500$用户) - 响应时间:关键操作≤2秒(TP90) - 事务成功率:≥99.9% - 系统资源占用:CPU≤80%,内存≤85% ### 二、LoadRunner配置流程 #### 1. 协议选择 ```bash Web(HTTP/HTML)协议 # 适用于B/S架构的iHRM系统[^2] ``` #### 2. 脚本录制(VuGen) ```C Action() { web_url("iHRM_login", "URL=http://ihrm.example.com/login", "TargetFrame=", LAST); web_submit_data("auth", "Action=http://ihrm.example.com/api/auth", "Method=POST", ITEMDATA, "Name=username", "Value={username}", ENDITEM, "Name=password", "Value={password}", ENDITEM, LAST); lr_start_transaction("薪资查询"); web_custom_request("salary_query", "URL=http://ihrm.example.com/api/salary", "Method=GET", "EncType=application/json", LAST); lr_end_transaction("薪资查询", LR_AUTO); return 0; } ``` #### 3. 参数化配置(以登录为例) - 使用`Parameters List`加载CSV文件[^1] - 动态参数替换: $$ \text{username} = \text{user}_{seq}@ihrm.com $$ $$ \text{password} = \text{AES256}(\text{pass}_{rand}) $$ #### 4. 关联设置(动态会话处理) ```C web_reg_save_param_ex( "ParamName=Cookie", "LB=Set-Cookie: ", "RB=;", SEARCH_FILTERS, "Scope=Headers", LAST); ``` ### 三、场景设计(Controller配置) 1. **负载模式**: - 渐进加载:每30秒增加50用户 - 峰值保持:持续15分钟 - 阶梯式退出 2. **监控配置**: ```mermaid graph TD A[Load Generator] --> B[应用服务器] A --> C[数据库服务器] B --> D(Nginx监控) C --> E(MySQL慢查询日志) ``` ### 四、测试执行注意事项 1. **测试数据隔离**:使用独立测试数据库,避免污染生产数据 2. **异常处理**:添加检查点验证HTTP状态码: ```C web_reg_find("Text=HTTP/1.1 200 OK", "SaveCount=StatusCode_200", LAST); ``` ### 五、结果分析要点 1. **关键指标可视化**: $$ \text{Throughput} = \frac{\text{Total Transactions}}{\text{Test Duration}} $$ 2. **错误类型分析**: - 504超时错误 - 500内部错误 - 数据库连接池耗尽
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

我爱布朗熊

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值