【设计模式】03:单例模式

单例模式


OVERVIOW

  • 单例模式
      • 1.单例模式实现
      • 2.饿汉与懒汉
        • (1)饿汉模式
        • (2)懒汉模式
      • 3.懒汉线程安全1
        • (1)引入互斥锁
        • (2)引入双重检查锁定
        • (3)引入原子变量
      • 4.懒汉线程安全2
        • (1)设置局部静态对象
      • 5.简单案例运用
        • (1)任务队列简单实现
        • (2)用户登录

项目全局范围内,某个类的实例有且仅有一个,通过这个实例向其他模块提供数据的全局访问,这种模式就叫单例模式。

单例模式的典型应用就是任务队列。使用单例模式来替代全局变量(对全局变量进行管理),直接使用全局变量会破坏类的封装(全局变量随意读写),通过单例模式的类提供的成员函数进行访问。

单例模式优点:

  1. 提高性能:避免频繁的创建销毁对象,提高性能,
  2. 节省内存空间:在内存中只有一个对象,节省内存空间,
  3. 避免多重占用:避免对共享资源的多重占用,
  4. 全局访问:可全局访问,利用单例模式避免全局变量的出现

单例模式缺点:

  1. 扩展困难:单例模式中没有抽象层,因此扩展困难,

  2. 不适用于变化的对象:如果同类型的对象总是要在不同的用例场景发生变化,单例就会引起数据错误,不能保存状态。

  3. 职责过重:违背了单一职责原则

  4. 负面问题:

    为了节省资源将数据库连接池对象设计为单例类,可能会导致共享连接池对象的程序过多,而出现连接池溢出。

    如果实例化的单例对象长时间不被利用,系统会认为是垃圾而被回收,这将导致对象状态的丢失。

单利模式使用场景:

  1. 需要频繁实例化然后销毁的对象
  2. 创建对象耗时过多or消耗资源过多,但又经常使用到的对象,
  3. 有状态的工具类对象,
  4. 频繁访问数据库或文件的对象,
  5. 要求只有一个对象的场景

1.单例模式实现

如果使用单例模式,首先要保证这个类的实例有且仅有一个。因此就必须采取一些操作,涉及一个类多对象操作的函数有以下几个:

  • 构造函数:创建一个新的对象
  • 拷贝构造函数:根据已有对象拷贝出一个新的对象
  • 拷贝赋值操作符重载函数:两个对象之间的赋值

为了把一个类可以实例化多个对象的路堵死,需要对以上几个函数做如下处理:

  1. 构造函数私有化,在类内部只调用一次这是可控的。
    • 由于类外部不能使用构造函数,所以在类内部创建的唯一的对象必须是静态的,这样就可以通过类名来访问了,为了不破坏类的封装把这个静态对象设置为私有。
    • 在类中只有它的静态成员函数才能访问其静态成员变量,所以给这个单例类提供一个静态函数用于得到这个静态的单例对象。
  2. 拷贝构造函数私有化或者禁用(使用 = delete)
  3. 拷贝赋值操作符重载函数私有化或者禁用(从单例的语义上讲该函数已经毫无意义,所以在类中不再提供这样一个函数,故将它也一并处理)

单例模式就是给类创建一个唯一的实例对象,UML类图如下:

在这里插入图片描述

#include<iostream>
using namespace std;
/*
   	1.关于类创建后的默认提供的函数
  	 - 在创建一个新的类之后 会默认提供3个构造函数 1个析构函数
  	 - 2个操作符重载(移动赋值操作符重载、拷贝赋值操作符重载)移动构造函数 拷贝构造函数
  	2.关于单例模式下类的实例化
	 - 在通过将 无参构造函数、拷贝构造函数、拷贝赋值操作符重载函数禁用之后 TaskQueue类已经无法在外部创建任何的对象
	 - 要得到TaskQueue的实例无法通过new操作符得到 只能通过类名得到(需要将对象设置为静态对象)
	 - 通过类名访问类内部的属性和方法 其属性和方法一定是静态的(若不是静态需要通过对象来调用)
	 - 能够操作静态成员变量的函数 只有静态成员函数
*/

//单例模式任务队列
class TaskQueue {
public:
	//无参构造函数
	//TaskQueue() = delete;
	//拷贝构造函数
	TaskQueue(const TaskQueue &t) = delete;
	//赋值操作符重载函数
	TaskQueue& operator=(const TaskQueue &t) = delete;
	// = delete 代表函数禁用, 也可以将其访问权限设置为私有
	
