[头歌][PHP][MySQL] PHP语言之MySQL操作

声明

本文章为公开免费文章,授权非商业转载,如果本文章收费,纯属CSDN平台行为,和作者本人无关。

第1关:PHP对MySQL的基本操作

任务描述

随着公司规模的扩大,老板希望能够利用数据库技术对员工与部门的信息进行管理,而这个任务正好落到了你的身上。你选择了PHP+MySQL的组合来完成这个任务,为了能让老板对你的工作有一个大致的了解,你准备第一步进行数据库的建立和员工以及部门信息数据的录入,并将录入的数据显示在页面中便于老板查看。为了完成这一步,你需要学习以下关于PHP与MySQL的知识。

相关知识

PHP与MySQL的连接

在PHP中,连接数据库的方法是:

$link=new mysqli($server, $user, $pwd);

server规定要连接的服务器,例如"hostname:port"。
userpwd分别为连接数据库时的用户名和密码。

PHP数据库与表的创建

CREATE DATABASE 语句用于在 MySQL 中创建数据库。
例如:

$sql = "CREATE DATABASE myDB;";
$link->query($sql);
$link->query(set names utf8);//这可以设置所使用数据库的字符集

CREATE TABLE 语句用于创建 MySQL 表。
创建表前,我们需要使用 use myDB 来选择要操作的数据库。

create table A (
  d_id int unsigned primary key auto_increment,
  d_name varchar(20) not null
)charset=utf8;

数据类型指定列可以存储什么类型的数据,请注意,最后的charset可以设置表的字符集,在本关中应设置为utf8。
在设置了数据类型后,你可以为每个列指定其他选项的属性:

  • NOT NULL - 每一行都必须含有值(不能为空),null 值是不允许的。
  • DEFAULT value - 设置默认值。
  • UNSIGNED - 使用无符号数值类型,0 及正数。
  • AUTO INCREMENT - 设置 MySQL 字段的值在新增记录时每次自动增长 1。
  • PRIMARY KEY - 设置数据表中每条记录的唯一标识。 通常列的 PRIMARY KEY 设置为 ID 数值,与 AUTO_INCREMENT 一起使用。
    每个表都应该有一个主键(本列为 “d_id” 列),主键必须包含唯一的值。

PHP对数据库进行的基本操作

在创建完数据库和表后,我们可以向表中添加数据。
以下为一些语法规则:

  • PHP 中 SQL 查询语句必须使用引号。
  • 在 SQL 查询语句中的字符串值必须加引号。
  • 数值的值不需要引号。
  • NULL 值不需要引号。
插入数据

INSERT INTO 语句通常用于向 MySQL 表添加新的记录:

INSERT INTO table_name (column1, column2, column3,...) VALUES
(value1, value2, value3,...),
(value1, value2, value3,...),
(value1, value2, value3,...);
读取数据

SELECT 语句用于从数据表中读取数据:

SELECT column_name(s) AS name
FROM table_name
WHERE column_name operator value
ORDER BY column_name(s) ASC|DESC;

其中,WHERE 子句用于提取满足指定标准的的记录。AS可以对选中的列重命名。
ORDER BY 关键词用于对记录集中的数据进行排序,默认对记录进行升序排序。如果你想降序排序,请使用 DESC 关键字。
另外,我们可以使用 * 号来读取所有数据表中的字段。例如:

SELECT * FROM emp AS emp2
WHERE emp.d_id = dept.d_id
ORDER BY emp2 ASC;

在PHP中,通常用一个变量来保存SELECT返回的数据表,例如:

$result=$link->query($sql);
$row=$result->fetch_assoc();
//fetch_assoc()可以将表的第一行数据返回,
//并且指针自动后移
//直到遍历完整个表,返回false

温馨提示:下一关可以借鉴这里的方法。

JOIN操作

这是一种在MySQL中将两个表关联起来的方法,它主要分为:

  1. 笛卡尔积 两表关联,把左表的列和右表的列通过笛卡尔积的形式表达出来。 select * from t1 join t2;
  2. 左连接 两表关联,左表全部保留,右表关联不上用null表示。 select * from t1 left join t2 on t1.id = t2.id;
  3. 右连接 右表全部保留,左表关联不上的用null表示。 select * from t1 right join t2 on t1.id =t2.id;
  4. 内连接 两表关联,保留两表中交集的记录。 select * from t1 inner join t2 on t1.id = t2.id; 此外,在进行连接时,可以通过ON关键字来指定连接条件: select * from t1 left join t2 on t1.id = t2.id;
