drf知识-09

自定义频率类

# throttling 频率限制
# 简单方案
from rest_framework.throttling import SimpleRateThrottle
class CommonThrottle(SimpleRateThrottle):rate = '3/m'def get_cache_key(self, request, view):ip = request.META.get('REMOTE_ADDR')return ip# 复杂方案---》通用方案---》后期任何框架都可以使用这个逻辑
from rest_framework.throttling import BaseThrottle
import time
class MyThrottling(BaseThrottle):VISIT_RECORD = {}  # {192.168.1.11:[时间2,时间1],192.168.1.99:[时间3,时间2,时间1]}def __init__(self):self.history = Nonedef allow_request(self, request, view):ip = request.META.get('REMOTE_ADDR')  # 取出访问者ipctime = time.time()if ip not in self.VISIT_RECORD:self.VISIT_RECORD[ip] = [ctime, ]return Trueself.history = self.VISIT_RECORD.get(ip)  # 访问时间列表while self.history and ctime - self.history[-1] > 3600:self.history.pop()if len(self.history) < 5:self.history.insert(0, ctime)return Trueelse:return Falsedef wait(self):ctime = time.time()return 60 - (ctime - self.history[-1])
# exceptions 异常处理自己写个函数,处理drf异常和自己的异常
from rest_framework.views import exception_handler
from rest_framework.response import Response
from rest_framework.exceptions import Throttleddef common_exception_handler(exc, context):res = exception_handler(exc, context)  # 处理drf异常处理if res:if isinstance(exc, Throttled):return Response({'code': 20016, 'msg': '使用过于频繁,稍后再试:%s' % res.data.get('detail')})else:detail = res.data.get('detail') or res.data or "drf异常,请联系系统管理员"return Response({'code': 999, 'msg': detail})# return reselse:  # 如果没值,说明是自己的异常# exc 错误对象,判断具体是什么错误 :数据错误, 除以0,。。。。print(type(exc))if isinstance(exc, ZeroDivisionError):return Response({'code': 20001, 'msg': '不能除以0'})if isinstance(exc, IndexError):return Response({'code': 20002, 'msg': '超长了'})else:return Response({'code': 888, 'msg': '操作失败,请稍后再试:%s' % str(exc)})
# common 更改显示格式
from rest_framework.mixins import ListModelMixin
from rest_framework.response import Responseclass CommonListModelMixin(ListModelMixin):def list(self, request, *args, **kwargs):res = super().list(request, *args, **kwargs)return Response({'code': 100, 'msg': '查询所有成功', 'results': res.data})
# serializer.py 序列化类
from rest_framework import serializers
from .models import Book
class BookSerializer(serializers.ModelSerializer):class Meta:model = Bookfields = ['id', 'name', 'price', 'publish', 'authors', 'publish_detail', 'author_list']extra_kwargs = {'publish': {'write_only': True},'authors': {'write_only': True},'publish_detail': {'read_only': True},'author_list': {'read_only': True},}
# views.py 视图层
# 查询所有图书
from .models import Book
from .serializer import BookSerializer
from .common import CommonListModelMixin as ListModelMixin
from rest_framework.viewsets import GenericViewSetfrom .throttling import MyThrottling
class BookView(GenericViewSet, ListModelMixin):queryset = Book.objects.all()serializer_class = BookSerializer# throttle_classes = [CommonThrottling]throttle_classes = [MyThrottling]
# urls 路由
from . import views
from rest_framework.routers import SimpleRouter, DefaultRouterrouter = SimpleRouter()
router.register('books', views.BookView, 'books')
urlpatterns = [
]
urlpatterns += router.urls# 总路由
from django.contrib import admin
from django.urls import path,includeurlpatterns = [path('admin/', admin.site.urls),path('api/v1/', include('app01.urls')),
]

频率源码执行流程

