slint esp32 tokio

news2025/1/4 22:10:17

源码:https://github.com/xiaguangbo/slint_esp32_tokio
cpu 是 esp32c2,屏幕是 ili9341,触摸是 xpt2046,使用 spi 半双工
不使用DMA(esp-rs还没支持),SPI 40M,240*320全屏刷新为1.5秒,虽然比不了 lvgl,但类lvgl的slint是目前rust跨全平台唯一的选择
这是一个游戏,翻到两个一样的就成功,slint官网有入门示例,就是这个,然后把 .slint 和 控制逻辑拿过来直接用。就是slint平台需要稍微移植下,字体会自动打包
在这里插入图片描述

Cargo.toml

[package]
name = "esp32c2"
version = "0.1.0"
authors = ["xxx"]
edition = "2021"
resolver = "2"
rust-version = "1.71"

[profile.release]
opt-level = "s"

[profile.dev]
debug = true    # Symbols are nice and they don't increase the size on Flash
opt-level = "z"

[features]
default = ["std", "embassy", "esp-idf-svc/native"]

pio = ["esp-idf-svc/pio"]
std = ["alloc", "esp-idf-svc/binstart", "esp-idf-svc/std"]
alloc = ["esp-idf-svc/alloc"]
nightly = ["esp-idf-svc/nightly"]
experimental = ["esp-idf-svc/experimental"]
embassy = [
    "esp-idf-svc/embassy-sync",
    "esp-idf-svc/critical-section",
    "esp-idf-svc/embassy-time-driver",
]

[dependencies]
log = { version = "*", default-features = false }
esp-idf-svc = { version = "*", default-features = false }

tokio = { version = "*", features = ["rt", "time", "sync"] }
num-traits = "*"
chrono = "*"
rand = "*"
slint = { version = "*", default-features = false, features = [
    "compat-1-2",
    "renderer-software",
    "unsafe-single-threaded",
] }

[build-dependencies]
embuild = "*"
slint-build = "*"

appwindow.slint

struct TileData {
    image: image,
    image_visible: bool,
    solved: bool,
}

component MemoryTile inherits Rectangle {
    in property <bool> open_curtain;
    in property <bool> solved;
    in property <image> icon;
    callback clicked;

    height: 50px;
    width: 50px;
    border-radius: self.width / 2;
    background: solved ? #34CE57 : #3960D5;
    clip: true;

    animate background { duration: 800ms; }

    Image {
        source: icon;
        width: parent.width;
        height: parent.height;
    }

    // Left curtain
    Rectangle {
        background: #193076;
        x: 0px;
        width: open_curtain ? 0px : (parent.width / 2);
        height: parent.height;
        clip: true;

        animate width {
            duration: 250ms;
            easing: ease-in;
        }

        Image {
            width: root.width - 25px;
            height: root.height - 25px;
            x: 13px;
            y: 13px;
            source: @image-url("../icons/tile_logo.png");
        }
    }

    // Right curtain
    Rectangle {
        background: #193076;
        x: open_curtain ? parent.width : (parent.width / 2);
        width: open_curtain ? 0px : (parent.width / 2);
        height: parent.height;
        clip: true;

        animate width {
            duration: 250ms;
            easing: ease-in;
        }
        animate x {
            duration: 250ms;
            easing: ease-in;
        }

        Image {
            width: root.width - 25px;
            height: root.height - 25px;
            x: parent.width - self.width - 13px;
            y: 13px;
            source: @image-url("../icons/tile_logo.png");
        }
    }

    TouchArea {
        clicked => {
            // Delegate to the user of this element
            root.clicked();
        }

        width: 100%;
        height: 100%;
    }
}