删除数据

DELETE FROM 语句用于从数据库表中删除记录。

DELETE FROM table_name
WHERE some_column = some_value

注意: DELETE 语法中的 WHERE 子句。WHERE 子句规定了哪些记录需要删除。如果你想省去 WHERE 子句,所有的记录都会被删除!
DROP用于删除表和数据库。

DROP TABLE emp_dept;
DROP DATABASE test;
更改数据

UPDATE 语句用于更新数据库表中已存在的记录。

UPDATE table_name
SET column1=value, column2=value2,...
WHERE some_column=some_value

注意: UPDATE 语法中的 WHERE 子句。WHERE 子句规定了哪些记录需要更新。如果您想省去 WHERE 子句,所有的记录都会被更新!

编程要求

将右侧代码中相应的功能补全,具体要求:

  1. 实现数据库连接,其中url为’127.0.0.1’,用户名’root’,密码’123123’。
  2. 创建部门信息表emp_dept,包含两个属性,d_id表示部门编号,将其声明为无符号的int类型,并作为该表的主键且为自增形式。d_name表示部门名称,将其声明为varchar类型。
  3. 向部门信息表中插入以下数据: (1, ‘开发部’), (2, ‘媒体部’), (3, ‘人事部’),(4, ‘后勤部’), (5, ‘市场部’), (6, ‘运维部’), (7, ‘销售部’)
  4. 修改用户表emp_info,将用户表 e_id int unsigned primary key auto_increment, e_name varchar(20) not null, d_name varchar(20) not null, date_of_birth timestamp not null, date_of_entry timestamp not null 中的部门名称d_name字段删除,再添加部门ID字段 d_id int unsigned not null
  5. 向用户表中添加信息: (1, ‘小红’, 1, ‘2015-4-9 17:51:00’, ‘2015-4-9 17:52:00’), (2, ‘李四’, 5, ‘2008-4-3 13:33:00’, ‘2013-10-24 17:53:00’), (3, ‘王五’, 4, ‘2008-4-3 13:33:00’, ‘2015-4-21 13:33:00’), (4, ‘赵六’, 4, ‘2008-4-3 13:33:00’, ‘2015-3-20 17:54:00’), (5, ‘小兰’, 2, ‘1989-5-4 17:33:00’, ‘2012-6-18 17:54:00’), (6, ‘小新’, 5, ‘1993-9-18 17:36:00’, ‘2015-2-28 17:36:00’), (7, ‘小白’, 2, ‘1991-10-17 17:37:00’, ‘2014-8-16 17:37:00’), (8, ‘小智’, 7, ‘1987-6-20 17:37:00’, ‘2015-1-10 17:38:00’), (9, ‘大头’, 6, ‘1991-2-14 08:49:00’, ‘2014-7-12 08:49:00’), (10, ‘小明’, 3, ‘1991-2-14 08:49:00’, ‘2015-3-4 09:10:00’), (11, ‘小刘’, 1, ‘1992-3-18 14:52:00’, ‘2014-7-21 09:00:00’);
  6. 进行两个表的连接查询: 为emp_info表定义别名emp,为emp_dept表定义别名dept,通过别名来标识要获取的字段。使用left join进行左连接查询。最后需要使用on关键字标识出两个表的关联条件,即部门id。 要查询的字段有 emp.e_id,emp.e_name,emp.date_of_birth,emp.date_of_entry,dept.d_name

测评说明

平台将自动编译补全后的代码,并根据程序的输出判断是否正确。

代码

<?php
/**
 * 初始化数据库连接
 */

require 'public_function.php';
//创建对象,连接数据库
/*****begin*********/
$link = new mysqli('127.0.0.1', 'root', '123123');
/*****end*********/

//判断数据库连接是否成功,如果不成功则显示错误信息并终止脚本继续执行
if($link->connect_error){
    die('连接数据库失败!'.$link->connect_error);
}
//设置字符集,选择数据库
$link->query("drop database if exists 'educoder';");
$link->query("create database educoder;");
$link->query('set names utf8;');
$link->query('use educoder;');

