- JSON Web Token
一、安装JWT库(推荐)#安装必要库pipinstalldjangorestframeworkdjangorestframework-simplejwt二、配置Django项目INSTALLED_APPS=[...'rest_framework','rest_framework_simplejwt',#添加JWT支持]REST_FRAMEWORK={'DEFAULT_AUTHENTICATION_
- Django Rest Framework 视图和路由
Matrix 工作室
从源代码学Python
DRF的视图APIView我们django中写CBV的时候继承的是View,rest_framework继承的是APIView,那么他们两个有什么不同呢urlpatterns=[url(r'^book$',BookView.as_view()),url(r'^book/(?P\d+)$'
- Django REST framework - 设置
djangopython
settings.py命名空间是个绝妙的主意,让我们多用用吧!——《Python之禅》DjangoREST框架的配置都放在一个命名空间内,即Django的一个设置,名为REST_FRAMEWORK。例如,项目的settings.py文件可能包含类似以下内容:REST_FRAMEWORK={'DEFAULT_RENDERER_CLASSES':['rest_framework.renderers.J
- Django REST Framework(十七)Authentication
yjjpp2301
DjangoRESTframeworkdjangopython
1.认证Authentication在DjangoRESTframework(DRF)中,可以在配置文件中配置全局默认的认证方案。常见的认证方式包括cookie、session、和token。DRF提供了灵活的认证机制,可以在全局配置文件中设置默认认证方式,也可以在具体的视图类中设置单独的认证方式。以下是默认的配置文件示例,位于REST_FRAMEWORK={#配置认证方式的选项'DEFAULT_
- django rest_framework
一.什么是RESTfulREST与技术无关,代表的是一种软件架构风格,REST是RepresentationalStateTransfer的简称,中文翻译为“表征状态转移”REST从资源的角度类审视整个网络,它将分布在网络中某个节点的资源通过URL进行标识,客户端应用通过URL来获取资源的表征,获得这些表征致使这些应用转变状态REST与技术无关,代表的是一种软件架构风格,REST是Represen
- 【Django DRF】使用rest_framework_simplejwt搭建jwt——全解
患得患失949
DjangoDRFdjangopython后端simplejwtjwt
一、Django中配置JWT身份认证(使用rest_framework_simplejwt)(一)安装rest_framework_simplejwtpipinstallrest_framework_simplejwt(二)在Django设置中启用JWT配置1.修改settings.py#settings.pyINSTALLED_APPS=[#其他应用'rest_framework','rest_
- django rest_framework 前端网页实现Token认证
bluemliu
django前端python
rest_framework提供了几种认证方式:Session、Token等。Session是最简单的,几乎不用写任何代码就可以是实现,Token方式其实也不复杂,网上的教程一大把,但是最后都是用Postman这类工具来实现API调用的,通过这类工具来增加HTTP头信息以回传Token。那么真正的前端网页应该怎么办呢?网上基本上就是基于Aixos来实现的,但是我就不想用Vue,纯Javascrip
- rest_framework permission_classes 无效的解决方法
bluemliu
python开发语言django
写了一个特别简单的view:@csrf_exempt@login_required()@authentication_classes([TokenAuthentication])@permission_classes([IsAdminUser,IsAuthenticated])defdepartment_management_view(request):ifrequest.method=='POS
- Django中为api自定义一些装饰器:如参数校验等
在Django中使用了rest_framework时,一般我们会定义ModelSerializer来校验request.data中参数是否存在和参数类型。但当我们只是想简单校验一些api的url上是否存在某些参数时,该怎么办?当然我们也可以通过定义Serializer来实现,但很麻烦。我们可以自定义请求参数验证装饰器来实现。类似如下(代码不是完整的,只是示例):fromfunctoolsimpor
- DRF使用文档功能
__tian__
PythonRoadDRFDRF
DRF自带的的文档生成功能,非常方便。配置:fromrest_framework.documentationimportinclude_docs_urlsurlpatten=[……url(r'docs/',include_docs_urls(title="yitao")),……]在settings.py文件中,需加上REST_FRAMEWORK={'DEFAULT_SCHEMA_CLASS':'r
- 快速入手-基于Django-rest-framework的限流操作(十二)
神奇侠2024
djangopythonDRF限流
限流:对接口访问的频次进行限制,以减轻服务器压力或者实现特定的业务。一般用于付费购买次数、投票等场景使用。配置方式有两种:全局配置和局部配置。1、全局配置REST_FRAMEWORK={"DEFAULT_AUTHENTICATION_CLASSES":("rest_framework_simplejwt.authentication.JWTAuthentication",),"DEFAULT_TH
- 快速入手-Django-rest-framework(一)
神奇侠2024
django
1、安装DjangoRESTFrameworkpipinstalldjangorestframework2、快速构建django项目基本结构,参考以下链接创建api模块,并注册应用快速入手-Django项目创建(一)-CSDN博客3、添加到INSTALLED_APPSINSTALLED_APPS=[...'rest_framework',]4、在api模块里创建urls.pyfromdjango.
- DRF框架使用djangorestframework-simplejwt实现自定义用户类的token校验
lj907722644
PythonpythondjangojwtDRFsimplejwt
1.安装simplejwt库并修改settings.py安装simplejwtpipinstalldjangorestframeworkpipinstalldjangorestframework-simplejwt修改settings.py注册应用INSTALLED_APPS=[...'rest_framework','rest_framework_simplejwt',]设置jwt鉴权REST_
- DRF 分页器的使用
.咖啡加剁椒.
软件测试软件测试自动化测试功能测试程序人生职场和发展
drf提供了三个内置分页器,根据前端需求选择使用。全局配置在配置文件中设置全局的分页方式,如:REST_FRAMEWORK={'DEFAULT_PAGINATION_CLASS':'rest_framework.pagination.PageNumberPagination','PAGE_SIZE':100#每页数目}也可通过继承内置的分页器类自定义Pagination类,来为视图添加不同分页器。
- Django rest_framework 后端接口开发 开发与用户相关的一组接口 登录注册与用户信息查询修改
勇敢牛马 不怕困难
全栈开发djangopythonmvcrestful后端
Djangorest_framework后端接口开发开发与用户相关的一组接口DjangoDRF框架用起来还是有一些难度的,需要做的配置,需要导的包很多,所以需要多多练习才能掌握它的使用。此文记录了使用Djangorest_famework框架开发用户模块相关接口的流程,需要注意的点,以及源码。期间重写了Django用户模型类,自定义了DjangoResponse消息格式等。此文不过多赘述环境及各种
- Django的web框架Django Rest_Framework精讲(四)
景天科技苑
Django框架djangopythonDRF
文章目录1.DRF认证组件Authentication2.权限Permissions3.限流Throttling4.过滤Filtering5.排序6.分页Pagination7.异常处理Exceptions8.自动生成接口文档大家好,我是景天,今天我们继续DRF的最后一讲,Django的web框架DjangoRest_Framework(四)1.DRF认证组件AuthenticationDRF除了
- Django的web框架Django Rest_Framework精讲(二)
景天科技苑
Django框架djangopythonDRF
文章目录1.自定义校验功能(1)validators(2)局部钩子:单字段校验(3)全局钩子:多字段校验2.raise_exception参数3.context参数4.反序列化校验后保存,新增和更新数据(1)保存数据方式1(2)数据保存方式2(3)更新数据1、更新数据方式12、更新数据方式25.ORM中的CharField6.ModelSerializer模型类序列化器大家好,我是景天,今天我们继
- 全面掌握Django的web框架Django Rest_Framework(一)
景天科技苑
Django框架djangopython后端DRF
文章目录DjangoRest_Framework1.DRF介绍2.DRF特点3.环境安装与配置(1)DRF需要以下依赖(2)创建django项目4.序列化器的使用(1)创建序列化器5.反序列化器使用DjangoRest_Framework1.DRF介绍DjangoRESTframework是一个建立在Django基础之上的Web应用开发框架,可以快速的开发RESTAPI接口应用,简称DRF。在RE
- reful实现分页、过滤、搜索、软删除
kris_lp
一、分页1.配置setting.pyREST_FRAMEWORK={#实现分页'DEFAULT_PAGINATION_CLASS':'rest_framework.pagination.PageNumberPagination','PAGE_SIZE':2,二、搜索1.配置setting.py#配置restfulapi返回结果REST_FRAMEWORK={#实现过滤'DEFAULT_FILTER
- Django笔记(七):JWT认证
垃圾管理员
Django笔记django笔记python
首前后端分离的项目更多使用JWT认证——JsonWebToken。本文记录djangorestframework-simplejwt的使用方式。文档安装pipinstalldjangorestframework-simplejwt配置settings.py:INSTALLED_APPS=['rest_framework_simplejwt',]REST_FRAMEWORK={'DEFAULT_AU
- django rest_framework 部署doc文档
小赖同学啊
pythondjangopython后端
1.背景在实际开发过程中,前后端分离的项目,是需要将一份完整的接口文档交付给前端开发人员,这样有利于开发速度和开发质量,以及有可能减少协同时间。2.内容本项目是以Python+django+rest_framework作为技术框架,在这套框架中,是有自己支持的api文档,现将实现方式整理如下2.1默认基础框架已经安装成功2.2在主项目中的url中配置如下代码,基于经验在应用路由也可以实现#在url
- DRF(Django rest_framework)(2 视图部分和路由)
Invictus path
djangopython后端restful
这边先分享一个DRF的比较全的各种组件的用法的地址(DRF|YUAN),但是讲得比较专业,我这边更通俗,是按照封装过程来讲解的,所以可能会更清晰一步一步怎么来的,为什么DRF视图!!!!!DRF视图DRF提供的视图的主要作用:基于View的接口(原生)fromdjango.viewsimportView就是从FBV到CBV的第一步,面向资源编程,将对一个资源的操作全放到一个类中,需要继承View类
- DRF(Django rest_framework)(1序列化器部分)
Invictus path
djangorestfulpython
简介:restfulrestful就是一个接口的规范,也就是面向资源的url接口视图函数只返回Json类型给前端,然后把json数据交给js,让前端的js去处理html页面变化1.序列化器序列化要传参instance(传模型类对象),反序列化要传参data(传request中的数据)因为是增删改查查,所以一个View不能有两个get方法,那么就分开两个视图类一个放不需要传id的BookView,一
- 过滤、排序、分页、异常处理
-wellplayed-
drfrestful后端python开发语言django
一过滤Filtering对于列表数据可能需要根据字段进行过滤,我们可以通过添加django-fitlter扩展来增强支持。pipinstalldjango-filter在配置文件中增加过滤后端的设置:INSTALLED_APPS=[...'django_filters',#需要注册应用,]REST_FRAMEWORK={...'DEFAULT_FILTER_BACKENDS':('django_f
- Django REST framework框架API接口序列化之简易实现
十里染林
一、重点是用到rest_framework框架的generics脚本文件,在view视图中导入fromrest_frameworkimportgenerics,定义视图类时直接继承调用generics的两个类分别是:ListCreateAPIView,RetrieveUpdateDestroyAPIView这两个类封装了增删改查方法1.png2.png二、浅析generics.pygenerics
- Django Rest Framework框架的安装
笛秋白
Djangodjangopython后端个人开发
DjangoRestFramework框架的安装DjangoRestFramework框架的安装1.DRF简介2.安装依赖3.安装使用pip安装添加rest_framework应用1.DRF简介DjangoRESTFramework是Webapi的工具包。它是在Django框架基础之上,进行了二次开发。2.安装依赖链接python安装Python(3.5,3.6,3.7,3.8,3.9)链接Dja
- Django rest_framework实现RESTful API-晒酷学院
晒酷学院
一、什么是REST面向资源是REST最明显的特征,资源是一种看待服务器的方式,将服务器看作是由很多离散的资源组成。每个资源是服务器上一个可命名的抽象概念。因为资源是一个抽象的概念,所以它不仅仅能代表服务器文件系统中的一个文件、数据库中的一张表等等具体的东西,可以将资源设计的要多抽象有多抽象,只要想象力允许而且客户端应用开发者能够理解。与面向对象设计类似,资源是以名词为核心来组织的,首先关注的是名词
- rest_framework_django学习笔记一(序列化器)
严不纯
django学习sqlite
rest_framework_django学习笔记一(序列化器)一、引入DjangoRestFramework1、安装pipinstalldjangorestframework2、引入INSTALLED_APPS=[...'rest_framework',]3、原始RESTful接口写法models.pyfromdjango.dbimportmodels'''测试数据仅供参考INSERTINTO`
- rest_framework之视图
aq_wzj
对请求响应到的视图函数的处理,及视图函数的封装注:mySer.BookSerializers为序列化的类一,对之前的增删改查进行封装方式一:fromrest_framework.mixinsimportDestroyModelMixin,CreateModelMixin,ListModelMixin,UpdateModelMixin,RetrieveModelMixinfromrest_frame
- 身份验证drf版
互联网中的一个咸鱼
参考官方文档设置认证方案全局设置REST_FRAMEWORK={'DEFAULT_AUTHENTICATION_CLASSES':('rest_framework.authentication.BasicAuthentication','rest_framework.authentication.SessionAuthentication',)}单个视图内设置使用基于APIView的视图fromr
- SAX解析xml文件
小猪猪08
xml
1.创建SAXParserFactory实例
2.通过SAXParserFactory对象获取SAXParser实例
3.创建一个类SAXParserHander继续DefaultHandler,并且实例化这个类
4.SAXParser实例的parse来获取文件
public static void main(String[] args) {
//
- 为什么mysql里的ibdata1文件不断的增长?
brotherlamp
linuxlinux运维linux资料linux视频linux运维自学
我们在 Percona 支持栏目经常收到关于 MySQL 的 ibdata1 文件的这个问题。
当监控服务器发送一个关于 MySQL 服务器存储的报警时,恐慌就开始了 —— 就是说磁盘快要满了。
一番调查后你意识到大多数地盘空间被 InnoDB 的共享表空间 ibdata1 使用。而你已经启用了 innodbfileper_table,所以问题是:
ibdata1存了什么?
当你启用了 i
- Quartz-quartz.properties配置
eksliang
quartz
其实Quartz JAR文件的org.quartz包下就包含了一个quartz.properties属性配置文件并提供了默认设置。如果需要调整默认配置,可以在类路径下建立一个新的quartz.properties,它将自动被Quartz加载并覆盖默认的设置。
下面是这些默认值的解释
#-----集群的配置
org.quartz.scheduler.instanceName =
- informatica session的使用
18289753290
workflowsessionlogInformatica
如果希望workflow存储最近20次的log,在session里的Config Object设置,log options做配置,save session log :sessions run ;savesessio log for these runs:20
session下面的source 里面有个tracing 
- Scrapy抓取网页时出现CRC check failed 0x471e6e9a != 0x7c07b839L的错误
酷的飞上天空
scrapy
Scrapy版本0.14.4
出现问题现象:
ERROR: Error downloading <GET http://xxxxx CRC check failed
解决方法
1.设置网络请求时的header中的属性'Accept-Encoding': '*;q=0'
明确表示不支持任何形式的压缩格式,避免程序的解压
- java Swing小集锦
永夜-极光
java swing
1.关闭窗体弹出确认对话框
1.1 this.setDefaultCloseOperation (JFrame.DO_NOTHING_ON_CLOSE);
1.2
this.addWindowListener (
new WindowAdapter () {
public void windo
- 强制删除.svn文件夹
随便小屋
java
在windows上,从别处复制的项目中可能带有.svn文件夹,手动删除太麻烦,并且每个文件夹下都有。所以写了个程序进行删除。因为.svn文件夹在windows上是只读的,所以用File中的delete()和deleteOnExist()方法都不能将其删除,所以只能采用windows命令方式进行删除
- GET和POST有什么区别?及为什么网上的多数答案都是错的。
aijuans
get post
如果有人问你,GET和POST,有什么区别?你会如何回答? 我的经历
前几天有人问我这个问题。我说GET是用于获取数据的,POST,一般用于将数据发给服务器之用。
这个答案好像并不是他想要的。于是他继续追问有没有别的区别?我说这就是个名字而已,如果服务器支持,他完全可以把G
- 谈谈新浪微博背后的那些算法
aoyouzi
谈谈新浪微博背后的那些算法
本文对微博中常见的问题的对应算法进行了简单的介绍,在实际应用中的算法比介绍的要复杂的多。当然,本文覆盖的主题并不全,比如好友推荐、热点跟踪等就没有涉及到。但古人云“窥一斑而见全豹”,希望本文的介绍能帮助大家更好的理解微博这样的社交网络应用。
微博是一个很多人都在用的社交应用。天天刷微博的人每天都会进行着这样几个操作:原创、转发、回复、阅读、关注、@等。其中,前四个是针对短博文,最后的关注和@则针
- Connection reset 连接被重置的解决方法
百合不是茶
java字符流连接被重置
流是java的核心部分,,昨天在做android服务器连接服务器的时候出了问题,就将代码放到java中执行,结果还是一样连接被重置
被重置的代码如下;
客户端代码;
package 通信软件服务器;
import java.io.BufferedWriter;
import java.io.OutputStream;
import java.io.O
- web.xml配置详解之filter
bijian1013
javaweb.xmlfilter
一.定义
<filter>
<filter-name>encodingfilter</filter-name>
<filter-class>com.my.app.EncodingFilter</filter-class>
<init-param>
<param-name>encoding<
- Heritrix
Bill_chen
多线程xml算法制造配置管理
作为纯Java语言开发的、功能强大的网络爬虫Heritrix,其功能极其强大,且扩展性良好,深受热爱搜索技术的盆友们的喜爱,但它配置较为复杂,且源码不好理解,最近又使劲看了下,结合自己的学习和理解,跟大家分享Heritrix的点点滴滴。
Heritrix的下载(http://sourceforge.net/projects/archive-crawler/)安装、配置,就不罗嗦了,可以自己找找资
- 【Zookeeper】FAQ
bit1129
zookeeper
1.脱离IDE,运行简单的Java客户端程序
#ZkClient是简单的Zookeeper~$ java -cp "./:zookeeper-3.4.6.jar:./lib/*" ZKClient
1. Zookeeper是的Watcher回调是同步操作,需要添加异步处理的代码
2. 如果Zookeeper集群跨越多个机房,那么Leader/
- The user specified as a definer ('aaa'@'localhost') does not exist
白糖_
localhost
今天遇到一个客户BUG,当前的jdbc连接用户是root,然后部分删除操作都会报下面这个错误:The user specified as a definer ('aaa'@'localhost') does not exist
最后找原因发现删除操作做了触发器,而触发器里面有这样一句
/*!50017 DEFINER = ''aaa@'localhost' */
原来最初
- javascript中showModelDialog刷新父页面
bozch
JavaScript刷新父页面showModalDialog
在页面中使用showModalDialog打开模式子页面窗口的时候,如果想在子页面中操作父页面中的某个节点,可以通过如下的进行:
window.showModalDialog('url',self,‘status...’); // 首先中间参数使用self
在子页面使用w
- 编程之美-买书折扣
bylijinnan
编程之美
import java.util.Arrays;
public class BookDiscount {
/**编程之美 买书折扣
书上的贪心算法的分析很有意思,我看了半天看不懂,结果作者说,贪心算法在这个问题上是不适用的。。
下面用动态规划实现。
哈利波特这本书一共有五卷,每卷都是8欧元,如果读者一次购买不同的两卷可扣除5%的折扣,三卷10%,四卷20%,五卷
- 关于struts2.3.4项目跨站执行脚本以及远程执行漏洞修复概要
chenbowen00
strutsWEB安全
因为近期负责的几个银行系统软件,需要交付客户,因此客户专门请了安全公司对系统进行了安全评测,结果发现了诸如跨站执行脚本,远程执行漏洞以及弱口令等问题。
下面记录下本次解决的过程以便后续
1、首先从最简单的开始处理,服务器的弱口令问题,首先根据安全工具提供的测试描述中发现应用服务器中存在一个匿名用户,默认是不需要密码的,经过分析发现服务器使用了FTP协议,
而使用ftp协议默认会产生一个匿名用
- [电力与暖气]煤炭燃烧与电力加温
comsci
在宇宙中,用贝塔射线观测地球某个部分,看上去,好像一个个马蜂窝,又像珊瑚礁一样,原来是某个国家的采煤区.....
不过,这个采煤区的煤炭看来是要用完了.....那么依赖将起燃烧并取暖的城市,在极度严寒的季节中...该怎么办呢?
&nbs
- oracle O7_DICTIONARY_ACCESSIBILITY参数
daizj
oracle
O7_DICTIONARY_ACCESSIBILITY参数控制对数据字典的访问.设置为true,如果用户被授予了如select any table等any table权限,用户即使不是dba或sysdba用户也可以访问数据字典.在9i及以上版本默认为false,8i及以前版本默认为true.如果设置为true就可能会带来安全上的一些问题.这也就为什么O7_DICTIONARY_ACCESSIBIL
- 比较全面的MySQL优化参考
dengkane
mysql
本文整理了一些MySQL的通用优化方法,做个简单的总结分享,旨在帮助那些没有专职MySQL DBA的企业做好基本的优化工作,至于具体的SQL优化,大部分通过加适当的索引即可达到效果,更复杂的就需要具体分析了,可以参考本站的一些优化案例或者联系我,下方有我的联系方式。这是上篇。
1、硬件层相关优化
1.1、CPU相关
在服务器的BIOS设置中,可
- C语言homework2,有一个逆序打印数字的小算法
dcj3sjt126com
c
#h1#
0、完成课堂例子
1、将一个四位数逆序打印
1234 ==> 4321
实现方法一:
# include <stdio.h>
int main(void)
{
int i = 1234;
int one = i%10;
int two = i / 10 % 10;
int three = i / 100 % 10;
- apacheBench对网站进行压力测试
dcj3sjt126com
apachebench
ab 的全称是 ApacheBench , 是 Apache 附带的一个小工具 , 专门用于 HTTP Server 的 benchmark testing , 可以同时模拟多个并发请求。前段时间看到公司的开发人员也在用它作一些测试,看起来也不错,很简单,也很容易使用,所以今天花一点时间看了一下。
通过下面的一个简单的例子和注释,相信大家可以更容易理解这个工具的使用。
- 2种办法让HashMap线程安全
flyfoxs
javajdkjni
多线程之--2种办法让HashMap线程安全
多线程之--synchronized 和reentrantlock的优缺点
多线程之--2种JAVA乐观锁的比较( NonfairSync VS. FairSync)
HashMap不是线程安全的,往往在写程序时需要通过一些方法来回避.其实JDK原生的提供了2种方法让HashMap支持线程安全.
- Spring Security(04)——认证简介
234390216
Spring Security认证过程
认证简介
目录
1.1 认证过程
1.2 Web应用的认证过程
1.2.1 ExceptionTranslationFilter
1.2.2 在request之间共享SecurityContext
1
- Java 位运算
Javahuhui
java位运算
// 左移( << ) 低位补0
// 0000 0000 0000 0000 0000 0000 0000 0110 然后左移2位后,低位补0:
// 0000 0000 0000 0000 0000 0000 0001 1000
System.out.println(6 << 2);// 运行结果是24
// 右移( >> ) 高位补"
- mysql免安装版配置
ldzyz007
mysql
1、my-small.ini是为了小型数据库而设计的。不应该把这个模型用于含有一些常用项目的数据库。
2、my-medium.ini是为中等规模的数据库而设计的。如果你正在企业中使用RHEL,可能会比这个操作系统的最小RAM需求(256MB)明显多得多的物理内存。由此可见,如果有那么多RAM内存可以使用,自然可以在同一台机器上运行其它服务。
3、my-large.ini是为专用于一个SQL数据
- MFC和ado数据库使用时遇到的问题
你不认识的休道人
sqlC++mfc
===================================================================
第一个
===================================================================
try{
CString sql;
sql.Format("select * from p
- 表单重复提交Double Submits
rensanning
double
可能发生的场景:
*多次点击提交按钮
*刷新页面
*点击浏览器回退按钮
*直接访问收藏夹中的地址
*重复发送HTTP请求(Ajax)
(1)点击按钮后disable该按钮一会儿,这样能避免急躁的用户频繁点击按钮。
这种方法确实有些粗暴,友好一点的可以把按钮的文字变一下做个提示,比如Bootstrap的做法:
http://getbootstrap.co
- Java String 十大常见问题
tomcat_oracle
java正则表达式
1.字符串比较,使用“==”还是equals()? "=="判断两个引用的是不是同一个内存地址(同一个物理对象)。 equals()判断两个字符串的值是否相等。 除非你想判断两个string引用是否同一个对象,否则应该总是使用equals()方法。 如果你了解字符串的驻留(String Interning)则会更好地理解这个问题。
- SpringMVC 登陆拦截器实现登陆控制
xp9802
springMVC
思路,先登陆后,将登陆信息存储在session中,然后通过拦截器,对系统中的页面和资源进行访问拦截,同时对于登陆本身相关的页面和资源不拦截。
实现方法:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23