export component AppWindow inherits Window {
    width: 240px;
    height: 320px;

    callback check_if_pair_solved();
    // Added
    in property <bool> disable_tiles;
    // Added

    in-out property <[TileData]> memory_tiles: [
        { image: @image-url("../icons/at.png") },
        { image: @image-url("../icons/balance-scale.png") },
        { image: @image-url("../icons/bicycle.png") },
        { image: @image-url("../icons/bus.png") },
        { image: @image-url("../icons/cloud.png") },
        { image: @image-url("../icons/cogs.png") },
        { image: @image-url("../icons/motorcycle.png") },
        { image: @image-url("../icons/video.png") },
    ];

    for tile[i] in memory_tiles: MemoryTile {
        x: mod(i, 4) * (root.width / 4);
        y: floor(i / 4) * (root.width / 4);
        width: 50px;
        height: 50px;
        icon: tile.image;
        open_curtain: tile.image_visible || tile.solved; // 任何一个满足都打开帘子
        // propagate the solved status from the model to the tile
        solved: tile.solved;
        clicked => {
            // old: tile.image_visible = !tile.image_visible;
            // new:
            // 可不可以点击
            if (!root.disable_tiles) {
                tile.image_visible = !tile.image_visible;
                root.check_if_pair_solved();
            }
        }
    }
}

ui

use std::{borrow::Borrow, cell::RefCell, rc::Rc};

use slint::platform::{software_renderer::*, PointerEventButton, WindowAdapter, WindowEvent};
use slint::Model;
use tokio::time;

use esp_idf_svc::hal::{gpio::*, peripheral::*, spi::*};

use crate::component::{ili9341, xpt2046};

slint::include_modules!();

pub async fn work<SPI, CS, CS2, DC>(spi1: SPI, spi2: SPI, cs1: CS, cs2: CS2, dc: DC)
where
    SPI: Borrow<SpiDriver<'static>> + 'static,
    CS: Peripheral<P = CS> + OutputPin,
    CS2: Peripheral<P = CS2> + OutputPin,
    DC: Peripheral<P = DC> + OutputPin,
{
    let mut ili9341 = ili9341::ILI9341::new(spi1, cs1, dc);
    let xpt2046 = xpt2046::XPT2046::new(spi2, cs2);

    ili9341.open();

    let buffer_provider = DrawBuffer {
        display: ili9341,
        buffer: vec![Rgb565Pixel::default(); ili9341::ILI9341_WIDTH as usize].leak(),
    };

    slint::platform::set_platform(Box::new(SlintBackend {
        window: Default::default(),
        now: std::time::Instant::now().into(),
        buffer_provider: buffer_provider.into(),
        touch: xpt2046.into(),
        last_touch: None.into(),
    }))
    .unwrap();

    let main_window = AppWindow::new().unwrap();

    // Fetch the tiles from the model
    let mut tiles: Vec<TileData> = main_window.get_memory_tiles().iter().collect();
    // Duplicate them to ensure that we have pairs
    tiles.extend(tiles.clone());

    // Randomly mix the tiles
    use rand::seq::SliceRandom;
    let mut rng = rand::thread_rng();
    tiles.shuffle(&mut rng);

    // Assign the shuffled Vec to the model property
    let tiles_model = std::rc::Rc::new(slint::VecModel::from(tiles));
    main_window.set_memory_tiles(tiles_model.clone().into());

    let main_window_weak = main_window.as_weak();
    // 点击的回调函数
    main_window.on_check_if_pair_solved(move || {
        // 如果元素的(image_visible && !solved)为真,则得到他
        // 就是被打开看的且没有被标记的对象
        let mut flipped_tiles = tiles_model
            .iter()
            .enumerate()
            .filter(|(_, tile)| tile.image_visible && !tile.solved);

        // 当检查出有两个这样的元素就进入判断
        if let (Some((t1_idx, mut t1)), Some((t2_idx, mut t2))) =
            (flipped_tiles.next(), flipped_tiles.next())
        {
            let is_pair_solved = t1 == t2; // 比较两个元素的值是不是一样的,包括图片的 rgba 和元素属性,也就是 TileData 的所有成员

            // 一样
            if is_pair_solved {
                t1.solved = true; // 彻底打开帘子
                tiles_model.set_row_data(t1_idx, t1);
                t2.solved = true;
                tiles_model.set_row_data(t2_idx, t2);
            }
            // 不一样
            else {
                let main_window = main_window_weak.unwrap();
                main_window.set_disable_tiles(true); // 防止继续点击
                let tiles_model = tiles_model.clone();

                // 延时 1s
                slint::Timer::single_shot(std::time::Duration::from_secs(1), move || {
                    main_window.set_disable_tiles(false); // 可继续点击
                    t1.image_visible = false; // 关闭帘子
                    tiles_model.set_row_data(t1_idx, t1);
                    t2.image_visible = false;
                    tiles_model.set_row_data(t2_idx, t2);
                });
            }
        }
    });

    loop {
        slint::run_event_loop().unwrap();
        time::sleep(time::Duration::from_millis(20)).await;
    }
}