//创建部门信息表
/*****begin*********/
$sql = "CREATE TABLE emp_dept (
            d_id INT UNSIGNED PRIMARY KEY AUTO_INCREMENT,
            d_name VARCHAR(20) NOT NULL
        ) CHARSET=utf8;";
$link->query($sql);
/*****end*********/
//向部门信息表中插入数据
/*****begin*********/
$sql = "INSERT INTO emp_dept (d_id, d_name) VALUES 
        (1, '开发部'), 
        (2, '媒体部'), 
        (3, '人事部'),
        (4, '后勤部'), 
        (5, '市场部'), 
        (6, '运维部'), 
        (7, '销售部')";
$link->query($sql);
/*****end*********/

//修改用户表
/*****begin*********/
// 创建新的用户表
$sql = "CREATE TABLE emp_info (
            e_id INT UNSIGNED PRIMARY KEY AUTO_INCREMENT,
            e_name VARCHAR(20) NOT NULL,
            d_id INT UNSIGNED NOT NULL,
            date_of_birth TIMESTAMP NOT NULL,
            date_of_entry TIMESTAMP NOT NULL
        ) CHARSET=utf8;";
$link->query($sql);
/*****end*********/

//向其中添加用户数据
/*****begin*********/
$sql = "INSERT INTO emp_info (e_id, e_name, d_id, date_of_birth, date_of_entry) VALUES 
        (1, '小红', 1, '2015-04-09 17:51:00', '2015-04-09 17:52:00'), 
        (2, '李四', 5, '2008-04-03 13:33:00', '2013-10-24 17:53:00'), 
        (3, '王五', 4, '2008-04-03 13:33:00', '2015-04-21 13:33:00'), 
        (4, '赵六', 4, '2008-04-03 13:33:00', '2015-03-20 17:54:00'), 
        (5, '小兰', 2, '1989-05-04 17:33:00', '2012-06-18 17:54:00'), 
        (6, '小新', 5, '1993-09-18 17:36:00', '2015-02-28 17:36:00'), 
        (7, '小白', 2, '1991-10-17 17:37:00', '2014-08-16 17:37:00'), 
        (8, '小智', 7, '1987-06-20 17:37:00', '2015-01-10 17:38:00'), 
        (9, '大头', 6, '1991-02-14 08:49:00', '2014-07-12 08:49:00'), 
        (10, '小明', 3, '1991-02-14 08:49:00', '2015-03-04 09:10:00'), 
        (11, '小刘', 1, '1992-03-18 14:52:00', '2014-07-21 09:00:00')";
$link->query($sql);
/*****end*********/
//左连接查询
//填充下面sql语句
$sql="SELECT emp.e_id, emp.e_name, emp.date_of_birth, emp.date_of_entry, dept.d_name 
      FROM emp_info AS emp 
      LEFT JOIN emp_dept AS dept ON emp.d_id = dept.d_id";
$result=$link->query($sql);
$db=new Init();
$emp_info =$db->fetchAll($sql);
//设置常量,用以判断视图页面是否由此页面加载
define('APP', 'educoder');
//加载视图页面,显示数据
require 'list_html.php';

常见问题

由于这道题目的判题程序写的很普通,所以可能出现包括但不限于“未连接到数据库”,“找不到表”之类的情况,如果确定自己的答案没问题可以多提交几次,全部都不通过再考虑其他可能性。

第2关:PHP实现下拉菜单显示数据 上

任务描述

在添加员工信息时,一般员工所属部门是不允许用户手动输入的,而是通过动态查询数据库,获取到部门表数据,以下拉菜单的形式展示到表单中,以供选择。你的下一个任务是完成员工添加及修改时动态获取部门信息的功能。

相关知识

< select >标签

select 元素可创建单选或多选菜单。
< select > 元素中的 < option > 标签用于定义列表中的可用选项:

<select>
  <option value ="volvo">Volvo</option>
  <option value ="saab">Saab</option>
  <option value="opel">Opel</option>
  <option value="audi">Audi</option>
</select>

效果如图:
效果图

封装数据库连接函数