# 1 APIView--->dispatch--->self.initial(request, *args, **kwargs)--->self.check_throttles(request)
# 2 APIView的check_throttlesdef check_throttles(self, request):throttle_durations = []# self.get_throttles() 是视图类中配置的一个个频率类的对象列表for throttle in self.get_throttles():# throttle.allow_request 返回false ,说明到频率了if not throttle.allow_request(request, self):# 到了频率,走了它# throttle.wait() 返回还剩多长时间能访问throttle_durations.append(throttle.wait())# 被频率限制住了,它就有值if throttle_durations:durations = [duration for duration in throttle_durationsif duration is not None]#duration=[35,54]# duration=[]duration = max(durations, default=None)# duration=56# duration=Noneself.throttled(request, duration)# 3 self.throttled(request, duration) ---》APIViewdef throttled(self, request, wait):# wait=56或Noneraise exceptions.Throttled(wait)# 4 Throttled类实例化得到对象,传了数字进去
from rest_framework.exceptions import Throttled
内部 拼接错误信息--》但是是英文的,如果要换成中文
# 5 超过了频率,就不返回False了,直接抛异常raise Throttled(None, '超过了限制,还剩:%s 时间' % self.wait())

SimpleRateThrottle 执行流程

# 1 咱们写的  CommonThrottling没有写allow_request
class CommonThrottling(SimpleRateThrottle):rate = '3/m'def get_cache_key(self, request, view):return request.META.get('REMOTE_ADDR')# 2 一定是 SimpleRateThrottle 写了,完成了频率校验
# 3 SimpleRateThrottle--》allow_request---》跟咱们的逻辑一样def allow_request(self, request, view):if self.rate is None: # 自己写了 '3/m'return True# 返回了 ip地址,self.key 就是访问者的ip地址self.key = self.get_cache_key(request, view)if self.key is None:return True# self.history 当前ip,访问的时间列表,self.key是ip# self.cache 缓存,去缓中,根据ip,取出访问者时间列表,如果没有就是 []self.history = self.cache.get(self.key, [])# 当前时间self.now = self.timer() # timer = time.time#  self.now 当前时间,#  self.duration 就是60while self.history and self.history[-1] <= self.now - self.duration:self.history.pop()# self.num_requests 是3if len(self.history) >= self.num_requests:return self.throttle_failure()return self.throttle_success()# 4  self.duration 就是60    self.num_requests 是3
# 3                   60
self.num_requests, self.duration = self.parse_rate(self.rate)
#5 self.parse_rate(self.rate)def parse_rate(self, rate):if rate is None: # '3/minute'return (None, None)# 3   period=mmmmmnum, period = rate.split('/')# 3num_requests = int(num)duration = {'s': 1, 'm': 60, 'h': 3600, 'd': 86400}[period[0]]# duration 是60,# num_requests 是3return (num_requests, duration)

drf回顾之入门规范

# 前后端开发模式:混合开发(不分离),前后端分离
# API接口
# 接口测试:postman
           测试:接口测试: jmeter是java开发的软件
# restful规范:10条规范
           drf 帮助我们快速实现符合restful规范的接口
APIView执行流程:重写了as_view,重写了dispatch
           1、去除了csrf认证
           2、包装了新的request
           3、三大认证  --根据请求方式,执行视图类的方法(认证组件、权限组件、频率组件)
           4、处理了全局异常

# Request 类的对象
        1、多了data
        2、重写了 __getattr__  对象.属性触发
                    去旧的request中找:通过反射  getattr(self._request,'path')
        3、以后用起来跟之前一样
        4、query_params                    

drf回顾之序列化组件

# 序列化
         1、写一个类,继承(序列化类)
         2、写字段,需要跟表中有对应关系(手动对应:Serializer 自动对应:ModelSerializer)
         3、视图类中使用:                        

# 多条Queryset对象:使用orm查询出来的
# django执行原生sql-->使用序列化类完成序列化
ser=BookSerializer(instnce=qs,many=True)
ser.data+Response

反序列化
            1、写一个类,继承(序列化类)
            2、写字段,需要跟表中有对应关系(手动: Serializer 自动:ModelSerializer)
            3、视图类中使用

                        修改ser=BookSerializer(instance=对象,data=request.data)
                        新增ser=BookSerializer(data=request.data)
                        ser.is_valid()
                        ser.save()--->触发序列化类的:update,create

# 序列化和反序列化用在一起:
                    read_only 和 write_only
                    extra_kwargs={ }

# 校验:三层
             Serializer-重写 update和create
             ModelSerializer-重写字段-class Meta:  model 、fields 、extra_kwargs