	//静态成员公共函数用于获取实例
	static TaskQueue *getInstance() { return m_taskq; }

	void printTest() { cout << "i am a public method of a singleton class" << endl; }

private:
	//无参构造函数
	TaskQueue() = default;
	//拷贝构造函数
	//TaskQueue(const TaskQueue &t) = default;
	//赋值操作符重载函数
	//TaskQueue& operator=(const TaskQueue &t) = default;
	//通过类名访问静态属性或方法来创建类实例(需要在类外部做初始化处理)
	static TaskQueue *m_taskq;
};

//静态成员初始化放到类外部处理
TaskQueue* TaskQueue::m_taskq = new TaskQueue;

int main() {
	//获取TaskQueue的单例对象 由m_taskq指针指向
	TaskQueue* m_taskq = TaskQueue::getInstance();
	//由m_taskq指针调用单例类内部的成员方法
	m_taskq->printTest();
	return 0;
}

以上为单例模式中的饿汉模式,在定义单例类的时候就将类对应的单例对象一并创建出来了。

2.饿汉与懒汉

在实现一个单例模式的类的时候,有两种处理模式:

  • 饿汉模式:在将单例类定义出来后实例就已经存在了,饿汉模式没有线程安全问题
  • 懒汉模式:在使用单例对象的时候才会去创建单例对象的实例(节省内存空间),懒汉模式存在线程安全问题(多个线程同时访问单例的实例)
(1)饿汉模式
  1. 多个线程在访问单例对象时,没有线程安全问题,单例对象已经存在,不会出现多个线程创建出多个单例对象的情况。
  2. 多线程拿到单例对象后,在访问单例对象内部的数据时,有线程安全问题(多线程共享资源),
//饿汉模式
class TaskQueue {
public:
	TaskQueue(const TaskQueue &t) = delete;
	TaskQueue& operator=(const TaskQueue &t) = delete;
	static TaskQueue *getInstance() { return m_taskq; }
	void printTest() { cout << "i am a public method of a singleton class" << endl; }

private:
	TaskQueue() = default;
	static TaskQueue *m_taskq;
};

TaskQueue* TaskQueue::m_taskq = new TaskQueue;
//饿汉模式
class TaskQueue {
public:
	TaskQueue(const TaskQueue &t) = delete;
	TaskQueue& operator=(const TaskQueue &t) = delete;
	static TaskQueue *getInstance() { return &m_taskq; }
	void printTest() { cout << "i am a public method of a singleton class" << endl; }

private:
	TaskQueue() = default;
	static TaskQueue m_taskq;//已经创建对象
};

TaskQueue* TaskQueue::m_taskq;//改为对象声明
(2)懒汉模式
//懒汉模式
class TaskQueue {
public:
	TaskQueue(const TaskQueue &t) = delete;
	TaskQueue& operator=(const TaskQueue &t) = delete;
	static TaskQueue *getInstance() {
		if (m_taskq == nullptr) m_taskq = new TaskQueue;
		return m_taskq;
	}
	void printTest() { cout << "i am a public method of a singleton class" << endl; }

private:
	TaskQueue() = default;
	static TaskQueue *m_taskq;
};

TaskQueue* TaskQueue::m_taskq = nullptr;

3.懒汉线程安全1

在单例模式中饿汉模式下,针对在多线程中可能存在的线程安全问题(创建多个实例),进行问题修改:

(1)引入互斥锁

在多线程环境下,有可能的情况是:多个线程同时进入到getInstance()方法中的if语句判断中,这时对象就可能被同时创建多个,

//懒汉模式 引入互斥锁
class TaskQueue {
public:
	TaskQueue(const TaskQueue &t) = delete;
	TaskQueue& operator=(const TaskQueue &t) = delete;
	static TaskQueue *getInstance() {
		m_mutex.lock();
		if (m_taskq == nullptr) m_taskq = new TaskQueue;
		m_mutex.unlock();
		return m_taskq;
	}
	void printTest() { cout << "i am a public method of a singleton class" << endl; }

private:
	TaskQueue() = default;
	static TaskQueue *m_taskq;
	static mutex m_mutex;
};

mutex TaskQueue::m_mutex;
TaskQueue* TaskQueue::m_taskq = nullptr;

使用互斥锁对new操作创建实例时进行加锁操作,防止同时创建多个实例,但是程序执行的效率太低(多线程访问单例对象时都是顺序访问)

(2)引入双重检查锁定

