【软件测试】日志第五天

news2024/11/26 2:44:44

系软件开发日志

提示:这里可以添加系列文章的所有文章的目录,目录需要自己手动添加
例如:软件测试


提示:写完文章后,目录可以自动生成,如何生成可参考右边的帮助文档

文章目录

  • 系软件开发日志
  • 前言web自动化测试
  • 一、web自动化测试是什么?
  • 二、使用步骤
    • 1.wait assert
    • 2.显式等待 参数化 测试前置
  • 一、po是什么?
    • POM原则
    • 1.wait assert
    • 2.po的使用
    • 3.出现问题如何解决
    • 4.POM原则
    • 1.wait assert
    • 1.wait assert
  • 总结


前言web自动化测试

提示:这里可以添加本文要记录的大概内容:

例如:web自动化测试,找到位置


提示:以下是本篇文章正文内容,下面案例可供参考

一、web自动化测试是什么?

示例:自动化(Automation)是指机器设备、系统或过程(生产、管理过程)在没有人或较少人的直接参与下,按照人的要求,经过自动检测、信息处理、分析判断、操纵 …
这是教科书里面的自动化的定义,回归到自动化测试其实自动化测试就是。

二、使用步骤

1.wait assert

代码如下(示例):

# Generated by Selenium IDE
import pytest
import time
import json
from selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.common.action_chains import ActionChains
from selenium.webdriver.support import expected_conditions
from selenium.webdriver.support.wait import WebDriverWait
from selenium.webdriver.common.keys import Keys
from selenium.webdriver.common.desired_capabilities import DesiredCapabilities

# 等待断言
class TestSearch():
    def setup_method(self, method):
        self.driver = webdriver.Edge()
        self.driver.implicitly_wait(5)
        self.vars = {}

    def teardown_method(self, method):
        self.driver.quit()

    def test_search(self):
        self.driver.get("https://ceshiren.com/")
        # self.driver.set_window_size(1920, 993)
        self.driver.maximize_window()
        self.driver.find_element(By.CSS_SELECTOR, ".d-icon-search").click()
        self.driver.find_element(By.CSS_SELECTOR, ".d-icon-sliders-h").click()
        self.driver.find_element(By.CSS_SELECTOR, '.search-query').send_keys("selenium")
        self.driver.find_element(By.CSS_SELECTOR, ".search-cta > .fa").click()
        assert len(self.driver.find_elements(By.CSS_SELECTOR, '.search-link')) > 0
        assert 'selenium' in self.driver.find_element(By.CSS_SELECTOR, '.search-link').text.lower()
        # self.driver.find_element(By.CSS_SELECTOR, '.search-link').click()
        # self.driver.find_element(By.LINK_TEXT, "selenium-Selenium 1 (Selenium RC)").click()

2.显式等待 参数化 测试前置

代码如下(示例):

from time import sleep

import pytest
from selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.support import expected_conditions
from selenium.webdriver.support.wait import WebDriverWait