员工数据的读取、添加及修改的过程中,有许多代码都是类似的,我们可以把这些相同的代码段抽取出来封装在一个类中,在需要使用的时候调用类中的函数即可。这样既做到了代码的复用性,也为以后的维护工作减轻了负担。因此,我们首先利用Init类的构造函数,将连接数据库、设置字符集以及选择数据库的功能封装。

 <?php
 /**
  * 初始化数据库连接
  */
 function __construct()
       {
         $this->link=new mysqli(self::$host,self::$root,self::$password);
           //判断数据库连接是否成功,如果不成功则显示错误信息并终止脚本继续执行
           if($this->link->connect_error){
               die('连接数据库失败!');
           }
           //设置字符集,选择数据库
           $this->link->query('set names utf8;');
           $this->link->query('use itcast;');
       }
封装执行SQL语句函数

在PHP操作MySQL时,增、删、改、查都是通过$con->query()函数来执行SQL语句得到结果。因此我们可以把这个功能也封装到一个函数中,这样能够在函数中对执行结果进行判断,返回统一内容。
接下来添加query()函数:

 <?php
 /**
  * 执行SQL的函数
  * @param string $sql待执行的SQL
  * @return mixed 失败时返回false,成功时,如果是查询语句返回结果集,如果非查询类返回true
  */
 public function query($sql) {
       //some codes here
 }

请根据注释的提示,在本关实现这个功能。

封装处理多条数据的函数

对结果集进行遍历,获取其中的所有数据,这在功能开发中是经常出现的。因此可以把这部分功能代码封装到一个函数中。接下来添加fetchAll()函数:

 <?php
 /**
  * 处理结果集中有多条数据的函数
  * @param string $sql 待执行的SQL
  * @return array 返回遍历结果集后的二维数组
  */
  public function fetchAll($sql) {
     
       if ($result =$this->link->query($sql)) {
               //执行成功
               //遍历结果集
              //some codes here
              
              
               //释放结果集资源
               $result->free();
               return $rows;
           } else {
               //执行失败
               return false;
           }
 } 

请根据注释的提示,在本关实现这个功能。

封装处理单条数据的函数

在实际开发中,我们经常需要从数据库获取指定的某一条数据,因此还需要封装一个处理单条数据的函数。
接下来添加fetchRow()函数来对单条数据进行处理:

 <?php
 /**
   *  处理结果集中只有一条数据的函数
   * @param string $sql 待执行的SQL语句
   * @return array 返回结果集处理后的一维数组
  */
 public function fetchRow($sql) {
     //执行query()函数
      if ($result = $this->link->query($sql)) {
               //从结果集取得一次数据即可
              //some codes here
           } else {
               return false;
           }
 }

请根据注释的提示,在本关实现这个功能。

编程要求

补全右侧代码,要求实现的功能:

  1. (本关代码)补全empAdd.php文件,为实现以下拉菜单来展现部门数据做准备。
  2. (下一关代码)补全update_html.php文件,下拉菜单的value为部门ID,下拉菜单的文字显示为部门名称。
    其中,部门信息表都存储在数组$emp_dept中,数组的下标即为表的属性名。
    表的结构如下:
create table emp_dept (
         d_id int unsigned primary key auto_increment,
         d_name varchar(20) not null
          );

测评说明

平台将自动编译补全后的代码,根据程序的输出判断是否正确。

代码

<?php
/**
 * 初始化数据库连接
 */



