JavaSE——面向对象10:抽象类、接口

目录

一、抽象类

(一)抽象类的引出

(二)抽象类基本介绍

(三)注意事项和使用细节

(四)抽象类的最佳实践——模板设计模式

二、接口

(一)接口快速入门

(二)基本介绍

(三)注意事项与使用细节

(四)接口VS继承

(五)接口的多态性

1.多态参数

2.多态数组

3.接口存在多态传递现象

(六)练习题


一、抽象类

(一)抽象类的引出

        当父类的某些方法,需要声明,但是又不确定如何实现时,就可以将其声明为抽象方法,那么这个就是抽象类。

class Animal{
    private String name;

    public Animal(String name) {
        this.name = name;
    }

    // 这里的eat实现了没有什么意义
    // 即:父类方法的不确定性问题
    public void eat(){
        System.out.println("这时一只动物,但是不知道吃什么");
    }
}

        父类方法有不确定性,所以可以将该方法设计为抽象(abstract)方法,所谓抽象方法就是没有实现的方法,即没有方法体。

改写后的代码:

abstract class Animal {
    private String name;

    public Animal(String name) {
        this.name = name;
    }
    
    public abstract void eat();
}

        当父类的一些方法不确定时,可以用abstract关键字来修饰该方法,这个方法就是抽象方法,用abstract来修饰该类就是抽象类。即:当一个类中存在抽象方法时,需要将该类声明为abstract类。一般来说,抽象类会被继承,由其子类来实现抽象方法。

(二)抽象类基本介绍

  1. 用abstract关键字来修饰一个类时,这个类就叫抽象类
    访问修饰符 abstract 类名{
    }
  2. 用abstract关键字来修饰一个方法时,这个方法就是抽象方法
    访问修饰符 abstract 返回类型 方法名(参数列表); // 没有方法体
  3. 抽象类的价值更多作用是在于设计,是设计者设计好后,让子类继承并实现抽象类,在框架和设计模式中使用比较多

(三)注意事项和使用细节

  1. 抽象类不能被实例化
  2. 抽象类不一定要包含abstract方法,也就是说,抽象类可以没有abstract方法,还可以有实现的方法
  3. 一旦类包含了abstract方法,则这个类必须声明为abstract
  4. abstract只能修饰类和方法,不能修饰属性和其他的。
  5. 抽象类可以有任意成员(因为抽象类仍然是类),比如:非抽象方法、构造器、静态属性等等
    abstract class A {
        private String name;
        public static int age = 10;
        public static final int FEED_TIMES = 3;
        
        public A() {
            System.out.println("无参构造器");
        }
    
        public A(String name) {
            System.out.println("有参构造器");
            this.name = name;
        }
    
        public static void ok() {
            System.out.println("静态方法");
        }
    
        public void hello() {
            System.out.println("成员方法");
        }
    
        public abstract void m1();
    
        static {
            System.out.println("static代码块");
        }
    
        {
            System.out.println("普通代码块");
        }
    }
  6. 抽象方法不能有方法体
  7. 如果一个类继承了抽象类,则它必须实现抽象类的所有抽象方法,除非它自己也声明为abstract类。
    ​
    abstract class E {
        public abstract void hi();
    }
    
    abstract class F extends E {
    
    }
    
    class G extends E {
        /**
         * 这里相当于子类G实现了父类E的抽象方法,
         * 所谓实现方法,就是有方法体
         */
        @Override
        public void hi() {
    
        }
    }
  8. 抽象方法不能使用private、final和static来修饰,因为这些关键字都是和重写相违背的。

(四)抽象类的最佳实践——模板设计模式

        模板设计模式(Template Method Pattern)是一种行为型设计模式,它定义了一个操作中的算法骨架,而将一些步骤的实现延迟到子类中。

        例如子类AA和子类BB中都有job方法,想要查看每个类的job方法的运行时间,就需要在每个job方法中进行时间差的计算,会造成代码重复,将计算时间差的重复代码抽取到父类中,形成一个模板,并且父类定义job的抽象方法,模板中调用job抽象方法。

        此时子类可以先重写各自的job方法,在main方法中调用模板即可,根据Java的动态绑定机制,哪个对象调用模板,最后就运行哪个对象的job方法。