class TestSearchAdvance:
    def setup_class(self):
        self.driver = webdriver.Edge()
        # 定位控件的最大超时时间 隐式等待
        self.driver.implicitly_wait(3)
        self.driver.get('https://ceshiren.com/t/topic/25780')
        self.driver.find_element(By.ID, 'search-button').click()
        self.driver.find_element(By.CSS_SELECTOR, '[title=打开高级搜索]').click()

    def setup_method(self):
        ...

    def teardown_method(self):
        self.driver.find_element(By.XPATH, '//summary[contains(., "高级筛选器")]').click()
        self.driver.find_element(By.CSS_SELECTOR, '.full-page-search.search-query').clear()
    def hingh_light(self,locator):
        self.driver.find_element(By.CSS_SELECTOR,locator)
        self.driver.execute_script(f"document.querySelector('{locator}').style.border = 'solid 2px red')")
        self.driver.execute_script(f"arguments[0].style.border='solid red 5px'",
                                   self.driver.find_element(*locator))
        sleep(0.5)



    @pytest.mark.parametrize('keyword, min, max', [
        ['selenium', 10, 1000],
        ['web', 10, 1000],
        ['selenium', 0, 1000],
        ['selenium', 100, 1000],
        # 拆分异常case
        # ['selenium', 100, 0],

    ])
    def test_search(self, keyword, min, max):
        self.driver.find_element(By.CSS_SELECTOR, '.full-page-search.search-query').send_keys('%s' % keyword)
        # self.driver.find_element(By.CSS_SELECTOR, '.search-advanced-additional-options').click()
        self.driver.save_screenshot('search.png')
        topic_number = (By.XPATH, '//summary[contains(., "话题浏览量")]')
        input_min = (By.CSS_SELECTOR, '[aria-label="按最小浏览量筛选"]')
        # WebDriverWait(self.driver, 5).until(expected_conditions.element_to_be_clickable(input_min))
        # WebDriverWait(self.driver, 5).until(expected_conditions.element_to_be_clickable(topic_number))
        # self.hingh_light(topic_number)
        self.driver.find_element(*topic_number).click()

        def wait(driver):
            # 编写显示等待的逻辑
            # 点击一个控件有时会失效 一直重试
            self.driver.find_element(*topic_number).click()
            # 直到某个控件出现
            # self.hingh_light(input_min)
            self.driver.save_screenshot('topic.png')
            return self.driver.find_element(*input_min).is_displayed()

        # 显式等待 使用自定义函数
        WebDriverWait(self.driver, 5).until(wait)
        # self.hingh_light(input_min)
        self.driver.find_element(*input_min).send_keys(min)
        self.driver.find_element(By.CSS_SELECTOR, '[aria-label="按最大浏览量筛选"]').send_keys(max)
        self.driver.find_element(By.CSS_SELECTOR, '.search-cta[aria-label="搜索"]').click()

        topic_title = (By.CSS_SELECTOR, '.topic-title')
        # 显式等待
        WebDriverWait(self.driver, 10).until(expected_conditions.visibility_of_element_located(topic_title))

        elements = self.driver.find_elements(*topic_title)
        assert len(elements) > 0
        for element in elements:
            assert keyword in element.text.lower()

这是显示等待的例子


一、po是什么?

示例:只有业务,用来解决抽象问题马丁福2015年page project
代码如下(示例)帮助文档:

https://www.selenium.dev/zh-cn/documentations/test_practices/encouraged/page_object_models/

page-object
POM的建模原则
不要暴漏页面的内部细节
只提供方法
方法可以返回其他的PO


POM原则

一、driver层的封装
这一层主要是对于webdriver方法的封装,这里来举一个栗子,最常用的定位方法,之前讲过统一定位方法的三种传参格式:webdriver的所有定位方法,使用find_element()方法通过BY类、字符串、元组三种方法传递定位类型和数据,这里我使用元组的形式(例如locator = ("id","name_box"))来传递参数。

1.1 定位元素方法封装示例

这个定位元素的公共方法中,加了很多东西;如果每次定位的时候写这些异常捕获、打印操作的话,那么程序会非常臃肿,所以需要单独提出来,每次需要定位的时候统一调用这个方法。

有一个入参locator,格式为("定位类型","定位参数值"),返回我们所定位到的元素
加入了元素等待,并判断该元素是否存在
对于关键信息的打印输出,方便定位监控
加入了异常捕获,定位失败后可以继续执行程序
 def find_element(self, *locator):
        try:
            print("定位元素:%s" % (locator,))
            return WebDriverWait(self.driver, 10).until(EC.presence_of_element_located(locator))
        except Exception as msg:
            print(u"%s 页面中未能找到 %s 元素" % (self, locator))
            print("错误信息%s" % msg)
1.2 封装类的初始化

对于webdriver的封装,我们要先创建一个class,这样方便我们继承调用这些封装的方法。在class中,我这里设计了一个初始化,每次调用封装的driver时,传递三个参数,一个必填项:driver、两个非必填项:page_url、page_title,我这样的想法是每次引用这个封装类时,传递一个driver进来,如果有打开网址页面的需要,则传递网址和网址页面的title,这样也可以做一次校验。

def __init__(self, driver, page_url=None, page_title=None):
        self.page_url = page_url
        self.page_title = page_title
        self.driver = driver
        self.driver.maximize_window()
        self.driver.implicitly_wait(30)
二、page层书写
page类在继承我们封装的webdriver后,主要写具体的操作步骤,例如输入登录名、输入登录密码、点击登录按钮等操作。

2.1继承pagedriver并初始化

这里的page层要继承pagedriver的类Action,然后在page层的初始化中,初始化Action。

