Andriod入门级开发

news2024/9/25 7:15:17

这学期有个课设,我们组我负责一个手机APP的开发,虽然刚开始说要实现什么智能导航,类似高德地图那种,但最后阉割的只剩一个Socket通信了,因为之前没有接触过(可能之后也不会再接触),记录一下开发中遇到的问题。

1.开发工具:Andriod Studio,主要功能:实现与开发板的Socket通信,具体是手机端收到板子消息按钮变色或者文本框显示对应文字,手机端也会发送消息给板子,让板子跳转用户操作界面,整个就是一个简单的前端,逻辑不需要手机端处理。手机端只负责显示。
2.实现效果:
在这里插入图片描述
在这里插入图片描述

3.一些关键问题和收获:
1)andriod studio 的使用:
感觉还是比较复杂的,前期的一些环境配置有点忘了,按照提示做就可以,主要是手机运行环境的一些配置,然后之后在打包程序安装在手机上出了点问题,需要密钥,按网上指导就可以。
对自带模板的灵活使用:as自带很多模板,比如我开发的这个丑陋的app就是在bottom Navigation Activity这个现成模板的基础上进行开发的。
在这里插入图片描述
之前as排版的Xml文件,下载手机上就会变乱。后面不知不觉就好了,注意这个问题。
2)Socket通信:
这个是这个项目的一个主要工作,其中遇到的第一个问题,怎么创建Socket通信:

class ConnectThread extends Thread {
        public void run() {
            try {
                //新建Socket
                socket = new Socket("192.168.4.1", 8086);
                System.out.println("创建Socket成功");
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

Socket的收发信息:注意Socket的收发信息或者说对网络的操作不能直接在主线程进行,一般都需要开子线程。
Socket的收信息:

final byte[] buffer = new byte[1024];//创建接收缓冲区
                    try {
                        inputStream = socket.getInputStream();
                        System.out.println("Socket接收数据成功");
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                    int len = 0;//数据读出来,并且返回数据的长
                    try {
                        len = inputStream.read(buffer);
                        System.out.println("接收数据大小" + len);
                    } catch (IOException e) {
                        e.printStackTrace();
                    }

                    //BUFFER 内容转为字符串
                    String recvStr = new String(buffer, 0, len);

Socket的发信息:

 //控制子线程开始
    public void buttonFunc() {
        // 在主线程中需要发送数据的代码片断前插入
        // 阻塞主线程,使子线程按照“同步”的方式执行
        try {
            System.out.println("线程开始");
            new Thread(sendthread).start();    // 线程启动
            new Thread(sendthread).join();     // 线程加入执行队列,主线程被阻塞,等待子线程执行完毕
        } catch (Exception e) {
            e.printStackTrace();
            System.out.println(e.toString());
        }
    }
    // 发送数据子线程
    final Runnable sendthread = new Runnable() {
        @Override
        public void run() {
            try {
                System.out.println("发送数据的子线程执行中");
                try {
                    try {
                        //输出流初始化
                        outputStream = socket.getOutputStream();
                        System.out.println("outputStream创建成功");
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                    try {
                        //写入数据
                        outputStream.write(sendbuf);
                        outputStream.flush();
                        System.out.println("数据写入成功");
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }
                System.out.println("数据发送成功,子线程执行完毕");
            } catch (Exception e) {
                e.printStackTrace();
                System.out.println("子线程:"+ e.toString());
            }
        }
    };

socket的数据解析及数据格式:
收信息需要先转换为string类型,发信息如果要发16进制,byte数组直接写成16进制的形式。

源码:
美食城:

package com.example.parkingassitant2.ui.home;

import android.annotation.SuppressLint;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Button;
import android.widget.Toast;

import androidx.annotation.NonNull;
import androidx.fragment.app.Fragment;
import androidx.lifecycle.ViewModelProvider;

import com.example.parkingassitant2.databinding.FragmentHomeBinding;

import java.io.IOException;
import java.io.OutputStream;
import java.net.Socket;

public class HomeFragment extends Fragment {
    private FragmentHomeBinding binding;
    //发送数据保存
    byte[] sendbuf = {(byte) 0xaa, (byte) 0xaa,0x03, (byte) 0x12,0x03,0x18, (byte) 0xaa};
    Socket socket = null;        //定义socket
    OutputStream  outputStream = null;    //定义输出流(发送)
    public View onCreateView(@NonNull LayoutInflater inflater,
                             ViewGroup container, Bundle savedInstanceState) {
        HomeViewModel homeViewModel =
                new ViewModelProvider(this).get(HomeViewModel.class);
        binding = FragmentHomeBinding.inflate(inflater, container, false);
        View root = binding.getRoot();
        ConnectThread ct;
        ct = new ConnectThread();
        ct.start();
        //江底捞
        final Button jiangdilao = binding.xiaocaiyuan;
        //湖底捞按钮
        final Button hudilao = binding.A2;
        //河底捞按钮
        final Button hedilao = binding.B1;
        //海底捞按钮
        final Button haidilao = binding.B2;
        //KFC按钮
        final Button KFC = binding.C1;
        //小菜园按钮
        final Button xiaocaiyuan = binding.C2;
        //万达影城按钮
        final Button wanda = binding.D1;
        //星巴克按钮
        final Button starbuck = binding.D2;
        //江底捞按钮监听函数
        jiangdilao.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //不同按钮对应不同的数据流
                sendbuf = new byte[]{(byte) 0xaa, (byte) 0xaa, 0x03, (byte) 0x12, 0x03, 0x18, (byte) 0xaa};
                buttonFunc();
                Toast.makeText(getContext(), "预约成功,请更新停车场信息!", Toast.LENGTH_SHORT).show();
            }
        });
        //湖底捞按钮监听函数
        hudilao.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //不同按钮对应不同的数据流
                sendbuf = new byte[]{(byte) 0xaa, (byte) 0xaa, 0x03, (byte) 0x21, 0x03, 0x27, (byte) 0xaa};
                buttonFunc();
                Toast.makeText(getContext(), "预约成功,请更新停车场信息!", Toast.LENGTH_SHORT).show();
            }
        });
        //河底捞按钮监听函数
        hedilao.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //不同按钮对应不同的数据流
                sendbuf = new byte[]{(byte) 0xaa, (byte) 0xaa, 0x03, (byte) 0x22, 0x03, 0x28, (byte) 0xaa};
                buttonFunc();
                Toast.makeText(getContext(), "预约成功,请更新停车场信息!", Toast.LENGTH_SHORT).show();
            }
        });
        //海底捞按钮监听函数
        haidilao.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //不同按钮对应不同的数据流
                sendbuf = new byte[]{(byte) 0xaa, (byte) 0xaa, 0x03, (byte) 0x11, 0x03, 0x17, (byte) 0xaa};
                buttonFunc();
                Toast.makeText(getContext(), "预约成功,请更新停车场信息!", Toast.LENGTH_SHORT).show();
            }
        });
        //小菜园按钮监听函数
        xiaocaiyuan.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //不同按钮对应不同的数据流
                sendbuf = new byte[]{(byte) 0xaa, (byte) 0xaa, 0x03, (byte) 0x13, 0x03, 0x19, (byte) 0xaa};
                buttonFunc();
                Toast.makeText(getContext(), "预约成功,请更新停车场信息!", Toast.LENGTH_SHORT).show();
            }
        });
        //星巴克按钮监听函数
        starbuck.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //不同按钮对应不同的数据流
                sendbuf = new byte[]{(byte) 0xaa, (byte) 0xaa, 0x03, (byte) 0x14, 0x03, 0x1a, (byte) 0xaa};
                buttonFunc();
                Toast.makeText(getContext(), "预约成功,请更新停车场信息!", Toast.LENGTH_SHORT).show();
            }
        });
        //肯德基按钮监听函数
        KFC.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //不同按钮对应不同的数据流
                sendbuf = new byte[]{(byte) 0xaa, (byte) 0xaa, 0x03, (byte) 0x23, 0x03, 0x29, (byte) 0xaa};
                buttonFunc();
                Toast.makeText(getContext(), "预约成功,请更新停车场信息!", Toast.LENGTH_SHORT).show();
            }
        });
        //电影院按钮监听函数
        wanda.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //不同按钮对应不同的数据流
                sendbuf = new byte[]{(byte) 0xaa, (byte) 0xaa, 0x03, (byte) 0x24, 0x03, 0x2a, (byte) 0xaa};
                buttonFunc();
                Toast.makeText(getContext(), "预约成功,请更新停车场信息!", Toast.LENGTH_SHORT).show();
            }
        });
        return root;
    }
    class ConnectThread extends Thread {
        public void run() {
            try {
                //两个不同IP地址。网络助手和板子
                //socket = new Socket("192.168.71.1", 8086);
                //新建Socket
                socket = new Socket("192.168.4.1", 8086);
                System.out.println("创建Socket成功");
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
    @SuppressLint("SuspiciousIndentation")
    @Override
    public void onDestroyView() {
        super.onDestroyView();
        binding = null;
        try {
            if(socket != null) {
                socket.close();
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
    //控制子线程开始
    public void buttonFunc() {
        // 在主线程中需要发送数据的代码片断前插入
        // 阻塞主线程,使子线程按照“同步”的方式执行
        try {
            System.out.println("线程开始");
            new Thread(sendthread).start();    // 线程启动
            new Thread(sendthread).join();     // 线程加入执行队列,主线程被阻塞,等待子线程执行完毕
        } catch (Exception e) {
            e.printStackTrace();
            System.out.println(e.toString());
        }
    }
    // 发送数据子线程
    final Runnable sendthread = new Runnable() {
        @Override
        public void run() {
            try {
                System.out.println("发送数据的子线程执行中");
                try {
                    try {
                        //输出流初始化
                        outputStream = socket.getOutputStream();
                        System.out.println("outputStream创建成功");
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                    try {
                        //写入数据
                        outputStream.write(sendbuf);
                        outputStream.flush();
                        System.out.println("数据写入成功");
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }
                System.out.println("数据发送成功,子线程执行完毕");
            } catch (Exception e) {
                e.printStackTrace();
                System.out.println("子线程:"+ e.toString());
            }
        }
    };
}

停车场:

package com.example.parkingassitant2.ui.dashboard;

import android.annotation.SuppressLint;
import android.graphics.Color;
import android.os.Bundle;
import android.os.Looper;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;
import android.widget.Toast;

import androidx.annotation.NonNull;
import androidx.fragment.app.Fragment;
import androidx.lifecycle.ViewModelProvider;

import com.example.parkingassitant2.databinding.FragmentDashboardBinding;

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;

public class DashboardFragment extends Fragment {

    private FragmentDashboardBinding binding;
    Socket socket = null;		//定义socket
    OutputStream outputStream = null;    //定义输入流(接收)
    public View onCreateView(@NonNull LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        DashboardViewModel dashboardViewModel = new ViewModelProvider(this).get(DashboardViewModel.class);
        binding = FragmentDashboardBinding.inflate(inflater, container, false);
        View root = binding.getRoot();
        ConnectThread ct;
        ct = new ConnectThread();
        ct.start();
        return root;
    }
    class ConnectThread extends Thread{
        InputStream inputStream=null;	//定义输入流(接收)
        final Button A1 =binding.A1;
        final Button A2 = binding.A2;
        final Button D1 = binding.D1;
        public void run(){
            System.out.println(Thread.currentThread().getName()+": Hello");
            try {
                //socket = new Socket("192.168.71.1", 8086);
                socket = new Socket("192.168.4.1", 8086);
                binding.textView5.setText("连接成功");
                System.out.println("创建Socket成功");
            } catch (IOException e) {
                System.out.println("创建Socket失败");
                e.printStackTrace();
            }
            try{
                while (true)
                {
                    final byte[] buffer = new byte[1024];//创建接收缓冲区
                    try {
                        inputStream = socket.getInputStream();
                        System.out.println("Socket接收数据成功");
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                    int len = 0;//数据读出来,并且返回数据的长
                    try {
                        len = inputStream.read(buffer);
                        System.out.println("接收数据大小" + len);
                    } catch (IOException e) {
                        e.printStackTrace();
                    }

                    //BUFFER 内容转为字符串
                    String recvStr = new String(buffer, 0, len);
                    //找到读取开始处,一般为0
                    int stadex = recvStr.indexOf("aaaa");
                    System.out.println("解析数据开始位置" + stadex);
                    int check_3,check_4,check_5,check_6;
                    while(stadex < len){
                        stadex = recvStr.indexOf("aaaa",stadex);
                        int funcType_sta = stadex+4;
                        int funcType_end = stadex+6;
                        if(funcType_end > len || funcType_sta >= len){
                            break;
                        }
                        //位置3是操作类型 02 是显示车牌预约停车场
                        String pos3 = recvStr.substring(funcType_sta,funcType_end);
                        // check_3 = Integer.parseInt(pos3);
                        stadex = funcType_end;
                        //停车预约功能中,位置4是车牌号
                        String pos4 = recvStr.substring(stadex,stadex+2);
                        //check_4 = Integer.parseInt(pos4);
                        stadex += 2;
                        //停车预约功能中,位置5是目的地
                        String pos5 = recvStr.substring(stadex,stadex+2);
                        //check_5 = Integer.parseInt(pos5);
                        stadex += 2;

                        String pos6 = recvStr.substring(stadex,stadex+2);
                        //check_6 = Integer.parseInt(pos6);
                        stadex += 2;

                        //if( check_6 != check_3+check_4+check_5) break;
                        stadex += 2;
                        System.out.println("解析数据包" + "aaaa" + pos3 + pos4 + pos5 + pos6 + "aa");
                        System.out.println("操作类型指示值:" + pos3);
                        switch(pos3){
                            case "01": {
                                System.out.println("操作类型为欢迎车牌进站");
                                String showText = null;
                                switch (pos4) {
                                    case "01":
                                        showText = "欢迎京A00001";
                                        break;
                                    case "02":
                                        showText = "欢迎青ASB520";
                                        break;
                                    case "03":
                                        showText = "欢迎皖A66666";
                                        break;
                                    case "04":
                                        showText = "欢迎宁E88888";
                                        break;
                                    default:
                                        break;
                                }
                                binding.textView5.setText(showText);
                                break;
                            }
                            case "04": {
                                System.out.println("操作类型为停车预约");
                                String showText = null;
                                switch (pos4) {
                                    case "01":
                                        showText = "京A00001";
                                        break;
                                    case "02":
                                        showText = "青ASB520";
                                        break;
                                    case "03":
                                        showText = "皖A66666";
                                        break;
                                    case "04":
                                        showText = "宁E88888";
                                        break;
                                    default:
                                        break;
                                }
                                String address = null;
                                switch (pos5){
                                    case "11":
                                        address = "海底捞";
                                        break;
                                    case "12":
                                        address = "江底捞";
                                        break;
                                    case "13":
                                        address = "小菜园";
                                        break;
                                    case "14":
                                        address = "星巴克";
                                        break;
                                    case "21":
                                        address = "湖底捞";
                                        break;
                                    case "22":
                                        address = "河底捞";
                                        break;
                                    case "23":
                                        address = "肯德基";
                                        break;
                                    case "24":
                                        address = "电影院";
                                        break;
                                    case "a1":
                                        address = "A1";
                                        break;
                                    case "a2":
                                        address = "A2";
                                        break;
                                    case "d1":
                                        address = "D1";
                                        break;
                                    default:
                                        break;
                                }
                                binding.textView5.setText(showText+"车主预约车位:" + address);
                                break;
                            }
                            case "03": {
                                System.out.println("操作类型为车位状态变更");
                                switch (pos4) {
                                    case "a1": {
                                        if (pos5.equals("00")) {
                                            binding.A1.setBackgroundColor(Color.parseColor("#00FF00"));
                                        } else if (pos5.equals("01")) {
                                            binding.A1.setBackgroundColor(Color.parseColor("#FF0000"));
                                        } else if (pos5.equals("02")) {
                                            binding.A1.setBackgroundColor(Color.parseColor("#FFFF00"));
                                        }
                                        break;
                                    }
                                    case "a2": {
                                        System.out.println("pos5: ");
                                        System.out.println(pos5);
                                        if (pos5.equals("00")) {
                                            binding.A2.setBackgroundColor(Color.parseColor("#00FF00"));
                                        } else if (pos5.equals("01")) {
                                            binding.A2.setBackgroundColor(Color.parseColor("#FF0000"));
                                        } else if (pos5.equals("02")) {
                                            binding.A2.setBackgroundColor(Color.parseColor("#FFFF00"));
                                        }
                                        break;
                                    }
                                    case "d1": {

                                        if (pos5.equals("00")) {
                                            binding.D1.setBackgroundColor(Color.parseColor("#00FF00"));
                                        } else if (pos5.equals("01")) {
                                            binding.D1.setBackgroundColor(Color.parseColor("#FF0000"));
                                        } else if (pos5.equals("02")) {
                                            binding.D1.setBackgroundColor(Color.parseColor("#FFFF00"));
                                        }
                                        break;
                                    }
                                    default:
                                        break;
                                }
                            }
                            default:
                                break;
                        }
                    }
                }
            } catch (Exception ex) {
                ex.printStackTrace();
            }
        }

    }
    @SuppressLint("SuspiciousIndentation")
    @Override
    public void onDestroyView() {
        super.onDestroyView();
        try {
            if(socket != null) {
                socket.close();
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
        binding = null;
    }
}


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

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

相关文章

【数据管理】谈谈哈希原理和散列表

一、说明 提起哈希&#xff0c;有人要说&#xff1a;不就是一个稀疏表格么&#xff0c;谈的上什么原理&#xff1f;我说&#xff1a;非也&#xff0c;哈希是是那种看似无物&#xff0c;其实解决大问题的东西。如何提高数据管理效率&#xff1f;这是个问题&#xff0c;随着这个问…

测试2:编写测试用例的方法

2.编写测试用例的方法 7种 测试常用的方法&#xff1a;code review 代码静态分析、CI/CD CI–持续集成–开发成员经常集成它们的工作&#xff0c;尽快发现集成错误 CD–持续部署–将集成后的代码部署到更贴近真实运行的环境 2.1 测试用例的描述&#xff1a; 用例编号 用例…

Python纯Numpy手撕SGD

文章目录简介问题建模数据加载和预处理数据加载预处理分batch损失函数训练运行简介 本博客用多元线性回归展示如何从零实现一个随机梯度下降SGD, 不使用torch等AI框架 问题建模 给定一个数据集X∈RN(D1)\large X \in \R^{N \times (D1)}X∈RN(D1)和对应标签向量Y∈RN\large …

centos7防火墙工具firewall-cmd使用

centos7防火墙工具firewall-cmd使用防火墙概述centos7防火墙工具firewall-cmd使用介绍firewalld的基本使用服务管理工具相关指令配置firewalld-cmd防火墙概述 防火墙是可以帮助计算机在内部网络和外部网络之间构建一道相对隔绝的保护屏障&#xff0c;从而保护数据信息的一种技…

Vulnhub 渗透练习(七)—— FRISTILEAKS: 1.3

环境搭建 下载链接 virtualbox 打开靶机设置为 host-only&#xff0c;攻击机同样。 具体可点此处 信息收集 开了个 80 端口。 用的是 apache 2.2.15 &#xff0c;这个版本有个解析漏洞。 目录 根据首页的图片猜测 /fristi/ 目录&#xff08;不过我没想到 -_-&#x…

由浅入深掌握各种 Python multiprocessing 进程间通信方式

由浅入深掌握各种 Python 多进程间通信方式1、为什么要掌握进程间通信2、进程间各类通信方式简介3、消息机制通信1) 管道 Pipe 通信方式2) 消息队列Queue 通信方式4、同步机制通信(1) 进程间同步锁 – Lock(2) 子进程间协调机制 -- Event5、共享内存方式通信(1) 共享变量(2) 共…

【Python】控制自己的手机摄像头拍照,并自动发送到邮箱

前言 嗨喽&#xff0c;大家好呀~这里是爱看美女的茜茜呐 今天这个案例&#xff0c;就是控制自己的摄像头拍照&#xff0c; 并且把拍下来的照片&#xff0c;通过邮件发到自己的邮箱里。 想完成今天的这个案例&#xff0c;只要记住一个重点&#xff1a;你需要一个摄像头 思路…

Android 7.0 OTA升级(高通)

文章目录1. Full OTA 方式升级介绍1.1 Full OTA 制作第一步&#xff1a;生成 msm89xx-target_files-eng.XXX.zip1.2 Full OTA 制作第二步&#xff1a;Modem 等非 HLOS 加入升级包的方法1.3 Full OTA 制作第三步&#xff1a;生成 update.zip 升级包2. Incremental OTA 方式升级介…

Android 基础知识4-2.6LinearLayout(线性布局)

一、LinearLayout的概述 线性布局&#xff08;LinearLayout&#xff09;主要以水平或垂直方式来排列界面中的控件。并将控件排列到一条直线上。在线性布局中&#xff0c;如果水平排列&#xff0c;垂直方向上只能放一个控件&#xff0c;如果垂直排列&#xff0c;水平方向上也只能…

Java基础-xml

1.xml 1.1概述 万维网联盟(W3C) 万维网联盟(W3C)创建于1994年&#xff0c;又称W3C理事会。1994年10月在麻省理工学院计算机科学实验室成立。 建立者&#xff1a; Tim Berners-Lee (蒂姆伯纳斯李)。 是Web技术领域最具权威和影响力的国际中立性技术标准机构。 到目前为止&#…

python基础语法【自用】

✨始发站&#x1f6a9;Python的基础语法&#xff0c;冲冲冲&#xff01; &#x1f6a9;注&#xff1a;本篇为python基础语法篇&#xff0c;因博主之前使用java&#xff0c;所以本基础语法篇实为自用丐版&#xff01; &#x1f332; 你好&#xff0c;世界&#xff01; 安装环境…

虚拟机快照

1. 快照有什么作用&#xff1f; 通俗理解&#xff1a;快照就是备份。 2. VMware Workstation 和 VMware Fusion 都支持制作快照去使用 一、快照 保存当前虚拟机状态。可以恢复 二、 在VMware Workstation Pro中制作并还原快照 三、在VMware Fusion Pro中制作并还原快照 快照制…

210天从外包踏进华为跳动那一刻,我泪目了

前言 没有绝对的天才&#xff0c;只有持续不断的付出。对于我们每一个平凡人来说&#xff0c;改变命运只能依靠努力幸运&#xff0c;但如果你不够幸运&#xff0c;那就只能拉高努力的占比。 2021年4月&#xff0c;我有幸成为了华为的一名高级测试工程师&#xff0c;正如标题所…

【软件测试】python接口自动化测试编写脚本,资深测试总结方法,你的实用宝典......

目录&#xff1a;导读前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09;前言 接口测试&#xff0…

美团前端一面手写面试题

实现斐波那契数列 // 递归 function fn (n){if(n0) return 0if(n1) return 1return fn(n-2)fn(n-1) } // 优化 function fibonacci2(n) {const arr [1, 1, 2];const arrLen arr.length;if (n < arrLen) {return arr[n];}for (let i arrLen; i < n; i) {arr.push(arr[…

vulnhub Kioptrix4

总结&#xff1a;sql注入&#xff0c;受限shell绕过&#xff0c;mysql提权 目录 下载地址 漏洞分析 信息收集 sql注入 ssh登录绕过受限shell 提权 下载地址 Kioptrix4_Hyper_v.rar (Size: 210 MB)Download: http://www.kioptrix.com/dlvm/Kioptrix4_Hyper_v.rarDownload …

Linux驱动开发详细解析

Linux驱动开发详细解析 驱动概念 驱动与底层硬件直接打交道&#xff0c;充当了硬件与应用软件中间的桥梁。 具体任务 读写设备寄存器&#xff08;实现控制的方式&#xff09;完成设备的轮询、中断处理、DMA通信&#xff08;CPU与外设通信的方式&#xff09;进行物理内存向虚…

关于字符设备驱动的通用概念和写法

概述 设备驱动程序可以使用模块的方式动态加载到内核中去。加载模块的方式与以往的应用程序开发有很大的不同。以往在开发应用程序时都有一个 main()函数作为程序的入口点&#xff0c;而在驱动开发时却没有 main()函数&#xff0c;模块在调用 insmod 命令时被加载&#xff0c;…

JVM学习笔记一:类加载子系统

目录 前言 类加载子系统的作用 类加载器角色的位置 类加载器分类 虚拟机自带的加载器 启动类加载器&#xff08;引导类加载器&#xff09; 扩展类加载器 系统类加载器 用户自定义类加载器 什么时候需要自定义类加载器&#xff1f; 如何自定义类加载器&#xff1f; …

【验证码的识别】—— 极验验证码的识别

前言 &#xff08;结尾有彩蛋欧&#xff09; 目前&#xff0c;许多网站采取各种各样的措施来反爬虫&#xff0c;其中一个措施便是使用验证码。随着技术的发展&#xff0c;验证码的花样越来越多。验证码最初是几个数字组合的简单的图形验证码&#xff0c;后来加入了英文字母和混…