public class TestTemplate {
    public static void main(String[] args) {
        AA aa = new AA();
        aa.calculateTime(); // 执行时间:4

        BB bb = new BB();
        bb.calculateTime(); // 执行时间:7
    }
}

abstract class Template {
    // 抽象方法
    public abstract void job();

    // 普通父类方法,调用抽象方法
    // 下面的方法时一个模板,可以用final修饰,防止被其他类重写
    public final void calculateTime() {
        long start = System.currentTimeMillis();
        job(); // 动态绑定机制,子类可以自定义job方法的内部逻辑
        long end = System.currentTimeMillis();
        System.out.println("执行时间:" + (end - start));
    }
}

class AA extends Template {

    @Override
    public void job() {
        long sum = 0;
        for (long i = 1; i <= 8000000; i++) {
            sum += i;
        }
    }
}

class BB extends Template {

    @Override
    public void job() {
        long sum = 0;
        for (long i = 1; i <= 8000000; i++) {
            sum *= i;
        }
    }
}

二、接口

(一)接口快速入门

        现实生活中我们常说的接口例如USB接口,你可以把手机、相机、U盘都插在USB插槽上,而不用担心哪个插槽是专门插哪个的,原因是做USB插槽的厂家,和做各种设备的厂家都遵守了同意的规定,包括尺寸、排线等等。

        下面我们先看用代码实现接口的案例:

首先定义一个USB接口:

public interface UsbInterface { //接口
    //规定接口的相关方法
    public void start();
    public void stop();
}

再定义一个Phone类,实现这个接口,并实现接口中的全部方法:

// 即 Phone类需要实现 UsbInterface接口 规定/声明的方法
public class Phone implements UsbInterface {

    @Override
    public void start() {
        System.out.println("手机开始工作...");
    }

    @Override
    public void stop() {
        System.out.println("手机停止工作.....");
    }
}

再定义一个Camera类,也实现这个接口,并实现接口中的全部方法:

public class Camera implements UsbInterface {
    // 实现接口,就是把接口方法实现

    @Override
    public void start() {
        System.out.println("相机开始工作...");
    }

    @Override
    public void stop() {
        System.out.println("相机停止工作....");
    }
}

定义一个Computer类,能够接收接口:

public class Computer {
    //编写一个方法, 计算机工作:
    //1. UsbInterface usbInterface 形参是接口类型 UsbInterface
    //2. 看到 接收 实现了 UsbInterface接口的类的对象实例
    public void work(UsbInterface usbInterface) {
        // 通过接口,来调用方法
        usbInterface.start();
        usbInterface.stop();
    }
}

main方法中创建对象运行:

public class Interface01 {
    public static void main(String[] args) {
        //创建手机,相机对象
        //Camera 实现了 UsbInterface
        Camera camera = new Camera();
        //Phone 实现了 UsbInterface
        Phone phone = new Phone();
        //创建计算机
        Computer computer = new Computer();
        computer.work(phone);// 把手机接入到计算机
        System.out.println("===============");
        computer.work(camera);// 把相机接入到计算机
    }
}

运行结果:

 

(二)基本介绍

        接口就是给出一些没有实现的方法,封装到一起,直到某个类要使用的时候,再根据具体情况把这些方法实现了、

基本语法:

interface 接口名{
    // 属性
    // 方法(1.抽象方法; 2.默认实现方法 3.静态方法)
}


class 类名 implements 接口{
    自己的属性;
    自己的方法;
    必须实现的接口的抽象方法;
}

小结:

  1. 在JDK7.0以前,接口里的所有方法都没有方法体,即都是抽象方法。
  2. JDK8.0及以后,接口类可以有静态方法、默认方法,也就是说,接口中可以有方法的具体实现。即(接口类中可以定义:1.抽象方法; 2.默认实现方法 3.静态方法)
    public interface AInterface {
        //属性:
        public int n1 = 10;
    
        // 方法:
        // 在接口中,抽象方法,可以省略abstract关键字
        public void hi();
    
        // 在jdk8后,接口中可以有默认实现方法,需要使用default关键字修饰
        default public void ok() {
            System.out.println("ok ...");
        }
    
        // 在jdk8后, 接口中可以有静态方法
        public static void cry() {
            System.out.println("cry ....");
        }
    }
    
    
    // 1.如果一个类 implements实现 接口
    // 2.需要将该接口的所有抽象方法都实现
    class A implements AInterface {
        @Override
        public void hi() {
            System.out.println("hi()....");
        }
    }