pub struct SlintBackend<'a, SPI, DC>
where
    SPI: Borrow<SpiDriver<'a>> + 'a,
    DC: Peripheral<P = DC> + OutputPin,
{
    window: RefCell<Option<Rc<MinimalSoftwareWindow>>>,
    now: RefCell<std::time::Instant>,
    buffer_provider: RefCell<DrawBuffer<'a, SPI, DC>>,
    touch: RefCell<xpt2046::XPT2046<'a, SPI>>,
    last_touch: RefCell<Option<slint::LogicalPosition>>,
}

impl<'a, SPI, DC> slint::platform::Platform for SlintBackend<'a, SPI, DC>
where
    SPI: Borrow<SpiDriver<'a>> + 'a,
    DC: Peripheral<P = DC> + OutputPin,
{
    fn create_window_adapter(&self) -> Result<Rc<dyn WindowAdapter>, slint::PlatformError> {
        let window = MinimalSoftwareWindow::new(RepaintBufferType::ReusedBuffer);

        self.window.replace(Some(window.clone()));
        self.window
            .borrow()
            .as_ref()
            .unwrap()
            .set_size(slint::PhysicalSize::new(
                ili9341::ILI9341_WIDTH as u32,
                ili9341::ILI9341_HEIGHT as u32,
            ));

        Ok(window)
    }

    fn duration_since_start(&self) -> std::time::Duration {
        self.now.borrow().elapsed()
    }

    fn run_event_loop(&self) -> Result<(), slint::PlatformError> {
        let while_now = std::time::Instant::now();
        let mut touch_now = std::time::Instant::now();
        let mut touch_ed = false;

        // 连续绘制达到 100ms 就跳过
        while while_now.elapsed().as_millis() < 100 {
            slint::platform::update_timers_and_animations();

            if let Some(window) = self.window.borrow().clone() {
                if !touch_ed {
                    touch_ed = !touch_ed;
                    touch_now = std::time::Instant::now();

                    if let Some(event) = match self.touch.borrow_mut().read() {
                        Some(v) => {
                            let position = slint::PhysicalPosition::new(
                                (v.x * ili9341::ILI9341_WIDTH as f32) as i32,
                                (v.y * ili9341::ILI9341_HEIGHT as f32) as i32,
                            )
                            .to_logical(window.scale_factor());

                            Some(match self.last_touch.borrow_mut().replace(position) {
                                Some(_) => WindowEvent::PointerMoved { position },
                                _ => WindowEvent::PointerPressed {
                                    position,
                                    button: PointerEventButton::Left,
                                },
                            })
                        }
                        _ => self.last_touch.borrow_mut().take().map(|position| {
                            WindowEvent::PointerReleased {
                                position,
                                button: PointerEventButton::Left,
                            }
                        }),
                    } {
                        let is_pointer_release_event =
                            matches!(event, WindowEvent::PointerReleased { .. });

                        window.dispatch_event(event);

                        if is_pointer_release_event {
                            window.dispatch_event(WindowEvent::PointerExited);
                        }
                    }
                } else {
                    if touch_now.elapsed().as_millis() >= 20 {
                        // 每隔一段时间才能再次读取触摸,避免频繁处理
                        touch_ed = !touch_ed;
                    }
                }

                window.draw_if_needed(|renderer| {
                    renderer.render_by_line(&mut *self.buffer_provider.borrow_mut());
                });

                if !window.has_active_animations() {
                    // 如果没有需要绘制的东西就跳出,否则就继续绘制
                    break;
                }
            }
        }
        Ok(())
    }
}

struct DrawBuffer<'a, SPI, DC>
where
    SPI: Borrow<SpiDriver<'a>> + 'a,
    DC: Peripheral<P = DC> + OutputPin,
{
    display: ili9341::ILI9341<'a, SPI, DC>,
    buffer: &'a mut [Rgb565Pixel],
}