class BookSerializer(serializers.Serializer):   # 手动name = serializers.CharField()  # 公共的price = serializers.CharField()  # 公共的publish = serializers.IntegerField(write_only=True)  # 只用来做反序列化authors = serializers.ListField(write_only=True)  # 只用来做反序列化publish_detail = serializers.SerializerMethodField(read_only=True)  # 只用来做序列化author_list = serializers.SerializerMethodField(read_only=True)  # 只用来做序列化def get_publish_detail(self, obj):return {'name': obj.publish.name, 'city': obj.publish.city}def get_author_list(self, obj):l = []for author in obj.authors.all():l.append({'name': author.name, 'age': author.age})return ldef create(self, validated_data):# {name,price,publish:1,authors:[1,2]}authors = validated_data.pop('authors')book = Book.objects.create(name=validated_data.get('name'), price=validated_data.get('price'),publish_id=validated_data.get('publish'))book.authors.add(*authors)return bookdef update(self, instance, validated_data):# {name,price,publish:1,authors:[1,2]}authors = validated_data.pop('authors')validated_data['publish_id'] = validated_data.pop('publish')for key in validated_data:setattr(instance, key, validated_data[key])instance.save()# 先清空在放入# instance.authors.clear()# instance.authors.add(*authors)# 直接存instance.authors.set(authors)return instance
from rest_framework import serializers
from .models import Book
class BookSerializer(serializers.ModelSerializer):class Meta:model = Bookfields = ['id', 'name', 'price', 'publish', 'authors', 'publish_detail', 'author_list']extra_kwargs = {'publish': {'write_only': True},'authors': {'write_only': True},'publish_detail': {'read_only': True},'author_list': {'read_only': True},}

source定制:表中字段,表中方法,跨表

        #1 修改字段,映射字段:
                publish_name表中不存在
                publish_name = serializers.CharField(source='name')

        #2 修改字段,映射方法:
                sb_name是表模型中一个方法
                name = serializers.CharField(source='sb_name')

        #3 修改字段,跨表查询
                book表中可以链表查询
                publish=models.ForeignKey(to='Publish.name')

# 定制返回格式:序列化
        表模型中写方法
        序列化类中:SerializerMethod--》get_字段名)      

# models.py
from django.db import models
class Book(models.Model):name = models.CharField(max_length=32)price = models.DecimalField(max_digits=5, decimal_places=2)publish = models.ForeignKey(to='Publish', on_delete=models.CASCADE)authors = models.ManyToManyField(to='Author')def __str__(self):return self.name# def book_name(self):#     return self.name+'sb'# def publish_detail(self):#     return {'name': self.publish.name, 'city': self.publish.city}## def author_list(self):#     l = []#     for author in self.authors.all():#         l.append({'name': author.name, 'age': author.age})
# serializer.py
class BookSerializer(serializers.Serializer):name = serializers.CharField()price = serializers.CharField()#方案一:在表模型中写方法,在序列化类中做映射# publish_detail = serializers.DictField()  # publish_detail 会映射表模型中 publish_detail方法,方法返回值是 字典,用DictField接收# author_list = serializers.ListField()#方案二:在序列化类中写  SerializerMethodField# 只要写了这个字段类SerializerMethodField,必须配合一个方法:get_字段名,这个方法返回什么,前端这个字段就显示什么publish_detail = serializers.SerializerMethodField()def get_publish_detail(self, obj):return {'name': obj.publish.name, 'city': obj.publish.city}author_list = serializers.SerializerMethodField()def get_author_list(self, obj):l = []for author in obj.authors.all():l.append({'name': author.name, 'age': author.age})return l


# BookReadSerializer
   BookWriteSerializer
   LoginSerializer

          get_serializer_class()--影响:list  retrieve  update create    自己写的

          get_queryset()--影响:list  retrieve  update create   delete

          get_object--影响:list,retrieve,update,delete,不会影响到create      

class BookView(ModelViewSet):query_set=xxserializer_class=BookReadSerializerdef get_serializer_class(self):  # 重写 get_serializer_class 方法if self.action=='list' or self.action=='retrieve':# if self.request.method=='get':# get_serializer() 获取使用哪个序列化类return BookReadSerializerelif self.action=='login':return LoginSerializerelse:return BookWriteSerializer       def get_queryset(self):pass# 返回所有数据,但会影响 list,retrieve,update,delete ,自己写的@action(methods=['POST'],detail=False)def login(self,requset):self.get_serializer()    #让它拿到 LoginSerializerself.get_object()self.get_queryset     