(三)注意事项与使用细节

  1. 接口不能被实例化

  2. 接口中所有的方法是 public方法,接口中抽象方法,可以不用abstract修饰

  3. 一个普通类实现接口,就必须将该接口的所有方法都实现,可以使用alt+enter来解决

  4. 抽象类去实现接口时,可以不实现接口的抽象方法

  5. 一个类同时可以实现多个接口:class Pig implements IB, IC {}

  6. 接口中的属性,只能是final的,而且是 public static final 修饰符,并且必须被初始化:

    interface IB {int n1 = 10;} // public static final 部分可以省略

  7. 接口中属性的访问形式:接口名.属性名

  8. 接口不能继承其它的类,但是可以继承多个别的接口:interface ID extends IB, IC {}

  9. 接口的修饰符 只能是 public 和默认,这点和类的修饰符是一样的:interface IE {}

public class InterfaceDetail01 {
    public static void main(String[] args) {
        // 1.接口不能被实例化
        // new IA();
    }
}

interface IA {
    // 2.接口中所有的方法是 public方法, 接口中抽象方法,可以不用abstract修饰
    void say();// 修饰符 public protected 默认 private
    // 相当于 public abstract void say();

    void hi();
}

// 3.一个普通类实现接口,就必须将该接口的所有方法都实现,可以使用alt+enter来解决
class Cat implements IA {
    @Override
    public void say() {
        System.out.println("实现接口IA的say方法");
    }

    @Override
    public void hi() {
        System.out.println("实现接口IA的hi方法");
    }
}

// 4.抽象类去实现接口时,可以不实现接口的抽象方法
abstract class Tiger implements IA {}
public class InterfaceDetail02 {
    public static void main(String[] args) {
        // 接口中的属性,是 public static final
        // 7.接口中属性的访问形式:接口名.属性名
        System.out.println(IB.n1);// 说明n1 就是static
        // IB.n1 = 30; // 无法重新赋值,说明n1是final
    }

}

interface IB {
    // 6.接口中的属性,只能是final的,而且是 public static final 修饰符,并且必须被初始化
    int n1 = 10; // 等价于 public static final int n1 = 10;

    void hi();
}

interface IC {
    void say();
}

// 5.一个类同时可以实现多个接口
class Pig implements IB, IC {
    @Override
    public void hi() {
        System.out.println("实现IB的hi方法");
    }

    @Override
    public void say() {
        System.out.println("实现IC的say方法");
    }
}

// 8.接口不能继承其它的类,但是可以继承多个别的接口
interface ID extends IB, IC {}

// 9.接口的修饰符 只能是 public 和默认,这点和类的修饰符是一样的
interface IE {}

小练习:分析下面代码是否有错误与输出结果:

public class InterfaceExercise01 {
    public static void main(String[] args) {
        B b = new B();//ok
        System.out.println(b.a);  //23  因为a是public,所以可以访问
        System.out.println(AA.a);  //23  因为a是static类型,所以可以直接调用
        System.out.println(B.a);  //23  因为B实现了AA接口,所以AA的属性和方法也能访问到
    }
}

interface AA {
    int a = 23; //等价 public static final int a = 23;
}

class B implements AA {//正确
}

(四)接口VS继承

  • 接口和继承解决的问题不同

        继承的价值主要在于:解决代码的复用性和可维护性。

        接口的价值主要在于:设计好各种规范(方法),让其它类去实现这些方法。

  • 接口比继承更加灵活

        继承是满足is-a的关系,比如Cat可以继承Animal,而Computer不能;

        接口只需满足like-a的关系,比如,通过实现接口中的方法,Cat也可以像Computer一样具有上网的功能。

  • 接口在一定程度上实现代码的解耦[即:接口规范性+动态绑定机制]
public class ExtendsVsInterface {
    public static void main(String[] args) {
        LittleMonkey littleMonkey = new LittleMonkey("悟空");
        // 子类可以调用父类的属性和方法
        littleMonkey.climbing(); // 悟空可以爬树...
        
        // 子类可以调用重写的接口方法
        littleMonkey.swimming(); // 悟空通过学习,可以像鱼儿一样游泳...
        littleMonkey.flying(); // 悟空通过学习,可以像鸟儿一样飞翔...
    }
}