impl<'a, SPI, DC> LineBufferProvider for &mut DrawBuffer<'a, SPI, DC>
where
    SPI: Borrow<SpiDriver<'a>> + 'a,
    DC: Peripheral<P = DC> + OutputPin,
{
    type TargetPixel = Rgb565Pixel;

    fn process_line(
        &mut self,
        line: usize,
        range: std::ops::Range<usize>,
        render_fn: impl FnOnce(&mut [Rgb565Pixel]),
    ) {
        let buffer = &mut self.buffer[range.clone()];

        render_fn(buffer);
        self.display.write_pixel_slint(
            range.start as u16,
            line as u16,
            range.end as u16,
            line as u16,
            &buffer,
        );
    }
}

work

use std::{rc, thread};

use tokio::{runtime, task, time};

use esp_idf_svc::hal::{gpio, peripherals, spi};

use crate::module::*;

pub fn work() {
    thread::Builder::new()
        .stack_size(8 * 1024)
        .spawn(|| {
            task::LocalSet::new().block_on(
                &runtime::Builder::new_current_thread()
                    .enable_all()
                    .build()
                    .unwrap(),
                async {
                    let peripherals = peripherals::Peripherals::take().unwrap();

                    let spi = spi::SpiDriver::new::<spi::SPI2>(
                        peripherals.spi2,
                        peripherals.pins.gpio0,
                        peripherals.pins.gpio1,
                        Option::<gpio::AnyIOPin>::None,
                        &spi::SpiDriverConfig::new(),
                    )
                    .unwrap();

                    let spi = rc::Rc::new(spi);
                    let spi_1 = spi.clone();
                    let spi_2 = spi.clone();

                    task::spawn_local(async move {
                        ui::work(
                            spi_1,
                            spi_2,
                            peripherals.pins.gpio3,
                            peripherals.pins.gpio4,
                            peripherals.pins.gpio2,
                        )
                        .await;
                    });

                    loop {
                        time::sleep(time::Duration::MAX).await;
                    }
                },
            );
        })
        .unwrap();
}

build

fn main() {
    embuild::espidf::sysenv::output();

    slint_build::compile_with_config(
        "ui/appwindow.slint",
        slint_build::CompilerConfiguration::new()
            .embed_resources(slint_build::EmbedResourcesKind::EmbedForSoftwareRenderer),
    )
    .unwrap();
}

ili9341

use std::borrow::*;

use esp_idf_svc::hal::{delay::Delay, gpio::*, peripheral::*, prelude::*, spi::*};

use slint::platform::software_renderer::Rgb565Pixel;

pub const ILI9341_WIDTH: u16 = 240;
pub const ILI9341_HEIGHT: u16 = 320;

/*
# 初始化
第一个字节是命令
等待5ms

36, 48 左上右下竖屏、c8 右下左上竖屏、e8 左下右上横屏、28右上左下横屏
3a, 55 像素格式 565
11     退出睡眠
29     开显示、28 关显示。不会更改内存内容

# 设置区域
可设置一点、一行或一个方块,方块区域会自动换行

2a x坐标
16bit xs
16bit xe

2b y坐标
16bit ys
16bit ye

2c
16bit 565 颜色数据
*/

pub struct ILI9341<'a, SPI, DC>
where
    SPI: Borrow<SpiDriver<'a>> + 'a,
    DC: Peripheral<P = DC> + OutputPin,
{
    spi: SpiDeviceDriver<'a, SPI>,
    dc: PinDriver<'a, DC, Output>,
    line_buf: Vec<u8>,
}