双重检查锁定,只有第一次访问时是顺序执行的,在TaskQueue被实例化出来之后,其他线程再去访问单例对象就是并行的了(不会进入if内)。

//懒汉模式 引入双重检查锁定
class TaskQueue {
public:
	TaskQueue(const TaskQueue &t) = delete;
	TaskQueue& operator=(const TaskQueue &t) = delete;
	static TaskQueue *getInstance() {
		//双重检查锁定
		if (m_taskq == nullptr) {
			m_mutex.lock();
			if (m_taskq == nullptr) m_taskq = new TaskQueue;
			m_mutex.unlock();	
		}
		return m_taskq;
	}
	void printTest() { cout << "i am a public method of a singleton class" << endl; }

private:
	TaskQueue() = default;
	static TaskQueue *m_taskq;
	static mutex m_mutex;
};

mutex TaskQueue::m_mutex;
TaskQueue* TaskQueue::m_taskq = nullptr;
(3)引入原子变量

通过引入双重检查锁定的方式,解决了在懒汉模式下多线程访问单例对象时,出现的线程安全问题,

从表面上观察引入双重检查锁定的方式是十分完美的,但是从底层上依旧存在漏洞:

  1. 对于 m_taskq = new TaskQueue; 操作,其对应的机器指令并不是一条,而有三条(对于计算机来说代码都是二进制指令/机器指令),

    step1:创建一块内存(没有数据)
    step2:创建 TaskQueue 类型的对象,并将数据写入到对象中
    step3:为 m_taskq 对象指针初始化,将有效的内存地址传递给 m_taskq 对象指针
    
  2. 在实际的执行过程中,m_taskq = new TaskQueue; 对应的机器指令可能会被重新排序,成为

    step1:创建一块内存(没有数据)
    step3:为 m_taskq 对象指针初始化,将有效的内存地址传递给 m_taskq 对象指针
    step2:创建 TaskQueue 类型的对象,并将数据写入到对象中
    
  3. 如果线程A执行完成前两步之后失去CPU时间片被挂起,此时线程B在进行指针判断时,发现指针 m_taskq 不为空(但该指针指向内存没有被初始化),导致线程B使用了一个没有被初始化的队列对象,就会出现问题(出现问题是概率性的)

  4. 在C++11中引入原子变量 atomic,在底层控制机器指令的执行顺序,可以实现一种更加安全的懒汉模式,代码如下:

    使用原子变量 atomicstore() 方法来存储单例对象,使用 load() 方法来加载单例对象,

    在原子变量中这两个函数在处理指令的时候,默认的原子顺序是 memory_order_seq_cst 顺序原子操作,

    使用顺序约束原子操作库,整个函数的执行都将保证顺序执行,并且不会出现数据竞态 data races,

    缺点:使用这种方法实现的懒汉模式的单例执行效率更低一些,

    对代码进行以下修改:

    • 通过原子变量将类的实例对象保存起来(m_taskq 指针指向的内存)

    • 类外初始化 指针指向为nullptr

    • 对 getInstance 方法进行相关的修改操作

      多线层在调用 getInstance 方法时 需要从原子变量中加载任务队列的实例

      抢到互斥锁的线程将继续向下执行 创建实例对象

//懒汉模式 引入原子变量
class TaskQueue {
public:
	TaskQueue(const TaskQueue &t) = delete;
	TaskQueue& operator=(const TaskQueue &t) = delete;
	static TaskQueue *getInstance() {
		TaskQueue* taskq = m_taskq.load();
		if (taskq == nullptr) {
			m_mutex.lock();
			taskq = m_taskq.load();
			if (taskq == nullptr) {
				taskq = new TaskQueue;
				m_taskq.store(taskq);
			}
			m_mutex.unlock();
		}
		return m_taskq.load();
	}
	void printTest() { cout << "i am a public method of a singleton class" << endl; }

private:
	TaskQueue() = default;
	// static TaskQueue *m_taskq;
	static atomic<TaskQueue*> m_taskq;
	static mutex m_mutex;
	
};

mutex TaskQueue::m_mutex;
atomic<TaskQueue*> TaskQueue::m_taskq;
// TaskQueue* TaskQueue::m_taskq = nullptr;

4.懒汉线程安全2

在懒汉模式线程安全问题中,除了可以通过引入双重检查锁定来解决线程安全问题,还可以使用局部静态对象处理线程安全问题,

(1)设置局部静态对象