Class Init
{
    private $link;
    private static $host="127.0.0.1";
    private static $root="root";
    private static $password="123123";


    function __construct()
    {
        $this->link=new mysqli(self::$host,self::$root,self::$password);

        //判断数据库连接是否成功,如果不成功则显示错误信息并终止脚本继续执行
        if($this->link->connect_error){
            die('连接数据库失败!');
        }
        //设置字符集,选择数据库
        $this->link->query('set names utf8;');
        $this->link->query('use educoder;');
    }


    /**
     * 执行SQL的方法
     * @param string $sql 待执行的SQL
     * @return mixed 失败返回false,成功,如果是查询语句返回结果集,如果非查询类返回true
     */

    public function query($sql) {
       /***********Begin******/
        if (!$result = $this->link->query($sql)) {
            return false;
        }
        return (is_object($result) || is_bool($result)) ? $result : true;
        /*********end*******/
    }


    /**
     * 处理结果集中有多条数据的方法
     * @param string $sql 待执行的SQL
     * @return array 返回遍历结果集后的二维数组
     */
    public function fetchAll($sql) {

        if ($result =$this->link->query($sql)) {
            //执行成功
            //遍历结果集
            /***********Begin******/
            $rows = array();
            while ($row = $result->fetch_assoc()) {
                $rows[] = $row;
            }
            /*********end*******/
            //释放结果集资源
            $result->free();
            return $rows;

        } else {
            //执行失败
            return false;
        }
    }



    /**
     * 处理结果集中只有一条数据的方法
     * @param string $sql 待执行的SQL语句
     * @return array 返回结果集处理后的一维数组
     */
    public function fetchRow($sql) {
        //执行query()函数
        if ($result = $this->link->query($sql)) {
            //从结果集取得一次数据即可
            /***********Begin******/
            $row = $result->fetch_assoc();
            return $row ? $row : false;
            /*********end*******/
            
        } else {
            return false;
        }

    }

    /**
     * 对数据进行安全处理
     * @param string $data 待转义字符串
     * @return string 转义后的字符串
     */
    function safeHandle($data){
        //转义字符串中的HTML标签
        $data = htmlspecialchars($data);
        //转义字符串中的特殊字符
        $data = mysqli_real_escape_string($this->link,$data);
        return $data;
    }

}



//初始化数据库
$db=new Init();

//判断是否有表单提交
if(!empty($_POST)){

    //声明变量$value,用来保存字段信息
    $fields = array('e_name','e_dept','date_of_birth','date_of_entry');

    //声明变量$value,用来保存值信息
    $value = array();

    //遍历$allow_field,获取输入员工数据的键和值
    foreach($fields as $k => $v){

        $data = isset($_POST[$v]) ? $_POST[$v] : '';

        if($data=='') die($v.'字段不能为空');

        $data =$db-> safeHandle($data);

        //把字段使用反引号包裹,赋值给$fields数组
        $fields[$k] = "`$v`";

        //把值使用单引号包裹,赋值给$values数组
        $values[] = "'$data'";
    }

    //将$fields数组以逗号连接,赋值给$fields,组成insert语句中的字段部分
    $fields = implode(',', $fields);

    //将$values数组以逗号连接,赋值给$values,组成insert语句中的值部分
    $values = implode(',', $values);

    //最后把$fields和$values拼接到insert语句中,注意要指定表名
    $sql = "insert into `emp_info` ($fields) values ($values)";

    //执行SQL
    if($res = $db->query($sql)){
        //成功时返回到 showList.php
        header('Location: ./showList.php');
        //停止脚本
        die;
    }else{
        //执行失败
        die('员工添加失败!');
    }
}
$sql = 'select * from emp_dept';
//调用fetchAll()函数,执行SQL并进行数据处理,把处理后的部门数据赋值给$emp_dept
$emp_dept = $db->fetchAll($sql);

//没有表单提交时,显示员工添加页面
define('APP', 'educoder');
require 'add_html.php';

第3关:PHP实现下拉菜单显示数据 下

任务描述

在添加员工信息时,一般员工所属部门是不允许用户手动输入的,而是通过动态查询数据库,获取到部门表数据,以下拉菜单的形式展示到表单中,以供选择。你的下一个任务是完成员工添加及修改时动态获取部门信息的功能。

相关知识

< select >标签
select 元素可创建单选或多选菜单。
< select > 元素中的 < option > 标签用于定义列表中的可用选项:

<select>
  <option value ="volvo">Volvo</option>
  <option value ="saab">Saab</option>
  <option value="opel">Opel</option>
  <option value="audi">Audi</option>
</select>

效果如图:
效果图

封装数据库连接函数

员工数据的读取、添加及修改的过程中,有许多代码都是类似的,我们可以把这些相同的代码段抽取出来封装在一个类中,在需要使用的时候调用类中的函数即可。这样既做到了代码的复用性,也为以后的维护工作减轻了负担。因此,我们首先利用Init类的构造函数,将连接数据库、设置字符集以及选择数据库的功能封装。

 <?php
 /**
  * 初始化数据库连接
  */
 function __construct()
       {
           $this->link=new mysqli(self::$host,self::$root,self::$password);
           //判断数据库连接是否成功,如果不成功则显示错误信息并终止脚本继续执行
           if($this->link->connect_error){
               die('连接数据库失败!');
           }
           //设置字符集,选择数据库
           $this->link->query('set names utf8;');
           $this->link->query('use itcast;');
       }