drf回顾之请求和响应

# 请求
   Request源码
   能够解析的编码格式:默认三种:json,urlencoded,form-data

           1、视图类上配置

           2、settings.py  配置文件中配置

from rest_framework.parsers import JSONParser, FormParser, MultiPartParser
# JSONParser:json
# FormParser:urlencoded
# MultiPartParser:form-data
class TestView(APIView):# parser_classes = [JSONParser]parser_classes = [JSONParser,FormParser]def post(self, request):print(request.data)return Response('ok')
setting.py
# 所有drf的配置,都要写在REST_FRAMEWORK 字典中REST_FRAMEWORK = {'DEFAULT_PARSER_CLASSES': [# 'rest_framework.parsers.JSONParser',# 'rest_framework.parsers.FormParser','rest_framework.parsers.MultiPartParser',],}views.py
class TestView(APIView):parser_classes = [JSONParser,FormParser]

# 响应
   *Response源码
   *实例化时可传入的参数:

data=None----响应体                  
status=None---http响应状态码       import status 状态码          
headers=None---响应头
content_type=None---响应编码格式list方法:
res=super().list(request)   # res 就是Response 类的对象
res.status  res.data  res.headers
res.data   [{},{},{}]
{code:100,msg:成功,results:res.data}   # 响应编码格式  

drf回顾之视图组件

# APIView:
      1、继承了View
      2、重写了 as_view
      3、重写了 dipatch---相当于给之前的视图类的方法加了装饰器
      4、写跟请求方式同名的方法:get,post,delete,put,get
# GenericAPIView
   *继承了APIView        

*类属性:    
queryset =Book.object.all().filter(is_delete=False)
# self.get_queryset拿到要序列化的数据
lookup_field = 'pk'   # 分组,转换器出来的参数
filter_backends    # 过滤和排序 
pagination_class   # 分页
*方法:
get_serializer_class  # 通过重写控制视图类的方法使用哪个序列化类
serializer_class   # 获取序列化类,list,retrieve(get_object间接),put,create
get_queryset       # 要序列化的总数据或单条查询的数据源
get_object     # 获取单条,通过get_queryset拿到的
filter_queryset    # 过滤使用,必须配合list使用,视图类中重写它可以完成过滤,就不用配置过滤类了
get_paginated_response    # 获取分页后的返回数据,必须配合listdef get_paginated_response(self, data):assert self.paginator is not Nonereturn Response({'code': 100,'msg': 'asdfasfasfdasf','next': self.paginator.get_next_link(),'pre': self.paginator.get_previous_link(),'result': data})

# 5个视图扩展类--必须搭配GenericAPIView      

ListModelMixin  # 过滤,排序,分页
def list(self, request, *args, **kwargs):queryset = self.filter_queryset(self.get_queryset())page = self.paginate_queryset(queryset)   # 处理了分页if page is not None:   # 如果没有分页,正常返回serializer = self.get_serializer(page, many=True)return self.get_paginated_response(serializer.data)serializer = self.get_serializer(queryset, many=True)return Response(data=serializer.data)RetrieveModelMixin
def retrieve(self, request, *args, **kwargs):instance = self.get_object() # 序列化单挑数据--》重写serializer = self.get_serializer(instance) # 重写--》使用哪个序列化类return Response(serializer.data)   CreateModelMixin
def create(self, request, *args, **kwargs):serializer = self.get_serializer(data=request.data) # 重写serializer.is_valid(raise_exception=True)self.perform_create(serializer) #做了真正的保存serializer.save(),重写return Response(serializer.data, status=status.HTTP_201_CREATED)
def perform_create(self, serializer):  #可以重写serializer.save()DestroyModelMixin
def destroy(self, request, *args, **kwargs):instance = self.get_object()    # 通过重写它,决定删除谁self.perform_destroy(instance)  # 重写-软删除,不在数据库删除,is_delete# 设置为True,以后要序列化的时候,不查出来了return Response(status=status.HTTP_204_NO_CONTENT)
def perform_destroy(self, instance):instance.delete()UpdateModelMixin
def update(self, request, *args, **kwargs):instance = self.get_object()serializer = self.get_serializer(instance, data=request.data)serializer.is_valid(raise_exception=True)self.perform_update(serializer)return Response(serializer.data)
def perform_update(self, serializer):serializer.save()   
# views.py
from rest_framework.mixins import CreateModelMixin, ListModelMixin, RetrieveModelMixin, UpdateModelMixin, \DestroyModelMixinclass BookView(GenericAPIView, CreateModelMixin, ListModelMixin):queryset = Book.objects.all()serializer_class = BookSerializerdef get(self, request):return super().list(request)def post(self, request):# 做保存,加了这一句---》目的是:子类可以重写,增强扩展性# self.perform_create(serializer)return super().create(request)class BookDetailView(GenericAPIView, RetrieveModelMixin, DestroyModelMixin, UpdateModelMixin):queryset = Book.objects.all()serializer_class = BookSerializerdef put(self, request, *args, **kwargs):return super().update(request, *args, **kwargs)def get(self, request, *args, **kwargs):return super().retrieve(request, *args, **kwargs)def delete(self, request, *args, **kwargs):return super().destroy(request, *args, **kwargs)