使用静态的局部对象解决线程安全问题,要求编译器必修支持C++11标准,

  1. getInstance() 局部函数中定义一个静态局部对象 static TaskQueue taskq; (调用无参构造初始化)
  2. 在C++11标准中规定,如果指令逻辑进入一个未被初始化的声明变量,所有并发执行应当等待该变量完成初始化,

注:使用静态的局部对象没有线程安全问题,已经由C++11标准中的编译器解决,未被初始化的变量,必须等待其完成初始化才能并发执行,

step1:创建一块内存(没有数据)
step2:创建 TaskQueue 类型的对象,并将数据写入到对象中(完成初始化操作)
step3:为 m_taskq 对象指针初始化,将有效的内存地址传递给 m_taskq 对象指针
// 懒汉模式 静态局部对象
class TaskQueue {
public:
	TaskQueue(const TaskQueue &t) = delete;
	TaskQueue& operator=(const TaskQueue &t) = delete;
	static TaskQueue* getInstance() {
		static TaskQueue taskq;
		return &taskq;
	}
	void printTest() { cout << "i am a public method of a singleton class" << endl; }
private:
	TaskQueue() = default;
};

5.简单案例运用

(1)任务队列简单实现
  1. 多线程拿到单例对象后,在访问单例对象内部的数据时,有线程安全问题(多线程共享资源),使用互斥锁保护多线程中共享的资源,

  2. C++11中给互斥锁加/解锁有两种方式,

    方法1:调用mutex对象的 unlock(); lock(); 方法

    方法2:使用lock_gurd自动管理加/解锁 lock_guard<mutex> locker(m_mutex);

    使用 lock_gurd 可以有效的避免死锁的问题,自动加/解锁

// 饿汉模式
class TaskQueue {
public:
	TaskQueue(const TaskQueue &t) = delete;
	TaskQueue& operator=(const TaskQueue &t) = delete;
	static TaskQueue *getInstance() { return m_taskq; }
    void printTest() { cout << "i am a public method of a singleton class" << endl; }
    // 判断任务队列是否为空
    bool isEmpty() {
        lock_guard<mutex> locker(m_mutex);
        return m_data.empty();
    }
    // 添加任务
    void addTask(int node) { 
        lock_guard<mutex> locker(m_mutex);
        m_data.push(node);
    }
    // 删除任务
    bool removeTask() {
        lock_guard<mutex> locker(m_mutex);
        if (m_data.empty()) return false;
        m_data.pop();
        return true;
    }
    // 获取队头任务
    int takeTask() {
        lock_guard<mutex> locker(m_mutex);
        if (m_data.empty()) return -1;
        return m_data.front();
    }
private:
    TaskQueue() = default;
	static TaskQueue *m_taskq;
    // 任务队列
    queue<int> m_data;
    mutex m_mutex;
};

TaskQueue* TaskQueue::m_taskq = new TaskQueue;
#include <iostream>
#include <thread>
#include <queue>
#include <mutex>
using namespace std;

int main() {
    // 获取单例对象
    TaskQueue *taskq = TaskQueue::getInstance();
    taskq->printTest();
    // 生产者线程
    // 使用匿名函数指定线程的处理动作
    thread t1([=](){
        for (int i = 0; i < 25; ++i) {
            taskq->addTask(i + 100);
            cout << "++push data:" << i + 100 << ", threadId = " << this_thread::get_id() << endl;
            this_thread::sleep_for(chrono::milliseconds(500));//休眠500ms
        }
    });
    // 消费者线程
    thread t2([=](){
        this_thread::sleep_for(chrono::milliseconds(100));
        while(!taskq->isEmpty()) {
            // 开始消费
            cout << "--take data:" << taskq->takeTask() << ", threadId = " << this_thread::get_id() << endl;
            taskq->removeTask();
            this_thread::sleep_for(chrono::milliseconds(1000));//休眠500ms
        }
    });
    // 主线程阻塞 只有当t1、t2线程都结束后 主线程解除阻塞
    t1.join();
    t2.join();
    return 0;
}

在这里插入图片描述

(2)用户登录

当用户成功登录之后,用户名和密码就会被存储到内存中,可以创建一个单例类,将用户数据保存到单例对象中,