// 如果LittleMonkey想要实现父类Monkey没有的功能,就需要实现对应的接口
// 定义接口Fishable
interface Fishable {
    // 编写接口中的抽象方法,swimming
    void swimming();
}

// 另一个接口Birdable
interface Birdable {
    void flying();
}

// 编写父类Monkey
class Monkey {
    private String name;

    public void climbing() {
        System.out.println(name + "可以爬树...");
    }

    public Monkey(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

// 编写子类LittleMonkey继承父类Monkey
class LittleMonkey extends Monkey implements Fishable, Birdable {

    // 因为父类没有无参构造,只有有参构造,所以子类必须实现父类指定的有参构造
    public LittleMonkey(String name) {
        super(name);
    }

    @Override
    public void swimming() {
        System.out.println(getName() + "通过学习,可以像鱼儿一样游泳...");
    }

    @Override
    public void flying() {
        System.out.println(getName() + "通过学习,可以像鸟儿一样飞翔...");
    }
}

小结:

  • 当子类继承了父类,就自动的拥有父类的功能
  • 如果子类需要扩展功能,可以通过实现接口的方式扩展
  • 可以理解为:实现接口 是 对Java单继承机制的一种补充

(五)接口的多态性

1.多态参数

        在前面的USB接口案例,UsbInterface既可以接收手机Phone对象,又可以接收相机Camera对象,就体现了接口的多态:

(1)接口类型的变量 可以指向 实现接口类的对象实例

(2)父类类型的变量 可以指向 实现父类的子类的对象实例

public class InterfacePloyParameter {
    public static void main(String[] args) {
        // 接口体现的多态:
        // 接口类型的变量 if01 可以指向 实现了IF接口类的对象实例
        IF if01 = new Monster();
        if01 = new Fish();

        // 继承体现的多态:
        // 父类类型的变量 aaa 可以指向 继承了AAA父类的子类的对象实例
        AAA aaa = new BBB();
        aaa = new CCC();
    }
}
interface IF{}

class Monster implements IF{}
class Fish implements IF{}

class AAA{}

class BBB extends AAA{}
class CCC extends AAA{}

2.多态数组

案例:

        在Usb接口数组中,存放Phone和Camera对象,Phone类还有一个特有的方法call(),Camera类还有一个特有的方法photo()。

        遍历Usb数组:如果是Phone对象,除了调用Usb 接口定义的方法外,还需要调用Phone特有方法call();如果是Camera对象,除了调用Usb接口定义的方法外,还需要调用Camera特有方法photo()

代码实现:

public class InterfacePolyArr {
    public static void main(String[] args) {
        //多态数组 -> 接口类型数组
        Usb[] usbs = new Usb[2];
        usbs[0] = new Phone_();
        usbs[1] = new Camera_();
       
        for (int i = 0; i < usbs.length; i++) {
            usbs[i].work();//动态绑定..
            // 和前面一样,我们仍然需要进行类型的 向下转型
            if (usbs[i] instanceof Phone_) {// 判断usbs[i]的运行类型是 Phone_
                ((Phone_) usbs[i]).call();
            } else if (usbs[i] instanceof Camera_) {
                ((Camera_) usbs[i]).photo();
            }
        }
    }
}

interface Usb {
    void work();
}

class Phone_ implements Usb {
    // Phone_特有的方法
    public void call() {
        System.out.println("手机可以打电话...");
    }

    @Override
    public void work() {
        System.out.println("手机工作中...");
    }
}

class Camera_ implements Usb {
    // Camera_特有的方法
    public void photo() {
        System.out.println("相机拍照更清晰...");
    }

    @Override
    public void work() {
        System.out.println("相机工作中...");
    }
}

3.接口存在多态传递现象

        如果类A实现了接口IN1,类A必须要实现接口IN1的抽象方法;如果接口IN1还继承了接口IN2,那么类A也必须要实现接口IN2的抽象方法。这就是接口的多态传递现象。

public class InterfacePolyPass {
    public static void main(String[] args) {
        //接口类型的变量可以指向,实现了该接口的类的对象实例
        IG ig = new Teacher();
        //如果IG 继承了 IH 接口,而Teacher 类实现了 IG接口
        //那么,实际上就相当于 Teacher 类也实现了 IH接口.
        //这就是所谓的 接口多态传递现象.
        IH ih = new Teacher();
    }
}

interface IH {
    void hi();
}

interface IG extends IH {
}

class Teacher implements IG {
    @Override
    public void hi() {
    }
}

(六)练习题

System.out.println(x);是否正确?

public class InterfaceExercise02 {
    public static void main(String[] args) {
        new C().pX(); // 0  1
    }
}

interface AB {
    int x = 0;
}  // 等价 public static final int x = 0;

class BB {
    int x = 1;
} //普通属性

class C extends BB implements AB {
    public void pX() {
        //System.out.println(x); //错误,原因不明确x
        //可以明确指定x
        //访问接口的 x 就使用 AB.x
        //访问父类的 x 就使用 super.x
        System.out.println(AB.x + " " + super.x);
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/887671.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

文件上传之%00截断(00截断)以及pikachu靶场

pikachu的文件上传和upload-lab的文件上传 目录 mime type类型 getimagesize 第12关%00截断&#xff0c; 第13关0x00截断 差不多了&#xff0c;今天先学文件上传白名单&#xff0c;在网上看了资料&#xff0c;差不多看懂了&#xff0c;但是还有几个地方需要实验一下&#…

高性能架构—存储高性能

1 &#x1f4ca;关系型数据库 存储技术飞速发展&#xff0c;关系型数据的ACID特性以及强大的SQL查询让其成为各种业务系统的关键和核心存储系统。 很多场景下的高性能设计最核心的就是关系型数据库的设计&#xff0c;很多数据库厂商再优化和提升单个数据库服务器的性能方面做了…

统一 SASE 架构中的网络和安全融合

网络威胁情报技术的进步 传统的网络边界一片混乱&#xff0c;剩下的只是无人管理的设备、分散在私有云和公共云中的资产、无法读取的应用程序流量泛滥&#xff0c;混合工作结构正在给现有网络的功能带来压力。 更重要的是&#xff0c;这些问题早在生成式人工智能和大型语言模…

【C++11】新特性

前言&#xff1a; C11 是C编程语言的一个重要版本&#xff0c;于2011年发布。它带来了数量可观的变化&#xff0c;包含约 140 个新特性&#xff0c;以及对 C03 标准中约600个缺陷的修正&#xff0c;更像是从 C98/03 中孕育出的新语言 列表初始化 C11 中的列表初始化&#xff0…

智能手表(Smart Watch)项目

文章目录 前言一、智能手表&#xff08;Smart Watch&#xff09;简介二、系统组成三、软件框架四、IAP_F411 App4.1 MDK工程结构4.2 设计思路 五、Smart Watch App5.1 MDK工程结构5.2 片上外设5.3 板载驱动BSP5.4 硬件访问机制-HWDataAccess5.4.1 LVGL仿真和MDK工程的互相移植5…

免费版U盘数据恢复软件大揭秘,拯救你的重要数据

我们的生活和工作越来越离不开各种存储设备&#xff0c;其中优盘因其小巧便携、方便使用的特点&#xff0c;成为了我们存储和传输数据的重要工具之一。为了防止你像我一样会遇到数据丢失抓狂的情况&#xff0c;我分享几款u盘数据恢复软件免费版工具来即时补救。 1.福昕U盘数据…

Oracle中TRUNC()函数详解

文章目录 前言一、TRUNC函数的语法二、主要用途三、测试用例总结 前言 在Oracle中&#xff0c;TRUNC函数用于截取或截断日期、时间或数值表达式的部分。它返回一个日期、时间或数值的截断版本&#xff0c;根据提供的格式进行截取。 一、TRUNC函数的语法 TRUNC(date) TRUNC(d…

鸿蒙harmonyos next flutter混合开发之开发plugin(获取操作系统版本号)

创建Plugin为my_plugin flutter create --org com.example --templateplugin --platformsandroid,ios,ohos my_plugin 创建Application为my_application flutter create --org com.example my_application flutter_application引用flutter_plugin&#xff0c;在pubspec.yam…

一键生成PPT的AI工具-Kimi!

一键生成PPT的AI工具-Kimi&#xff01; 前言介绍Kimi为什么选择Kimi如何使用Kimi在线编辑PPT下载生成的PPT自己编辑 结语 &#x1f600;大家好&#xff01;我是向阳&#x1f31e;&#xff0c;一个想成为优秀全栈开发工程师的有志青年&#xff01; &#x1f4d4;今天不来讨论前后…

Jenkins Pipline流水线

提到 CI 工具&#xff0c;首先想到的就是“CI 界”的大佬--]enkjns,虽然在云原生爆发的年代,蹦出来了很多云原生的 CI 工具,但是都不足以撼动 Jenkins 的地位。在企业中对于持续集成、持续部署的需求非常多,并且也会经常有-些比较复杂的需求,此时新生的 CI 工具不足以支撑这些很…

前缀和算法详解

对于查询区间和的问题&#xff0c;可以预处理出来一个前缀和数组 dp&#xff0c;数组中存储的是从下标 0 的位置到当前位置的区间和&#xff0c;这样只需要通过前缀和数组就可以快速的求出指定区间的和了&#xff0c;例如求 l ~ r 区间的和&#xff0c;就可以之间使用 dp[l - 1…

鸿蒙OpenHarmony

开源鸿蒙系统编译指南 Ubuntu编译环境配置第一步&#xff1a;Shell 改 Bash第二步&#xff1a;安装Git和安装pip3工具第三步&#xff1a;远程仓配置第四步&#xff1a;拉取代码第五步&#xff1a;安装编译环境第六步&#xff1a;本地编译源码 Windows开发环境配置第一步&#x…

巧用armbian定时任务控制开发板LED的亮灭

新买了个瑞莎 3E 开发板,号称最小SBC,到了之后简直玩开了花,各种折腾后 安装好armbian系统,各种调优。 不太满意的地方:由于板子太小的原因,导致两个USBTYPEC的接口距离很近,所以买的OTG转接口如果有点宽的话 会显得特别拥挤。 还有就是每天晚上天黑了之后,卧室…

Uniapp API

1.uni.showToast 显示消息提示框 unishowToast({ obj参数 }) 2.uni.showLoading 显示 loading 提示框, 需主动调用 uni.hideLoading 才能关闭提示框。 3.uni.showModal 显示模态弹窗&#xff0c;可以只有一个确定按钮&#xff0c;也可以同时有确定和取消按钮。类似于一个A…

躺平成长:微信小程序运营日记第二天

在进行属于生活的开源之后&#xff0c;自己更加感受到自己存在的渺茫&#xff0c;同时更加开始深刻领会&#xff0c;开源的重要性&#xff0c;在开源&#xff0c;开放&#xff0c;创造&#xff0c;再创新的思维模式下&#xff0c;不发布八部金刚功相关的训练视频&#xff0c;自…

基于Node2Vec的图嵌入实现过程

目录 一、引言二、Node2Vec&#xff08;原理&#xff09;2.1 随机游走&#xff08;Random Walk&#xff09;2.2 嵌入学习2.3 Node2Vec 的优势 三、使用 Node2Vec 进行图嵌入&#xff08;实践&#xff09;3.1 读取和转换 JSON 文件为 Graph 对象3.2 训练 Node2Vec 模型3.3 二维嵌…

MySQL--三大范式(超详解)

目录 一、前言二、三大范式2.1概念2.2第一范式&#xff08;1NF&#xff09;2.3第二范式&#xff08;2NF&#xff09;2.3第三范式&#xff08;3NF&#xff09; 一、前言 欢迎大家来到权权的博客~欢迎大家对我的博客进行指导&#xff0c;有什么不对的地方&#xff0c;我会及时改进…

使用前端三剑客实现一个备忘录

一&#xff0c;界面介绍 这个备忘录的界面效果如下&#xff1a; 可以实现任务的增删&#xff0c;并且在任务被勾选后会被放到已完成的下面。 示例&#xff1a; &#xff08;1&#xff09;&#xff0c;增加一个任务 &#xff08;2&#xff09;&#xff0c;勾选任务 &#xff…

影视cms泛目录用什么程序?苹果cms二次开发泛目录插件

影视CMS泛目录一般使用的程序有很多种&#xff0c;&#xff08;maccmscn&#xff09;以下是其中几种常见的程序&#xff1a; WordPress&#xff1a;WordPress是一个非常流行的开源内容管理系统&#xff0c;可以通过安装一些插件来实现影视CMS泛目录功能。其中&#xff0c;一款常…