# 9 个视图子类--重写 get_serializer_class,get_queryset,get_object  

ListAPIView:GenericAPIView+ListModelMixin  # 重写:list,get
RetrieveAPIView   # 重写retrieve,get
CreateAPIView     # 重写create,perform_create
UpdateAPIView
DestroyAPIView
ListCreateAPIView
RetrieveUpdateDestroyAPIView,
RetrieveDestroyAPIView,
RetrieveUpdateAPIView
# views.py
from rest_framework.generics import CreateAPIView, ListAPIView
from rest_framework.generics import RetrieveAPIView, DestroyAPIView, UpdateAPIView
from rest_framework.generics import ListCreateAPIView
from rest_framework.generics import RetrieveUpdateDestroyAPIView, RetrieveDestroyAPIView, RetrieveUpdateAPIView
# from rest_framework.generics import DestroyUpdateAPIView # 一般不存在,所以就没有# 实现新增,查所有和查询一条
class BookView(ListCreateAPIView):# 配置两个类属性queryset = Book.objects.all()serializer_class = BookSerializerclass BookDetailView(RetrieveAPIView):queryset = Book.objects.all()serializer_class = BookSerializer

# 视图集       

        以后写视图类: 一般不会5个都写
        比如只想写 查询所有和删除一条--继承--》GenericViewSet,ListModelMixin

ViewSetMixin   # 路由写法变了---》多个视图类可以放到一起---》一个视图类中可以放多个方法
GenericViewSet
ViewSet
ModelViewSet   # 有get,post这些方法吗?没有,只有list,create。。这些
# 重写:get_serializer_class,get_queryset,perform_destroy
# 重写list,分页返回格式
ReadOnlyModelViewSet:只有 list,retrieve


 

drf回顾之路由组件

#  视图类没有继承了ViewSetMixin,路由写法跟之前一样
            path('books/', views.BookView.as_view())

# 映射写法:ViewSetMixin

    SimperRouter和DefaultRouter    实例化

from rest_framework.routers import SimpleRouter,DefaultRouter  
router = SimpleRouter()  # router = DefaultRouter()
router.register('books', views.BookView, 'books') 
urlpatterns = [# path('', include(router.urls)),
]
urlpatterns += router.urls

# 视图类中自己的方法,再做映射--action装饰器

        action装饰器:methods,detail

@action(methods=['POST'],detail=False,)
def login(self,request):return Response('login')

drf回顾之三大认证

# 认证类的使用
# 权限类:
        登录成功后,有没有权限---》request.user拿到登录用户,判断权限
# 频率类:
        ip,用户id
        用户访问记录:存在缓存中,默认在内存