impl<'a, SPI, DC> ILI9341<'a, SPI, DC>
where
    SPI: Borrow<SpiDriver<'a>> + 'a,
    DC: Peripheral<P = DC> + OutputPin,
{
    pub fn new<CS>(spi: SPI, cs: CS, dc: DC) -> Self
    where
        CS: Peripheral<P = CS> + OutputPin,
    {
        let config = config::Config::default()
            .baudrate(40.MHz().into())
            .duplex(config::Duplex::Half3Wire);

        Self {
            spi: SpiDeviceDriver::new(spi, Some(cs), &config).unwrap(),
            dc: PinDriver::output(dc).unwrap(),
            line_buf: vec![0u8; (ILI9341_WIDTH * 2) as usize],
        }
    }

    pub fn open(&mut self) {
        let delay = Delay::new_default();

        for _ in 0..2 {
            delay.delay_ms(20);
            self.write_cmd_data_u8(0x36, Some(&[0x48]));
            self.write_cmd_data_u8(0x3a, Some(&[0x55]));
            self.write_cmd_data_u8(0x11, None);
            self.write_cmd_data_u8(0x29, None);
        }
    }

    pub fn write_pixel_slint(
        &mut self,
        x: u16,
        y: u16,
        x_end: u16,
        y_end: u16,
        pixel: &[Rgb565Pixel],
    ) {
        self.write_draw_range(x, y, x_end, y_end);
        self.write_cmd_data_slint(0x2c, pixel);
    }

    fn write_cmd_data_u8(&mut self, cmd: u8, data: Option<&[u8]>) {
        self.dc.set_low().unwrap();
        self.spi.write(&[cmd]).unwrap();

        if let Some(v) = data {
            self.dc.set_high().unwrap();
            self.spi.write(v).unwrap();
        }
    }

    fn write_draw_range(&mut self, x: u16, y: u16, x_end: u16, y_end: u16) {
        let mut x_buf = [0u8; 4];
        let mut y_buf = [0u8; 4];

        x_buf[0..=1].copy_from_slice(&x.to_be_bytes());
        x_buf[2..=3].copy_from_slice(&x_end.to_be_bytes());

        y_buf[0..=1].copy_from_slice(&y.to_be_bytes());
        y_buf[2..=3].copy_from_slice(&y_end.to_be_bytes());

        self.write_cmd_data_u8(0x2a, Some(&x_buf));
        self.write_cmd_data_u8(0x2b, Some(&y_buf));
    }

    fn write_cmd_data_slint(&mut self, cmd: u8, data: &[Rgb565Pixel]) {
        let mut i = 0;
        data.iter().for_each(|v| {
            self.line_buf[i..=i + 1].copy_from_slice(v.0.to_be_bytes().as_ref());
            i += 2;
        });

        self.dc.set_low().unwrap();
        self.spi.write(&[cmd]).unwrap();

        self.dc.set_high().unwrap();
        self.spi.write(self.line_buf[0..i].as_ref()).unwrap();
    }
}

xpt2046

use std::borrow::*;

use esp_idf_svc::hal::{gpio::*, peripheral::*, prelude::*, spi::*};

/*
d0 读 x 轴
90 读 y 轴
*/

pub struct XPT2046Touch {
    pub x: f32,
    pub y: f32,
}

pub struct XPT2046<'a, SPI>
where
    SPI: Borrow<SpiDriver<'a>> + 'a,
{
    spi: SpiDeviceDriver<'a, SPI>,
}

impl<'a, SPI> XPT2046<'a, SPI>
where
    SPI: Borrow<SpiDriver<'a>> + 'a,
{
    pub fn new<CS>(spi: SPI, cs: CS) -> Self
    where
        CS: Peripheral<P = CS> + OutputPin,
    {
        let config = config::Config::default()
            .baudrate(2.MHz().into())
            .duplex(config::Duplex::Half3Wire);

        Self {
            spi: SpiDeviceDriver::new(spi, Some(cs), &config).unwrap(),
        }
    }

    pub fn read(&mut self) -> Option<XPT2046Touch> {
        let mut x_u16 = [0u16; 3];
        let mut y_u16 = [0u16; 3];

        for i in 0..x_u16.len() {
            let mut x = [0u8; 2];
            let mut y = [0u8; 2];

            self.spi
                .transaction(&mut [Operation::Write(&[0xd0]), Operation::Read(&mut x)])
                .unwrap();

            self.spi
                .transaction(&mut [Operation::Write(&[0x90]), Operation::Read(&mut y)])
                .unwrap();

            x_u16[i] = u16::from_be_bytes(x) << 1 >> 4;
            y_u16[i] = u16::from_be_bytes(y) << 1 >> 4;
        }

        x_u16.sort();
        y_u16.sort();

        // 实测最大最小值
        let x = x_u16[1].max(336).min(3847);
        let y = y_u16[1].max(184).min(3584);

        let x = (x - 336) as f32 / (3847 - 336) as f32;
        let y = (y - 184) as f32 / (3584 - 184) as f32;

        // 判断有没有触摸
        if x == 0 as f32 && y == 1 as f32 {
            None
        } else {
            Some(XPT2046Touch { x, y })
        }
    }
}

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

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

