抽象类
* 1.使用关键字: abstract
* 2.类中只要有一个方法声明为abstract抽象方法,那么这个类就必须声明为抽象类
* 3.抽象方法只允许有方法声明与参数列表,不允许有方法体;
* 4.因为抽象方法的不确定性,所以抽象类禁止实例化,仅允许通过继承来实例化;
* 5.继承抽象类的子类中,必须将抽象类中的所有抽象方法全部实现
* 6.子类成员的访问限制级别必须等于或小于抽象类的约定,例如抽象类是protected,子类必须是
* protected 或者 public 不允许是private
* 7.子类方法参数必须与抽象类方法参数完全一致,但允许增加默认参数
特别注意
* 1. 尽管抽象类不能实例化,但仍然可以为它创建构造器,但必须声明为final
* 2. 抽象类理论上说不应该拥有静态成员,部分编辑器会有E_STRICT2048提示,但仍然可以这样做
<?php
abstract class Fruits
{
//水果名称
protected $name;
//声明静态属性,因为要用到静态类中
protected static $name;
//抽象方法
abstract public function eat();
//静态抽象方法
abstract static public function eat();
//尽管不能直接实例化抽象类,但仍然可以有构造方法
public function __construct()
{
return '抽象类构造器,实例化时自动调用<br>';
}
}
//为了教学方便,将抽象类与它的子类全部写在一个类文件中
//实际开发中应该为每一个类创建独立的类文件
class Apple extends Fruits
{
protected $name = '苹果';
//声明为静态属性
public static $name = '苹果';
public function eat()
{
return $this->name.'可以直接生吃';
}
//子类构造方法
public function __construct()
{
echo parent::__construct();
}
//实现抽象类中的抽象静态方法eat()
public static function eat()
{
return self::$name.'可以直接生吃';
}
}
$apple = new Apple;
echo $apple->eat();
//此前是抽象静态类,不需要例化,可以用类直接访问
echo Apple::eat();点击 "运行实例" 按钮查看在线实例
遍历对象
* 1.仅能遍历属性,方法不可遍历
* 2.外部遍历仅能查看公共可见属性
* 3.如果要查看全部属性,需要在类中创建外部接口方法来实现
* 4.最终结果以关联数组格式呈现,使用foreach()语句进行遍历
<?php
class Lecture
{
public $name = 'Peter Zhu';
public $gender = '男';
public $age = 30;
public $course = 'php,java,python,c';
protected $email = 'peter@php.cn';
private $salary = 18000;
private $phone = 15905519988;
public function listPro()
{
foreach ($this as $key=>$value){
echo '['.$key.'] => '.$value.'<br>';
}
}
}
//类外只能访问到公共可见属性,不能查看受保护与私有属性
foreach((new Lecture) as $key=>$value){
echo '['.$key.'] => '.$value.'<br>';
}
echo '<hr>';
echo '<h3>全部属性</h3>';
(new Lecture)->listPro();
//更多遍历对象的方法,可以查阅官方手册:php.net中的SPL函数库点击 "运行实例" 按钮查看在线实例
一、对象的序列化
* 1.php中的任何值都可以序列化为包含字节流表示的字符串来表示
* 2.序列化对象可以保存到变量或者文件中,方便保存和传送
二、对象的反序列化
<?php
//数值序列化
$num = 500;
echo serialize($num),'<br>';
//字符串序列化
$name = 'peter';
echo serialize($name),'<br>';
//数组序列化
$course = ['php','mysql','thinkphp'];
echo serialize($course),'<br>';
//布尔序列化
$isPass = true;
echo serialize($isPass),'<br>';
//对象序列化:以一个数据库连接类为例
class Db
{
//连接参数与返回值
public $db = null;
public $host;
private $user;
private $pass;
//构造方法
public function __construct($host='localhost',$user='root',$pass='root')
{
//类属性初始化
$this->host = $host;
$this->user = $user;
$this->pass = $pass;
//创建对象时自动连接数据库
$this->connect();
}
//连接数据库的方法
private function connect()
{
$this->db = mysqli_connect($this->host,$this->user, $this->pass);
}
//serialize($obj)序列化的时候,会自动调用__sleep(void)
//主要用于对象休眠时的一些清理工作,例如指定哪些属性允许进入到休眠对象的属性序列中
public function __sleep()
{
//返回由属性名字符串组成的索引数组,指示序列化时要保存的字段名
return ['host','user','pass'];
//对于本案例来说,如果连接参数不变的情况下,只要将$this->db保存到对象序列中即可
// return ['db'];
}
//unserialize()反序列化的时候,会自动调用__wakeup(void)
//主要用于唤醒对象时要做的初始化工作,例如本例中的:自动连接数据库
public function __wakeup()
{
$this->connect();
}
}
$obj = new Db();
/**
* 对象序列化的特点:
* 1.只保存对象中的属性,不保存方法
* 2.只保存类名,不保存对象名
*/
echo serialize($obj);
//为了演示反序列化,我们将序列化的对象保存到一个变量中,当然也可以保存到文件中
$tmp1 = serialize($obj);
echo '<hr>';
//查看序列化后的变量内容,与之前序列化内容是一样的
echo $tmp1;
//现在进行反序列化操作,将保存到变量中的对象取出来
$tmp2 = unserialize($tmp1);
//检测$tmp2是否是一个对象
echo '<hr>';
echo is_object($tmp2) ? '对象' : '不是';
echo '<hr>';
//获取属性,查看数据库连接对象
var_dump($tmp2->db);
//现在考虑这样的一个问题?如果我只想序列化点击 "运行实例" 按钮查看在线实例
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号