# auth.py 认证类
from rest_framework.authentication import BaseAuthentication
from rest_framework.exceptions import AuthenticationFailed
from .models import UserTokenclass LoginAuth(BaseAuthentication):def authenticate(self, request):token = request.query_params.get('token') or request.META.get('HTTP_TOKEN')user_token = UserToken.objects.filter(token=token).first()if user_token:user = user_token.userreturn user, user_tokenelse:raise AuthenticationFailed("您没有登录")
# permission.py  权限类
from rest_framework.permissions import BasePermission
class CommonPermission(BasePermission):def has_permission(self, request, view):try:user_type = request.user.user_typeif user_type == 2:     # 只有超级管理员能访问,其他人都不能访问return Trueelse:self.message = '您是:%s,您没有权限操作这个' % request.user.get_user_type_display()return Falseexcept Exception as e:            if 'login' in request.path:   # 未登录用户也没有权限return Trueelse:self.message = '您没有登录,您没有权限操作这个'return False
# throttling.py  频率类
from rest_framework.throttling import  SimpleRateThrottleclass CommonThrottle(SimpleRateThrottle):rate = '3/m' def get_cache_key(self, request, view):ip = request.META.get('REMOTE_ADDR')return ip
# setting.py
REST_FRAMEWORK = {# 全局配置登录认证'DEFAULT_AUTHENTICATION_CLASSES': ['app01.auth.LoginAuth'],    # 认证组件'DEFAULT_PERMISSION_CLASSES': ['app01.permission.CommonPermission'],   # 权限组件'DEFAULT_THROTTLE_CLASSES': ['app01.throttling.CommonThrottle'],   # 频率组件
}
# views.py  登录入口   1、拿到验证码
class UserView(ViewSet):authentication_classes = []permission_classes = []throttle_classes = []@action(methods=['POST'], detail=False)def login(self, request):username = request.data.get('username')password = request.data.get('password')user = User.objects.filter(name=username, password=password).first()if user:token = str(uuid.uuid4())UserToken.objects.update_or_create(defaults={'token': token}, user_id=user.pk)return Response({'code': '100', 'msg': '登录成功', 'token': token})else:return Response({'code': '101', 'msg': '用户名或密码错误'})class BookView(ViewSetMixin, ListCreateAPIView):authentication_classes = [LoginAuth]  # 验证是否登录queryset = Book.objects.all()serializer_class = BookSerializer  
class BookDetailView(GenericViewSet, DestroyModelMixin, RetrieveModelMixin, UpdateModelMixin):queryset = Book.objects.all()serializer_class = BookSerializer

drf回顾之排序过滤和分页

-排序:内置排序  list接口,继承GenericAPIView+list
        -filter_backends 配置 排序类
        -ordering=id,-price
    -过滤:
        -内置
        -第三方
        -自定义:写个类,继承:BaseFilterBackend,重写filter_queryset在内部完成过滤,返回过滤后的qs对象
        
    -过滤,排序可以连用,多个过滤也可以连用

- list接口+GenericAPIView 配置分页类
    - 三种分页方式
        -都有自己的类属性       
 -继承APIView实现分页
    from rest_framework.viewsets import ViewSet
    from rest_framework.mixins import ListModelMixin,RetrieveModelMixin
    from rest_framework.generics import GenericAPIView
    from . import pagination
    class BookView(ViewSet):
        def list(self, request):
            book_list = Book.objects.all()
            # 调用咱们写的分页类对象的paginate_queryset方法返回了 分页后的qs对象
            # pagenation = pagination.CommonPageNumberPagination()
            # pagenation = pagination.CommonLimitOffsetPagination()
            pagenation = pagination.CommonCursorPagination()
            page = pagenation.paginate_queryset(book_list, request, self)
            ser = BookSerializer(instance=page, many=True)
            # return pagenation.get_paginated_response(ser.data)
            # CommonPageNumberPagination 返回格式
            # return Response({'code': 100,
            #                  'msg': '查询成功',
            #                  'count': pagenation.page.paginator.count,
            #                  'next': pagenation.get_next_link(),
            #                  'pre': pagenation.get_previous_link(),
            #                  'result': ser.data})
            # CommonLimitOffsetPagination
            # return Response({
            #     'code': 100,
            #     'msg': '查询成功',
            #     'count': pagenation.count,
            #     'next': pagenation.get_next_link(),
            #     'pre': pagenation.get_previous_link(),
            #     'result': ser.data})

            # CommonCursorPagination
            return Response({
                'code': 100,
                'msg': '查询成功',
                'next': pagenation.get_next_link(),
                'pre': pagenation.get_previous_link(),
                'result': ser.data})