封装执行SQL语句函数

在PHP操作MySQL时,增、删、改、查都是通过$con->query()函数来执行SQL语句得到结果。因此我们可以把这个功能也封装到一个函数中,这样能够在函数中对执行结果进行判断,返回统一内容。
接下来添加query()函数:

 <?php
 /**
  * 执行SQL的函数
  * @param string $sql待执行的SQL
  * @return mixed 失败时返回false,成功时,如果是查询语句返回结果集,如果非查询类返回true
  */
 public function query($sql) {
       //some codes here
 }

请根据注释的提示,在本关实现这个功能。

封装处理多条数据的函数

对结果集进行遍历,获取其中的所有数据,这在功能开发中是经常出现的。因此可以把这部分功能代码封装到一个函数中。接下来添加fetchAll()函数:

 <?php
 /**
  * 处理结果集中有多条数据的函数
  * @param string $sql 待执行的SQL
  * @return array 返回遍历结果集后的二维数组
  */
  public function fetchAll($sql) {
     
       if ($result =$this->link->query($sql)) {
               //执行成功
               //遍历结果集
              //some codes here
              
              
               //释放结果集资源
               $result->free();
               return $rows;
           } else {
               //执行失败
               return false;
           }
 } 

请根据注释的提示,在本关实现这个功能。

封装处理单条数据的函数

在实际开发中,我们经常需要从数据库获取指定的某一条数据,因此还需要封装一个处理单条数据的函数。
接下来添加fetchRow()函数来对单条数据进行处理:

 <?php
 /**
   *  处理结果集中只有一条数据的函数
   * @param string $sql 待执行的SQL语句
   * @return array 返回结果集处理后的一维数组
  */
 public function fetchRow($sql) {
     //执行query()函数
      if ($result = $this->link->query($sql)) {
               //从结果集取得一次数据即可
              //some codes here
           } else {
               return false;
           }
         
 } 

请根据注释的提示,在本关实现这个功能。

编程要求

补全右侧代码,要求实现的功能:
1.(上一关代码)补全右侧代码,为实现以下拉菜单来展现部门数据做准备。
2.(本关代码)补全右侧代码,下拉菜单的value为部门ID,下拉菜单的文字显示为部门名称。
其中,部门信息表都存储在数组$emp_dept中,数组的下标即为表的属性名。
表的结构如下

create table emp_dept (
         d_id int unsigned primary key auto_increment,
         d_name varchar(20) not null
          );

测评说明

平台将自动编译补全后的代码,根据程序的输出判断是否正确。

代码