from common.pagedriver import Action
class Login(Action):
 
    def __init__(self, driver, page_url=None, page_title=None):
        Action.__init__(self, driver, page_url, page_title)
2.2 操作步骤
比如我要写打开页面、输入用户名这两个方法:

其中元素定位放在类变量中,而登录账号我们放在case层来输入。

from common.pagedriver import Action
 
class Login(Action):
    input_name_loc = ("xpath", "//input[@placeholder='邮箱帐号或手机号码']")
    frame_loc = (0)
 
    def __init__(self, driver, page_url=None, page_title=None):
        Action.__init__(self, driver, page_url, page_title)
 
    def open(self):
        """打开页面"""
        self._open(self.page_url, self.page_title)
 
    def input_name(self, login_name):
        """输入登录名"""
        self.send_keys(self.input_name_loc, login_name)
三、case层调用
终于到了第三层,这里我们要做的就是把page层的方法,像搭积木一样搭起来,并且连成完整的操作。

3.1 使用unittest,并初始化数据

在unittest的框架基础上,主要是在setUp()方法中初始化我们的数据,例如网址、账号、driver的初始化

import unittest
from selenium import webdriver
 
 
class Demo(unittest.TestCase):
 
    def setUp(self):
        self.url = "https://mail.163.com/"
        self.title = "网易"
        self.user_name = ""  # 登录账户
        self.user_password = ""  # 登录密码
        self.driver = webdriver.Chrome()
 
    def tearDown(self):
        self.driver.close()
 
 