drf回顾之全局异常

-全局异常:
        -写个函数,完成处理,配置到配置文件---》以后只要出了异常,都会走咱么的函数

今日思维导图:

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

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

相关文章

【LeetCode-剑指offer】--15.找到字符串中所有字母异位词

15.找到字符串中所有字母异位词 方法&#xff1a;滑动窗口 class Solution {public List<Integer> findAnagrams(String s, String p) {List<Integer> ans new ArrayList<>();int m s.length(),n p.length();if(n > m){return ans;}int[] cnt1 new i…

YoloV7改进策略:AAAI 2024 最新的轴向注意力|即插即用,改进首选|全网首发,包含数据集和代码,开箱即用!

摘要 https://arxiv.org/pdf/2312.08866.pdf 本文提出了一种名为Multi-scale Cross-axis Attention(MCA)的方法,用于解决医学图像分割中的多尺度信息和长距离依赖性问题。该方法基于高效轴向注意力,通过计算两个平行轴向注意力之间的双向交叉注意力,更好地捕获全局信息。…

无人机集群反制与对抗技术探讨

源自&#xff1a;指挥与控制学报 作者&#xff1a;任 智 张 栋 唐 硕 王孟阳 李智军 “人工智能技术与咨询” 发布 摘 要 无人机集群系统是现代信息化战争体系对抗中重要的新质作战力量, 是未来网络化、智能化作战的关键发展方向. 随着无人集群系统装备实战化技术的…

第二十七章 正则表达式

第二十七章 正则表达式 1.正则快速入门2.正则需求问题3.正则底层实现14.正则底层实现25.正则底层实现36.正则转义符7.正则字符匹配8.字符匹配案例19.字符匹配案例211.选择匹配符&#xff08;|&#xff09;12.正则限定符{n}{n,m}&#xff08;1个或者多个&#xff09;*(0个或者多…

『华为云耀云服务器实战』|云服务器如何快速搭建个人博客(图文详解)

文章目录 引言一、云耀云服务器L实例介绍1.1 准备一个华为云耀云服务器1.2 重置实例密码1.3 利用xshell 远程连接 二、安装环境软件2.1 安装git准备远程拉取2.2 安装Docker 和 Docker compose 三、博客开源项目介绍3.1 操作界面展览 四、拉取项目搭建个人博客4.1 拉取项目进行配…

【linux kernel】linux的SPI框架分析

文章目录 一、linux内核中的SPI框架二、SPI核心的初始化三、SPI核心的数据结构1、struct spi_statistics2、struct spi_delay3、struct spi_device4、struct spi_driver5、struct spi_controller6、struct spi_res7、struct spi_transfer8、struct spi_message9、struct spi_bo…

CMake支持的编译平台和IDE

文章目录 简介支持的IDEVisual Studio支持示例 其他编译器和生成器支持MinGW示例 IDE集成Eclipse示例 实验性和特殊平台支持总结 简介 CMake是一个非常强大的跨平台自动化构建工具&#xff0c;它支持生成多种类型的项目文件&#xff0c;覆盖了广泛的开发环境和编译器。在这篇博…

基于PCA-WA(Principal Component Analysis-weight average)的图像融合方法 Matlab代码及示例

摘要&#xff1a; 高效地将多通道的图像数据压缩&#xff08;如高光谱、多光谱成像数据&#xff09;至较低的通道数&#xff0c;对提高深度学习&#xff08;DL&#xff09;模型的训练速度和预测至关重要。本文主要展示利用PCA降维结合weight-average的图像融合方法。文章主要参…

leetcode第206题反转链表❤

一&#xff1a;题目&#xff1a; 给你单链表的头节点 head &#xff0c;请你反转链表&#xff0c;并返回反转后的链表。 示例 1&#xff1a; 输入&#xff1a;head [1,2,3,4,5] 输出&#xff1a;[5,4,3,2,1] 题目链接&#xff1a;力扣&#xff08;LeetCode&#xff09;官网…

【OpenCV】在MacOS上源码编译OpenCV

在MacOS上源码编译OpenCV 1. 下载项目源码2. 创建CMake编译文件3. 编译安装4. 案例测试5. 总结 前言 在做视觉任务时&#xff0c;我们经常会用到开源视觉库OpenCV&#xff0c;OpenCV是一个基于Apache2.0许可&#xff08;开源&#xff09;发行的跨平台计算机视觉和机器学习软件…