相关文章

HTTP交互导致ECONNABORTED的原因之一

背景&#xff1a; 本次记录的&#xff0c;是一次使用HTTP交互过程中遇到的问题&#xff0c;问题不大&#xff0c;就是给题目上这个报错补充一种可能的解决方案。 程序大致流程&#xff1a; 1. 设备向服务器A请求信息 2. 拿到回复记录下回复内容中的数据包下载地址等信息 3…

sql聚合函数使用-笔记

sql聚合函数使用-笔记 SELECT SUM ( case when procurement_type 公益推送 then 1 else 0 end ) gywxTotal,SUM ( CASE WHEN (status 1 and procurement_type 公益推送) THEN 1 ELSE 0 END ) gywxYsc,SUM ( CASE WHEN (status ! 1 and procurement_type 公益推送) THEN 1 …

k8s部署presto

&#xff08;作者&#xff1a;陈玓玏&#xff09; 一、前提条件 已部署k8s&#xff1b;已部署hadoop和hive&#xff0c;可参考以下链接&#xff1a; https://blog.csdn.net/weixin_39750084/article/details/136750613?spm1001.2014.3001.5502 https://blog.csdn.net/wei…

Visual Studio 的调试(一)

最近事儿很多昂&#xff0c;更新速度相较以往慢了许多&#xff0c;备考六月份的四级&#xff0c;还有学校的期末等等&#xff0c;事儿真的太多啦&#xff0c;所以后面的更新速度也会放慢一点&#xff0c;实在是抽不开身啊诸位&#xff0c;相当抱歉&#xff0c;还望诸君见谅 言…

Dockerfile文件详细介绍

前言 Dockerfile是一个文本文件&#xff0c;包含了用于构建Docker镜像的所有命令和说明。它定义了容器的运行环境、依赖以及启动方式&#xff0c;是创建Docker镜像的核心部分。 由于制作镜像的过程中&#xff0c;需要逐层处理和打包&#xff0c;比较复杂&#xff0c;所以Docke…

Midjourney Describe API 使用文档

Midjourney Describe API 使用文档 Midjourney Describe API 的主要功能是通过上传图片&#xff0c;获取对图片的描述。使用该 API&#xff0c;只需要传递图片文件&#xff0c;API 会返回图片的详细描述。无需繁琐的参数设置&#xff0c;即可获得高质量的图片描述。 支持多种图…

腾讯Java社招面试题真题,最新面试题

Java中synchronized和ReentrantLock有什么区别&#xff1f; 1、锁的实现方式不同&#xff1a; synchronized是JVM层面的锁&#xff0c;主要依赖于监视器对象&#xff08;monitor&#xff09;实现。ReentrantLock是JDK层面的锁&#xff0c;通过Java代码实现&#xff0c;提供了更…

陪跑真正值钱的不是教程,是你遇到那个挡住你的问题时,身边有个靠谱的人

今天分享两个概念&#xff0c;一个是意识决定一切&#xff0c;一个是大道至简&#xff0c;做项目就是按部就班的遵循事情发展规律去做。 先说第一个概念&#xff0c;意识决定一切。我们说的凡事预则立不预则废&#xff0c;就是计划了去做就会有结果。 给你们一个表&#xff0c;…

力扣:1738. 找出第 K 大的异或坐标值

1738. 找出第 K 大的异或坐标值 给你一个二维矩阵 matrix 和一个整数 k &#xff0c;矩阵大小为 m x n 由非负整数组成。 矩阵中坐标 (a, b) 的 值 可由对所有满足 0 < i < a < m 且 0 < j < b < n 的元素 matrix[i][j]&#xff08;下标从 0 开始计数&…

简单的UDP网络程序:多人群聊系统

本章重点 能够实现一个简单的udp客户端/服务器; 1.创建套接字 我们把服务器封装成一个类&#xff0c;当我们定义出一个服务器对象后需要马上初始化服务器&#xff0c;而初始化服务器需要做的第一件事就是创建套接字。 ⭐参数说明&#xff1a; domain&#xff1a;创建套接字的域…

Spring中的Aware接口

Spring中的Aware接口 Aware接口介绍 Aware是Spring中的接口&#xff0c;它的作用是可以让Bean获取到运行环境的相关信息。比如获取到上下文、Bean在容器中的名称等。 Spring中提供了很多Aware接口的子类&#xff0c;具体如下&#xff1a; 常用接口的作用如下&#xff1a; …

设计软件有哪些?建模和造型工具篇(1),渲染100邀请码1a12

之前我们介绍的都是渲染软件&#xff0c;但对于设计师来说建模和造型是在渲染之前&#xff0c;所以从现在开始&#xff0c;我们会介绍一批建模和造型工具。 1、ZBrush ZBrush是由Pixologic公司开发的数字雕刻和绘画软件&#xff0c;专为艺术家和设计师而设计。它结合了3D建模…

Python打包篇-exe

文章目录 pyinstallerauto-py-to-exe pyinstaller 命令行工具&#xff0c;语法自行查看官方help pip install pyinstallerauto-py-to-exe 基于pyinstaller的一款GUI工具&#xff0c;会自行打包py文件中依赖的库 pip install auto-py-to-exe auto-py-to-exe.exe //运行即可

【教程】Linux部署Android安卓模拟器

转载请注明出处&#xff1a;小锋学长生活大爆炸[xfxuezhagn.cn] 如果本文帮助到了你&#xff0c;欢迎[点赞、收藏、关注]哦~ 未完成&#xff0c; 先简单记录下指令。 docker-android https://github.com/budtmo/docker-android 检查系统是否支持&#xff1a; sudo apt instal…

Linux网络编程:HTTPS协议

目录 1.预备知识 1.1.加密和解密 1.2.常见加密方式 1.2.1.对称加密 1.2.2.非对称加密 ​编辑 1.3.数据摘要&#xff08;数据指纹&#xff09;和数据签名 1.4.证书 1.4.1.CA认证 1.4.2.证书和数字签名 2.HTTPS协议 2.1.自行设计HTTPS加密方案 2.1.1.只使用对称加密 …

【微机原理及接口技术】可编程并行接口芯片8255A

【微机原理及接口技术】可编程并行接口芯片8255A 文章目录 【微机原理及接口技术】可编程并行接口芯片8255A前言一、8255A的内部结构和引脚1.与外设接口&#xff08;数据端口&#xff09;2.与处理器接口 二、8255A的工作方式三、8255A的编程1. 写入方式控制字&#xff1a;控制字…

[GUET-CTF2019]encrypt

我自己大致分析的是输入flag然后先 RC4加密再 base64加密&#xff0c;解了一下发现不对。 那就只能仔细分析了&#xff0c;看有没有魔改 嗯&#xff0c;可以动调试试&#xff0c;嗯就是要找其 key 的值 &#xff1a;0x10,0x20,0x30,0x30,0x20,0x10,0x40 aaaaaaaaaa--->dd…

HTML.

HTML:超文本标记语言&#xff08;Hyper Text Markup Language&#xff09; 超文本&#xff1a;不同于普通文本&#xff0c;可以定义图片&#xff0c;音频&#xff0c;视频等内容 标记语言&#xff1a;由标签构成的语言 HTML标签都是预定义好的HTML代码直接在浏览器中运行&#…

Llama 3没能逼出GPT-5!OpenAI怒“卷”To B战场,新企业级 AI 功能重磅推出!

Meta 是本周当之无愧的AI巨星&#xff01;刚刚推出的 Llama 3 凭借着强大的性能和开源生态的优势在 LLM 排行榜上迅速跃升。 按理说&#xff0c;Llama 3在开源的状态下做到了 GPT-3.7 的水平&#xff0c;必然会显得用户&#xff08;尤其是企业用户&#xff0c;他们更具备独立部…

ISCC2024个人挑战赛WP-WEB

&#xff08;非官方解&#xff0c;以下内容均互联网收集的信息和个人思路&#xff0c;仅供学习参考&#xff09; 还没想好名字的塔防游戏 GET /world.js HTTP/1.1 Host: 101.200.138.180:17345 Accept: text/html,application/xhtmlxml,application/xml;q0.9,image/avif,i…