class Test {
public:
	static Test* getInstance() { return &m_test; }
    // m_user
    void setUserName(QString name) {
        // 多线程下需要加锁解锁(涉及写操作)
        // lock();
        m_user = name;
        // unlock();
    }
    QString getUserName(){ return m_user; }
    // m_passwd
    // ....
    // ....
    // ....
private:
	Test();
	Test(const Test& t);
	static Test* m_test;
    // static Test m_test;
    // 定义变量 -> 属于唯一的单例对象
    QString m_user;
    QString m_passwd;
    QString m_ip;
    QString m_port;
    QString m_token;
}
Test* Test::m_test = new Test();	// 初始化
// Test Test::m_test;

tips:部分内容参考课程、书籍与网络等,题解、图示及代码内容根据老师课程、二次整理以及自己对知识的理解,进行整理和补充,仅供学习参考使用,不可商业化。

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

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

相关文章

[Android]AlertDialog对话框

1.介绍AlertDialog AlertDialog是Android中常用的对话框组件之一&#xff0c;用于在应用程序中显示一些提示信息、警告、确认信息或者提供用户进行选择的界面。AlertDialog通常用于与用户进行交互&#xff0c;例如要求用户确认某个操作、输入一些信息或者从一组选项中选择一个。…

【Linux】Linux下的代码调试器 ---gdb

&#x1f466;个人主页&#xff1a;Weraphael ✍&#x1f3fb;作者简介&#xff1a;目前正在学习c和Linux还有算法 ✈️专栏&#xff1a;Linux &#x1f40b; 希望大家多多支持&#xff0c;咱一起进步&#xff01;&#x1f601; 如果文章有啥瑕疵&#xff0c;希望大佬指点一二 …

高等数学上岸宝典笔记

①不单调的函数也可能有反函数 ②注意反函数与函数转换时的定义域与值域 ③收敛数列不一定有最值 收敛数列必有上界和下界&#xff0c;但不一定有最值&#xff0c;比如{An}1/n&#xff0c;下界为0&#xff0c;但永远取不到0 ④数列与其子数列的关系 例题&#xff1a; ⑤带根号…

使用jenkins插件Allure生成自动化测试报告

前言 以前做自动化测试的时候一直用的HTMLTestRunner来生成测试报告&#xff0c;后来也尝试过用Python的PyH模块自己构建测试报告&#xff0c;在后来看到了RobotFramework的测试报告&#xff0c;感觉之前用的测试报告都太简陋&#xff0c;它才是测试报告应该有的样子。也就是在…

Pytorch进阶教学——训练一个图像分类模型(GPU)

目录 1、前言 2、数据集介绍 3、获取数据 4、创建网络 5、训练模型 6、测试模型 6.1、测试整个模型准确率 6.2、测试单张图片 1、前言 编写一个可以分类蚂蚁和蜜蜂图片的模型&#xff0c;使用数据集对卷积神经网络进行训练。训练后的模型可以对蚂蚁或蜜蜂的图片进行…

全球79%的程序员都在考虑跳槽,你呢?

​在最近二十年中&#xff0c;全球行业都经历了一次数字化变革&#xff0c;各行各业对于技术开发的比重越来越高&#xff0c;而作为技术开发核心的开发人员们对于一个企业的未来发展也变得越来越重要。因此各企业对于技术人才的竞争变得火热&#xff0c;并且这个热度一年高过一…

“大+小模型”赋能油气行业高质量发展

近日&#xff0c;中国石油石化科技创新大会暨新技术成果展在北京盛大举行&#xff0c;九章云极DataCanvas公司携油气行业一站式AI综合解决方案重磅亮相&#xff0c;充分展示了公司助推油气行业实现AI规模化应用深厚的AI技术实力和领先的AI应用水准&#xff0c;赢得了行业专家和…

Gartner发布2024 年十大战略技术趋势

Gartner发布2024 年十大战略技术趋势 1. AI信任、风险和安全管理&#xff08;AI Trust, Risk and Security Management&#xff09;2.持续威胁暴露管理&#xff08;Continuous Threat Exposure Management&#xff09;3.可持续技术&#xff08;Sustainable Technology&#xff…

学习记录PCL-1 通过哈希表进行三维点云的虚拟格网划分

直接对整个场景的点云进行特征提取&#xff0c;效果很差&#xff0c;因此通过划分区域格网进行划分。格网划分有很多种方式&#xff0c;在这里尝试使用哈希表进行格网链接&#xff0c;后续通过在每个格网内基于点云特征进行提取。 参考博客&#xff1a; 点云侠的PCL 点云分块_p…

NFTScan 正式上线 Starknet NFTScan 浏览器和 NFT API 数据服务