if __name__ == "__main__":
    unittest.main(3.2 调用方法,完成用例
首先我们引用page层,然后使用page层的方法搭建case。

from page.login_page import Login
 
    def test_login(self):
        login_page = Login(self.driver, self.url, self.title)
        login_page.open()
        login_page.input_name(self.user_name)
四、实例演示:登录163网易邮箱
通过上述的分层步骤,演示登录163邮箱的操作,登录后通过断言登陆成功页面title,来判断是否登录成功。

运行结果:

打开网址:https://mail.163.com/
网址预期标题: 网易
定位元素:('xpath', "//input[@placeholder='邮箱帐号或手机号码']")
输入值:
定位元素:('xpath', "//input[@placeholder='输入密码']")
输入值:
定位元素:('xpath', "//input[@placeholder='输入密码']")
输入值:
(26封未读) 网易邮箱6.0版
.
----------------------------------------------------------------------
Ran 1 test in 202.126s
 
OK
运行代码:

pagedriver.py

from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.ui import WebDriverWait
 
 
class Action(object):
    """
     Action封装所有页面都公用的方法
    """
 
    # 初始化driver、url、title等
    def __init__(self, driver, page_url=None, page_title=None):
        self.page_url = page_url
        self.page_title = page_title
        self.driver = driver
        self.driver.maximize_window()
        self.driver.implicitly_wait(30)
 
    def open(self):
        """
        定义open方法,调用_open()进行打开链接
        """
        self._open(self.page_url, self.page_title)
 
    def on_page(self, page_title):
        """
        使用current_url获取当前窗口Url地址,进行与配置地址作比较,返回比较结果(True False)
        """
        return page_title in self.driver.title
 
    def _open(self, page_url, page_title):
        """
        打开页面,校验页面链接是否加载正确
        """
        # 使用get打开访问链接地址
        if page_url and page_title is not None:
            self.driver.get(page_url)
            print("打开网址:%s" % page_url)
            print("网址预期标题: %s" % page_title)
            # 使用assert进行校验,打开的链接地址是否与配置的地址一致。调用on_page()方法
            assert self.on_page(page_title), u"打开页面%s失败" % page_url
 
    def find_element(self, *locator):
        try:
            print("定位元素:%s" % (locator,))
            return WebDriverWait(self.driver, 20).until(EC.presence_of_element_located(locator))
        except Exception as msg:
            print(u"%s 页面中未能找到 %s 元素" % (self, locator))
            print("错误信息%s" % msg)
 
    def send_keys(self, locator, value, clear_first=True):
        """
        重写定义send_keys方法
        """
        element = self.find_element(*locator)
        if clear_first:
            element.clear()
            element.send_keys(value)
        else:
            element.send_keys(value)
        print("输入值:%s" % value)
 
    def switch_frame(self, frame_loc):
        """
        切换frame,
        :param frame_loc:id、name、element、index
        :return:
        """
        self.driver.switch_to.frame(frame_loc)
login_page.py

from common.pagedriver import Action
from selenium.webdriver.common.keys import Keys
 
 
class Login(Action):
    input_name_loc = ("xpath", "//input[@placeholder='邮箱帐号或手机号码']")
    input_password_loc = ("xpath", "//input[@placeholder='输入密码']")
    enter_login_loc = Keys.ENTER
    frame_loc = (0)
 
    def __init__(self, driver, page_url=None, page_title=None):
        Action.__init__(self, driver, page_url, page_title)
 
    def open(self):
        """打开页面"""
        self._open(self.page_url, self.page_title)
 
    def change_frame(self):
        """切换frame"""
        self.switch_frame(self.frame_loc)
 
    def input_name(self, login_name):
        """输入登录名"""
        self.send_keys(self.input_name_loc, login_name)
 
    def input_password(self, login_password):
        """输入密码"""
        self.send_keys(self.input_password_loc, login_password)
 
    def enter_login(self):
        """模拟登陆点击回车"""
        self.send_keys(self.input_password_loc, self.enter_login_loc, False)
 
    def get_login_message(self):
        """获取登录后的信息以断言"""
        return self.driver.title
test_163_login.py

# -*- coding: utf-8 -*-
import unittest
from time import sleep
from page.login_page import Login
from selenium import webdriver
 
 
class Demo(unittest.TestCase):
 
    def setUp(self):
        self.url = "https://mail.163.com/"
        self.title = "网易"
        self.user_name = ""  # 登录账户
        self.user_password = ""  # 登录密码
        self.driver = webdriver.Chrome()
 
    def test_wangyi_login(self):
        """登录网易邮箱"""
        login_page = Login(self.driver, self.url, self.title)
        login_page.open()
        login_page.change_frame()
        sleep(3)
        login_page.input_name(self.user_name)
        login_page.input_password(self.user_password)
        sleep(2)
        login_page.enter_login()
        sleep(5)
        print(login_page.get_login_message())
        assert "网易邮箱6.0版" in login_page.get_login_message()
 
    def tearDown(self):
        self.driver.close()
 
 
if __name__ == "__main__":
    unittest.main()

1.wait assert

代码如下(示例):

# Generated by Selenium IDE
import pytest
import time
import json
from selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.common.action_chains import ActionChains
from selenium.webdriver.support import expected_conditions
from selenium.webdriver.support.wait import WebDriverWait
from selenium.webdriver.common.keys import Keys
from selenium.webdriver.common.desired_capabilities import DesiredCapabilities

# 等待断言
class TestSearch():
    def setup_method(self, method):
        self.driver = webdriver.Edge()
        self.driver.implicitly_wait(5)
        self.vars = {}

    def teardown_method(self, method):
        self.driver.quit()

    def test_search(self):
        self.driver.get("https://ceshiren.com/")
        # self.driver.set_window_size(1920, 993)
        self.driver.maximize_window()
        self.driver.find_element(By.CSS_SELECTOR, ".d-icon-search").click()
        self.driver.find_element(By.CSS_SELECTOR, ".d-icon-sliders-h").click()
        self.driver.find_element(By.CSS_SELECTOR, '.search-query').send_keys("selenium")
        self.driver.find_element(By.CSS_SELECTOR, ".search-cta > .fa").click()
        assert len(self.driver.find_elements(By.CSS_SELECTOR, '.search-link')) > 0
        assert 'selenium' in self.driver.find_element(By.CSS_SELECTOR, '.search-link').text.lower()
        # self.driver.find_element(By.CSS_SELECTOR, '.search-link').click()
        # self.driver.find_element(By.LINK_TEXT, "selenium-Selenium 1 (Selenium RC)").click()

2.po的使用

代码如下(示例)search_page.py:

from datetime import datetime
from time import sleep

from selenium.webdriver.common.by import By
from selenium.webdriver.remote.webdriver import WebDriver
from selenium.webdriver.support import expected_conditions
from selenium.webdriver.support.wait import WebDriverWait


class SearchPage:

    def __init__(self, driver):
        self.driver: WebDriver = driver

    def highlight(self, locator):
        # self.driver.execute_script(f"document.querySelector('{locator}').style.border='solid red 5px'")
        # arguments
        self.driver.execute_script(f"arguments[0].style.border='solid red 5px'",
                                   self.driver.find_element(*locator))
        sleep(0.5)
        self.driver.save_screenshot(str(datetime.now().timestamp()) + '.png')

    def search(self, keyword, min=None, max=None) -> 'SearchPage':
        search_query = (By.CSS_SELECTOR, '.full-page-search.search-query')
        self.highlight(search_query)
        self.driver.find_element(*search_query).send_keys('%s' % keyword)
        # self.driver.find_element(By.CSS_SELECTOR, '.search-advanced-additional-options').click()

        topic_number = (By.XPATH, '//summary[contains(., "话题浏览量")]')
        input_min = (By.CSS_SELECTOR, '[aria-label="按最小浏览量筛选"]')
        # WebDriverWait(self.driver, 5).until(expected_conditions.element_to_be_clickable(input_min))
        # WebDriverWait(self.driver, 5).until(expected_conditions.element_to_be_clickable(topic_number))
        self.highlight(topic_number)
        self.driver.find_element(*topic_number).click()

        def wait(driver):
            # 编写显示等待的逻辑
            # 点击一个控件有时会失效 一直重试
            self.highlight(topic_number)
            self.driver.find_element(*topic_number).click()
            # 直到某个控件出现
            return self.driver.find_element(*input_min).is_displayed()

        # 显式等待 使用自定义函数
        WebDriverWait(self.driver, 5).until(wait)

        self.highlight(input_min)
        self.driver.find_element(*input_min).send_keys(min)

        input_max = (By.CSS_SELECTOR, '[aria-label="按最大浏览量筛选"]')
        self.highlight(input_max)
        self.driver.find_element(*input_max).send_keys(max)

        search = (By.CSS_SELECTOR, '.search-cta[aria-label="搜索"]')
        self.highlight(search)
        self.driver.find_element(*search).click()
        return self

    def get_search_results(self) -> list[str]:
        topic_title = (By.CSS_SELECTOR, '.topic-title')
        # 显式等待
        WebDriverWait(self.driver, 10).until(expected_conditions.visibility_of_element_located(topic_title))

        elements = self.driver.find_elements(*topic_title)
        r = []
        for element in elements:
            r.append(element.text)
        return r

    def get_search_non_results(self) -> list[str]:
        ...

    def clear(self) -> 'SearchPage':
        self.driver.find_element(By.XPATH, '//summary[contains(., "高级筛选器")]').click()
        self.driver.find_element(By.CSS_SELECTOR, '.full-page-search.search-query').clear()
        return self


代码如下(示例)topic_page.py:

from __future__ import annotations

from selenium import webdriver
from selenium.webdriver.common.by import By

from shixun2.web.tests.ceshiren.search.pages.search_page import SearchPage


class TopicPage:
    _search_button = (By.ID, 'search-button')
    _advance_button = (By.CSS_SELECTOR, '[title=打开高级搜索]')

    def __init__(self):
        self.driver = webdriver.Edge()
        # 定位控件的最大超时时间 隐式等待
        self.driver.implicitly_wait(3)
        self.driver.get('https://ceshiren.com/t/topic/25780')

    def get_topic(self) -> TopicPage:
        ...

    def search(self) -> SearchPage:
        self.driver.find_element(*self._search_button).click()
        self.driver.find_element(*self._advance_button).click()
        return SearchPage(self.driver)


代码如下(示例)test_search_advance_po.py:

import pytest

from shixun2.web.tests.ceshiren.search.pages.topic_page import TopicPage


class TestSearchAdvance:
    def setup_class(self):
        topic_page = TopicPage()
        self.search_page = topic_page.search()

    def setup_method(self):
        ...

    def teardown_method(self):
        self.search_page.clear()

    @pytest.mark.parametrize('keyword, min, max', [
        ['selenium', 10, 1000],
        ['web', 10, 1000],
        ['selenium', 0, 1000],
        ['selenium', 100, 1000],
        # 拆分异常case
        # ['selenium', 100, 0],

    ])
    def test_search(self, keyword, min, max):
        self.search_page.search(keyword, min, max)
        results = self.search_page.get_search_results()
        assert len(results) > 0
        for text in results:
            assert keyword in text.lower()


@pytest.mark.parametrize('keyword, min, max', [
    ['selenium20230609', 10, 1000],
    # 拆分异常case
    # ['selenium', 100, 0],

])
def test_search_non(self, keyword, min, max):
    self.search_page.search(keyword, min, max)
    results = self.search_page.get_search_results()
    assert len(results) == 0


在这里插入图片描述

3.出现问题如何解决

代码如下(示例):

常见问题
selenium.common.exceptions.StaleElementReferenceException: Message: stale element reference: stale element not found

解决方案:等待页面稳定,使用显式等待

selenium.common.exceptions.ElementClickInterceptedException: Message: Element is not clickable at point (1120,161) because another element

obscures it
解决方案: 显式等待

例如:显示等待


4.POM原则

代码如下(示例):


PO的六大原则
class 使用类代表页面
method 使用公开方法代表页面所提供的功能
params 用参数代表一个功能的输入数据,用返回值代表一个功能的结果
return 一个方法的结果可以是页面对象,可以是当前页面对象自己,或者其他的页面对象。也可以是返回的数据
testcase 调用页面对象,使用初始化实例、调用方法、获取数据进行断言,用到测试前置、测试数据、测试的步骤,断言。
automation 页面对象的初始化方法构造与传参,具体的页面对象方法自动化过程与传参、定位符

这是显示等待的例子


1.wait assert

代码如下(示例):



这是显示等待的例子


1.wait assert

代码如下(示例):



这是显示等待的例子


总结

提示:这里对文章进行总结:

例如:这是第五天实训内容。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

6月编程排行榜出炉,Python还是第一吗?

除了自己的薪水,程序员每个月最关注的就要数编程排行榜啦~快跟着播妞一起看看最新编程排行情况: Tiobe编程排行前15名 历年排行变化趋势图 详细榜单可参考官网:https://www.tiobe.com/tiobe-index/ 本月排行看点:没有意外&#…

又一经典之作开源,腾讯云架构师珍藏版:“redis深度笔记”

前言 作为这个时代码代码的秃头人员,对Redis肯定是不陌生的,如果连Redis都没用过,还真不好意思出去面试,指不定被面试官吊打多少次。 毕竟现在互联网公司和一些创业公司都要用到Redis,像亚马逊、谷歌、阿里、腾讯都要…

太赞了!阿里技术团队《Java 面试官手册》突击版免费对外开放!

现在已经不是几年前了,可能路边的大爷都知道学计算机挣钱了, 突然想起一个段子,就是当路边的大爷都开始进场买股票时,说明市场已经很危险了。不过大家大可放心,大爷永远无法进场搞计算机,大家放心卷。因为公…

帆软10移动端适配过程记录

背景 最近项目中嵌入的帆软报表在移动端进行适配,支持手机端展示,版本升级在前期(帆软FineReport之版本升级)有过介绍,本期主要记录适配过程出现的问题及处理方案。 异常处理 1、请求找不到js文件 问题描述 在手机…

活动笔记丨物业行业人效提升与灵活用工新路径

近日,盖雅工场成功举办物业行业人效提升专场交流,来自广深地区央企和民营的领先物业企业和现场服务业的多位代表齐聚深圳招商积余大厦,共同研讨行业人效提升的挑战和实践。 本次闭门交流会聚焦于人效提升,讨论话题包括各自企业在人…

[C语言实现]数据结构——手撕顺序栈之我出生就会写一个栈

🥰作者: FlashRider 🌏专栏: 数据结构 目录 栈的前置知识 1.什么是栈? 2.生活中哪些地方有栈的影子? 顺序表实现栈 1.为什么通常采用顺序表实现栈? 2.栈的实现 栈的前置知识 1.什么是栈? 栈&#xf…

spring实例化bean属性填充

实例化bean之后要执行属性填充,也就是serviceA注入了servcieB,实例化servcieA时要通过populateBean先实例化serviceB. 也就是最终要执行serviceB的getBean 只是字段注入的流程 关键的两个Processor AutowiredAnnotationBeanPostProcessor 处理Autowire…

Databend 开源周报第 96 期

Databend 是一款现代云数仓。专为弹性和高效设计,为您的大规模分析需求保驾护航。自由且开源。即刻体验云服务:https://app.databend.cn 。 Whats On In Databend 探索 Databend 本周新进展,遇到更贴近你心意的 Databend 。 虚拟列 查询 J…

Angular与PDF之四: 反思代码与模板的复用

在我们前面关于Angular与PDF的几篇博客中分别讲了如何在在如何在客户端渲染PDF(Angular与PDF之一:如何在客户端渲染PDF_angular pdf_KenkoTech的博客-CSDN博客) 和预览(Angular 与PDF之二:打印预览的实现_angular pdf预览_KenkoTech的博客-CS…

Easeui 02 tree组件.

1.添加tree组件. tree组件的位置:DataGrid and Tree(表格和树) → tree(树); 复制 tree组件到 "菜单管理"的div里面,如: 这里要动态绑定数据,所以把死数据删除,只留下一个 ul,如&am…

HCIA-动态路由

目录 动态路由: 动态路由的分类 按工作区域分类: 按算法和工作机制分类: 距离矢量路由协议: 链路状态路由协议: OSPF协议计算路由步骤: OSPF协议 OSPF协议报文: OSPF三张表 OSPF路由…

算法修炼之筑基篇——筑基一层后期(解决KMP算法,KMP算法模板)

✨博主:命运之光​​​​​​ 🦄专栏:算法修炼之练气篇​​​​​ 🍓专栏:算法修炼之筑基篇 ✨博主的其他文章:点击进入博主的主页​​​​​​ 前言:学习了算法修炼之练气篇想必各位蒟蒻们的基…

kafka 四 Kafka读写流程、LEO log end offset、物理存储 稠密索引 稀疏索引 、Kafka物理存储、深入了解读数据流程、删除消息

目录 Kafka读写流程 LEO log end offset 物理存储 稠密索引 稀疏索引 Kafka物理存储 深入了解读数据流程 删除消息 Kafka读写流程 写流程: 通过zookeeper 找leader分配开始读写Isr中的副本同步数据,并返回给leader ack返回给 分片ack 读流程&…

2023高考语文,用ChatGPT挑战全国卷作文,已达到双一流高校学生水平?

前言 2023年高考语文结束啦,今天我们用ChatGPT来挑战高考作文,一起来看看它的表现如何?ChatGPT突然爆火网络,它真的会取代人类的工作吗? 什么是ChatGPT? ChatGPT是由OpenAI开发的,OpenAI是一家…

BBA EDI 项目数据库方案开源介绍

近期为了帮助广大用户更好地使用 EDI 系统,我们根据以往的项目实施经验,将成熟的 EDI 项目进行开源。用户安装好知行之桥EDI系统之后,只需要下载我们整理好的示例代码,并放置在知行之桥指定的工作区中,即可开始使用。 …

排序算法的复杂度及稳定性详解(内含记忆小窍门)

排序算法的复杂度及稳定性 一、排序算法分类二、概念2.1 时间复杂度2.2 空间复杂度2.3 稳定性 三、表格比较注意 四、部分排序分析4.1 直接插入排序图示代码 4.2 冒泡排序图示代码 4.3 快速排序图示代码 五、结构化记忆(小窍门)5.1 结构化5.2 我的结构化…

2023 如何备考系统架构师?

高级系统架构设计师难度还是有的,所以一般千万不要裸考!!要时间充足,至少要接触过,反正没有基础的尽量还是不要去裸考了! 一、系统架构设计师考试题型 考试科目分为综合题(选择题)&a…

Stable Diffusion最全保姆级安装教程(建议收藏)

Midjourney 因细致的画图风格备受大家的欢迎,但由于其网络环境以及会员费,导致入门门槛过高,拦住了很多对AIGC感兴趣的小伙伴。 今天阿良就教大家,不需要魔法,也不用交会员费,尽情玩转AI出图的保姆级安装教…

力扣算法系统刷题详细题解记录二(字符串、双指针法、栈与队列)

力扣算法系统刷题题解记录二(字符串、双指针法、栈与队列) 前言 参考顺序和资料:《代码随想录》 二刷要认真做笔记啦,加油! 笔记模板: #### 解题思路#### 示意图#### 代码四、字符串 344.字符串反转 编…

求最小生成树(Kruskal算法和Prim算法)

目录 一、前言 二、相关概念 1、最小生成树 2、Prim算法(对结点进行操作) 3、kruskal 算法(对边进行操作) 三、例题 1、修建公路(lanqiaoOJ题号1124) 1、Prim算法题解 2、Kruskal算法 一、前言 很…