<?php
require "empUpdate.php";
if(!defined('APP')) die('error!');
?>
<!doctype html>
<html>
<head>
    <meta charset="utf-8">
    <title>员工信息编辑</title>
    <link rel="stylesheet" href="./js/jquery.datetimepicker.css"/ >
    <script src="./js/jquery.js"></script>
    <script src="./js/jquery.datetimepicker.js"></script>
    <script>
        $(function(){
            $('#date_of_birth').datetimepicker({lang:'ch'});
            $('#date_of_entry').datetimepicker({lang:'ch'});
        });
    </script>
    <style>
        body{background-color:#eee;margin:0;padding:0;}
        .box{width:400px;margin:15px auto;padding:20px;border:1px solid #ccc;background-color:#fff;}
        .box h1{font-size:20px;text-align:center;}
        .profile-table{margin:0 auto;}
        .profile-table th{font-weight:normal;text-align:right;}
        .profile-table input[type="text"]{width:180px;border:1px solid #ccc;height:22px;padding-left:4px;}
        .profile-table .button{background-color:#0099ff;border:1px solid #0099ff;color:#fff;width:80px;height:25px;margin:0 5px;cursor:pointer;}
        .profile-table .td-btn{text-align:center;padding-top:10px;}
        .profile-table th,.profile-table td{padding-bottom:10px;}
        .profile-table td{font-size:14px;}
        .profile-table .txttop{vertical-align:top;}
        .profile-table select{border:1px solid #ccc;min-width:80px;height:25px;}
        .profile-table .description{font-size:13px;width:250px;height:60px;border:1px solid #ccc;}
    </style>
</head>
<body>
<div class="box">
    <h1>修改员工信息</h1>
    <form method="post">
        <table class="profile-table">
            <tr><th>姓名:</th><td><input type="text" name="e_name" value="<?php echo $emp_info['e_name']; ?>"/></td></tr>
            <tr><th>所属部门:</th>
                <td>
                    <!—下拉菜单开始-->
                    <!—在输出每个部门信息时,判断是否为该员工当前所属部门,如果是设置为默认项selected='selected'-->
 					<!somecodes here-->
                    <select name="e_dept">
                        <?php foreach ($emp_dept as $dept): ?>
                            <option value="<?php echo $dept['d_id']; ?>"<?php if ($dept['d_id'] == $emp_info['d_id']) echo "selected='selected'"; ?>><?php echo $dept['d_name']; ?></option>
                        <?php endforeach; ?>
                    </select>
                    <!—下拉菜单结束-->
                </td></tr>
            <tr><th>出生年月:</th><td><input id="date_of_birth" name="date_of_birth" type="text" value="<?php echo $emp_info['date_of_birth']; ?>"></td></tr>
            <tr><th>入职日期:</th><td><input id="date_of_entry" name="date_of_entry" type="text" value="<?php echo $emp_info['date_of_entry']; ?>"></td></tr>
            <tr><td colspan="2" class="td-btn">
                    <input type="submit" value="保存资料" class="button" />
                    <input type="reset" value="重新填写" class="button" />
                </td></tr>
        </table>
    </form>
</div>
</body>
</html>

第4关: PHP实现分页显示数据

任务描述

为了进一步加强使用时的用户体验,你打算在网站上应用分页链接的显示样式。这个分页样式由“首页”、“尾页”以及指定数量的中间页码组成。具体效果如下图所示。
请添加图片描述

为了完成这样一个相对灵活的分页样式,你需要学习以下知识。

相关知识

为了能够灵活控制显示的页码数量,你首先需要编写一个分页链接生成函数:

<?php
/**
 * 分页链接生成函数
 * $page  URL传递的page值
 * $max_page  最大页码值
 * $show_page_num 页面中显示多少页码,默认为5
 */
function makePageHtml($page, $max_page, $show_page_num = 5) {
……
}

其中,页面链接可以以如下方式显示:
<a href="?page=1">1</a>
该语句在网页中显示为指向第一页的链接。
而如果是指向当前页的链接,则使用如下语句
<a class = "curl" href="?page=1">1</a>
该语句代表当前打开的即为第一页。

循环输出指定数量的页码链接的方法

我们需要实现的是:一次显示指定数量的页码,例如“1 2 3 4 5”、“6 7 8 9 10”。
这里面就需要我们确定2个问题:

  1. 当前显示的页码第一位的数字。
  2. 当前显示的页码最后一位的数字。

我们可以把“1 2 3 4 5”、“6 7 8 9 10”这样的页码看做一个小组,每组的最小值就是当前页码的第一位数字,最大值就是最后一位数字。
完成上述步骤后,就可以随意控制显示的页码数量进行分页了,例如我们选择显示4个页码,则结果如下图所示。
请添加图片描述

判断显示的页面前后是否还有分页的方法

此时还有一个问题没有解决,第4页之后还有分页该怎么办。
这就需要我们在循环页码的时候进行判断,如果当前显示的页码之后还有分页,则在页码最后显示“>”表示后面还有内容,例如1 2 3 4 5 >
如果当前显示的页码之前还有分页,则在页码最前显示<表示前面还有内容,例如< 6 7 8 9 10
注意:在html中,<要写成&lt>要写成&gt

编程要求

补全右侧代码,完成函数
makePageHtml($page, $max_page, $show_page_num = 5)
其中:

  • $page URL传递的page值
  • $max_page 最大页码值
  • $show_page_num 页面中显示多少页码
    返回值为该显示的分页链接,返回的分页链接格式为: 首页 1 2 3 4 5 > 尾页 以及 首页 < 6 尾页 其中,各个页码链接之间请用&nbsp;进行分隔。

测评说明

平台将自动编译补全后的代码,根据程序的输出判断是否正确。

代码

<?php
/**
 * 分页链接生成函数
 * $page  URL传递的page值
 * $max_page  最大页码值
 * $show_page_num 页面中显示多少页码
 */
function makePageHtml($page, $max_page, $show_page_num = 5) {


    //将首页的链接存入字符串$page_html中
    /*******Begin********/
    $page_html .= '<a href="?page=1">首页</a>&nbsp;';
    /*******end*********/

    //根据当前页码$page计算出它的所在分组,以及需要显示的第一页$i和最后一页$max_i
    /*******Begin********/
    $group = ceil($page / $show_page_num);
    $i = ($group - 1) * $show_page_num + 1;
    $max_i = min($group * $show_page_num, $max_page);
    if ($max_i < 1) $max_i = 1;
    /*******end*********/

    for ($i; $i <= $max_i; $i++) {
      //判断是否有上一个页面分组,若有则向$page_html中添加"<"的链接
      /*******Begin********/
      if ($i == ($group - 1) * $show_page_num + 1 && $i > 1) {
          $prev_page = $i - 1;
          $page_html .= '<a href="?page=' . $prev_page . '">&lt;</a>&nbsp;';
      }
      /*******end*********/

      //判断是否为当前选中页,并添加入$page_html。这是因为当前选择页的链接与其他页的链接不同
      /*******Begin********/
      if ($i == $page) {
          $page_html .= '<a class = "curl" href="?page=' . $i . '">' . $i . '</a>&nbsp;';
      } else {
          $page_html .= '<a href="?page=' . $i . '">' . $i . '</a>&nbsp;';
      }
    
      /*******end*********/

      //判断是否有下一页,若有则向$page_html中添加">"的链接
      /*******Begin********/
      if ($i == $max_i && $i < $max_page) {
          $next_page = $i + 1;
          $page_html .= '<a href="?page=' . $next_page . '">&gt;</a>&nbsp;';
      }
      /*******end*********/


    }
    //将尾页的链接存入字符串$page_html中
    /*******Begin********/
    $page_html .= '<a href="?page=' . $max_page . '">尾页</a>&nbsp;';
    /*******end*********/
    //返回显示分页链接的字符串
    return $page_html;

}
require "showList.php";

其他

都看到这里了,不点个赞或者收藏一下吗(●’◡’●)

关于头歌实践教学平台上的PHP函数资料和教程文档,目前并未提供明确的相关资源说明。然而,可以基于常见的PHP函数学习路径以及文件操作相关内容来推测可能的学习方向。 在PHP中,类似于Python中的`open`函数,用于处理文件的主要函数有以下几个核心部分: ### 常见的PHP文件操作函数 #### 1. `fopen()` 函数 此函数用于打开文件或者URL,并返回一个文件指针资源,在后续的操作中会用到该资源[^2]。 ```php $handle = fopen("example.txt", "r"); ``` #### 2. 文件模式 与Python类似,PHP也支持多种文件打开模式,例如只读 (`r`)、写入 (`w`) 和追加 (`a`) 等模式[^3]。 | 模式 | 描述 | |------|------| | r | 只读方式打开文件,文件指针指向文件开头。如果文件不存在则报错。 | | w | 打开只写文件,清除其中的内容。如果文件存在,则会被截断为零长度;如果文件不存在,则尝试创建它。 | | a | 打开以追加的方式写入文件,保留文件内容并添加新数据至末尾。 | #### 3. `fclose()` 函数 关闭由 `fopen()` 打开的文件指针资源,释放占用的内存和其他系统资源[^4]。 ```php fclose($handle); ``` #### 4. 其他常用文件操作函数 - **fgets()**: 从文件指针中读取一行。 - **fwrite()**: 将字符串写入文件。 - **file_get_contents()**: 把整个文件读入一个字符串中。 - **file_put_contents()**: 将一个字符串写入文件。 示例代码如下: ```php <?php // 打开文件 $file = 'data.txt'; $mode = 'w'; // 写入模式 $handle = fopen($file, $mode); if ($handle !== false) { echo "文件 {$file} 已成功打开,模式为:" . strtoupper($mode) . "\n"; // 向文件写入一些内容 fwrite($handle, "这是一个测试。\n"); // 关闭文件 fclose($handle); } ?> ``` 对于具体的头歌平台上是否有针对这些功能的教学材料或实训项目,建议访问其官方帮助中心或联系技术支持获取最新版本的文档和支持信息[^5]。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值