js中函数动态调用

文章目录 一、场景二、方法2.1、动态函数2.2、eval()函数 三、最后 一、场景 在JS开发中&#xff0c;例如有些场景下&#xff0c;后端要求一个功能要请求不同的接口&#xff0c;但是传参及后续逻辑其实都是一样的&#xff0c;有些同学可能会想到在接口url处统一处理就好&#…

在 docker 容器中配置双网卡,解决通讯的问题

目录 1. 查看当前网络信息 2. 创建自定义网络 3. 查看网卡信息 4. 建立双网卡模式 5. 查看容器的网络 6. 从双网卡中删除默认网卡 已经创建好了的 Docker 容器&#xff0c;要修改它的IP比较麻烦&#xff0c;网上找了几种不同的方法&#xff0c;经过试验都没有成功&…

HeyGen怎么使用想使用高级功能如何订阅

HeyGen是一款AI视频工具&#xff0c;可以让用户轻松地用不同语言说话。 HeyGen是一个用于生成“虚拟化身&#xff08;数字人&#xff09;”视频的工具。 根据其官网的介绍&#xff0c;HeyGen可以通过文本生成视频&#xff0c;有不同的视频模板&#xff0c;可以定制化人物形象、…

Android Studio报错:connect refused

报错信息 解决办法&#xff1a;在System settings里取消代理&#xff0c;将HTTP Proxy设置为 No proxy 但是我发现我的还不行&#xff0c;还是报错&#xff0c;还是connect refused&#xff1a; 我发现虽然在System settings里已经取消代理&#xff0c;但实际项目运行时还是走…

鸿蒙原生应用/元服务开发-Serverless账户验证码的问题

在应用/元服务早期使用过程中&#xff0c;-Serverless账户验证码的格式是[AGC][应用/元服务名称]&#xff0c;如下图。 但是&#xff0c;在最近&#xff0c;[应用/元服务]名称直接变成了【default】,用户收到这种验证码后&#xff0c;心里存有疑虑的&#xff0c;这是哪里配置…

OSG-渲染状态、纹理映射(一)

1、渲染状态 OSG 支持绝大部分的OpenGL固定功能管道(fixed function pipeline)渲染,如Alpha检验、Blending融合剪切平面、颜色蒙板、面选(face culling)深度和模板检验、雾效、点和线的光栅化(rasterization)等。OSG 的渲染状态也允许应用程序指定顶点着色(vertex shader)和片段…

Python 数据库(一):使用 mysql-connector-python 操作 MySQL 数据库

大家好&#xff0c;我是水滴~~ 当涉及到使用 Python 操作 MySQL 数据库时&#xff0c;mysql-connector-python 库是一个强大而常用的选择。该库提供了与 MySQL 数据库的交互功能&#xff0c;使您能够执行各种数据库操作&#xff0c;如连接数据库、执行查询和插入数据等。在本文…

R语言——R函数、选项参数、数学统计函数(六)

目录 一、R函数 二、选项参数 三、数学统计函数 四、参考 一、R函数 1.lm() lm()是R语言中经常用到的函数&#xff0c;用来拟合回归模型。它是拟合线性模型最基本的函数 lm()格式如下&#xff1a; fit<-lm(formula,data) 其中&#xff0c;formula指要拟合的模型形式…

draw流程图工具导入云原生(CNCF)相关控件

目录 1、通过draw导入xml文件&#xff0c;获取云原生相关的空间 2、引用自己的资源链接&#xff1a; 1、通过draw导入xml文件&#xff0c;获取云原生相关的空间 导入资源图库&#xff0c;资源放在下方&#xff0c;大家可以下载&#xff1a; 2、引用自己的资源链接&#xff1a;…

Vue实现JSON字符串格式化编辑器组件

相信很多同学都用过网上的在线JSON格式化工具来将杂乱的JSON数据转换成易于我们阅读和编辑的格式。那么&#xff0c;你有没有想过自己动手实现一个这样的工具呢&#xff1f;今天&#xff0c;我将介绍如何使用Vue.js来构建一个简单的JSON格式化工具。 功能简述 支持格式化JSON字…