2023 年 11 月 30 号&#xff0c;NFTScan 团队正式对外发布了 Starknet NFTScan 浏览器&#xff0c;将为 Starknet 生态的 NFT 开发者和用户提供简洁高效的 NFT 数据搜索查询服务。NFTScan 作为全球领先的 NFT 数据基础设施服务商&#xff0c;Starknet 是继 Bitcoin、Ethereum、…

RLHF:强化学习结合大预言模型的训练方式

RLHF (Reinforcement Learning from Human Feedback) 以强化学习方式依据人类反馈优化语言模型。 文章目录 一、简介二、一般的流程三、微调gpt介绍示例 参考文章 一、简介 强化学习从人类反馈中学习&#xff08;RLHF&#xff0c;Reinforcement Learning from Human Feedback&a…

MySQL组合索引,最左匹配原则失效

说明&#xff1a;在SQL优化时&#xff0c;建立组合索引&#xff0c;我们需要知道最左匹配失效的情况&#xff0c;本文通过实例介绍最左匹配原则失效&#xff1b; 建立组合索引 如下&#xff0c;是一张大表&#xff0c;有1000万条数据&#xff1b; 对表中password、sex和email…

Flat Ads将携6亿独家流量亮相白鲸GTC2023,在7V01展台等你

一年一度的白鲸出海全球流量大会GTC重磅来袭!今年GTC出海展区全面升级,规模扩增至15000平方米,覆盖游戏、应用、技术及品牌出海等热门行业,预计将迎来累计超30000名跨境出海相关从业者莅临参观。 Flat Ads受邀设展,现场互动100%中奖 从出海到全球化,中国互联网企业走向海外寻…

【MySQL】binlog数据恢复

binlog概述 binlog二进制日志记录保存了所有执行过的修改操作语句&#xff0c;不保存查询操作。如果 MySQL 服务意外停止&#xff0c;可通过二进制日志文件排查&#xff0c;用户操作或表结构操作&#xff0c;从而来恢复数据库数据。binlog 是逻辑日志&#xff0c;记录的是这个…

基于web宠颐生宠物医院系统设计与实现

基于web宠颐生医院系统开发与实现 摘要&#xff1a;时代飞速发展&#xff0c;网络也飞速发展&#xff0c;互联网许多的行业都可以用互联网实现了&#xff0c;互联网已经成为了人们生活中重要的一部分&#xff0c;或多或少的影响着我们的生活&#xff0c;互联网在给我带了方便的…

【UE】透视效果

效果 步骤 1. 新建一个空白工程 2. 添加一个第三人称游戏和初学者内容包到内容浏览器 3. 新建一个材质&#xff0c;这里命名为“M_Perspective” 打开“M_Perspective”&#xff0c;设置材质域为后期处理 添加三个“SceneTexture”节点&#xff0c;场景纹理ID选项分别设置为“…

netcore 获取应用程序或者站点根路径的一点知识和教训

最近在用abpvnext做报表导出&#xff0c;涉及到要在站点根目录生成pdf文件提供下载。于是就要获取站点根路径。 开头搜索资料提示用IWebHostEnvironment.ContentRootPath&#xff0c;来实现获取站点根目录。这个其实是正解。.netcore的通用规则&#xff0c;使用任何借口都是依…

Oracle(2-8)Configuring the Database Archiving Mode

文章目录 一、基础知识1、Redo Log History2、NOARCHIVELOG Mode 非归档模式3、ARCHIVELOG Mode 归档模式4、Changing the Archiving Mode 更改归档模式![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/d6a09f9a6de24de7bbcdad90b8d6b9ca.png)5、Auto and Manual Ar…

ZZULIOJ 2466: 楼上瞎说,楼下才是,Java

2466: 楼上瞎说&#xff0c;楼下才是 题目描述 《九章算术》的内容十分丰富&#xff0c;全书采用问题集的形式&#xff0c;收有246个与生产、生活实践有联系的应用问题&#xff0c;其中每道题有问&#xff08;题目&#xff09;、答&#xff08;答案&#xff09;、术&#xff…

剑指offer(C++)-JZ43:整数中1出现的次数(算法-其他)

作者&#xff1a;翟天保Steven 版权声明&#xff1a;著作权归作者所有&#xff0c;商业转载请联系作者获得授权&#xff0c;非商业转载请注明出处 题目描述&#xff1a; 输入一个整数 n &#xff0c;求 1&#xff5e;n 这 n 个整数的十进制表示中 1 出现的次数 例如&#xff0…
最新文章