session和cookie的关系

cookie:保存在客户端浏览器上的键值对
session_id = "eyJ1c2VyX2luZm8iOiJ" session:保存在服务器上的键值对
{
"eyJ1c2VyX2luZm8iOiJ":{'is_login':True, 'user':'standby',...},
"iJhbGV4In0.DYUE4A.A":{'is_login':True, 'user':'alex',...},
...
} - 用户第一次打开浏览器请求我的网站页面
- 在服务器端生成一个随机字符串,作为value发给客户端浏览器。
- 这个随机字符串在服务器的session中作为key,value={},保存起来。

知识准备

1. 通过给定字符串,如何实例化一个对象出来?

import importlib
path = "session_code.RedisSession"
md,cls = path.rsplit('.',maxsplit=1)
m = importlib.import_module(md)
cls = getattr(m,cls)
print(cls.__name__)

或者

import importlib
path = "scrapy.middleware.MiddlewareManager"
tmp_li = path.split('.')
cls_name = tmp_li.pop()
prefix = '.'.join(tmp_li)
m = importlib.import_module(prefix)
cls = getattr(m,cls_name)
print(cls.__name__)

rest_framework/setting.py

def import_from_string(val, setting_name):
"""
Attempt to import a class from a string representation.
"""
try:
# Nod to tastypie's use of importlib.
module_path, class_name = val.rsplit('.', 1)
module = import_module(module_path)
return getattr(module, class_name)
except (ImportError, AttributeError) as e:
msg = "Could not import '%s' for API setting '%s'. %s: %s." % (val, setting_name, e.__class__.__name__, e)
raise ImportError(msg)

from django.utils.module_loading import import_string

def import_string(dotted_path):
"""
Import a dotted module path and return the attribute/class designated by the
last name in the path. Raise ImportError if the import failed.
"""
try:
module_path, class_name = dotted_path.rsplit('.', 1)
except ValueError:
msg = "%s doesn't look like a module path" % dotted_path
six.reraise(ImportError, ImportError(msg), sys.exc_info()[2]) module = import_module(module_path) try:
return getattr(module, class_name)
except AttributeError:
msg = 'Module "%s" does not define a "%s" attribute/class' % (
module_path, class_name)
six.reraise(ImportError, ImportError(msg), sys.exc_info()[2])

2. 面向对象里的 __setitem__ __getitem__

class Foo(object):
def __getitem__(self, item):
return "123"
def __setitem__(self, key, value):
self.__dict__[key] = value
def __delitem__(self, key):
print('----')
self.__dict__.pop(key) obj = Foo()
print(obj['k1']) # __getitem__ obj['k1'] = 666 # __setitem__ del obj['k1'] # __delitem__ Django 里面操作session:
request.session['xxx'] = xxxxx

3. 程序结合配置文件以及工厂模式

class MemSession(object):
def __getitem__(self, item):
return "123" def __setitem__(self, key, value):
self.__dict__[key] = value def __delitem__(self, key):
print('----')
self.__dict__.pop(key) class RedisSession(object):
def __getitem__(self, item):
return "123" def __setitem__(self, key, value):
self.__dict__[key] = value def __delitem__(self, key):
print('----')
self.__dict__.pop(key) class SessionFactory(object):
"""
工厂模式
settings.py : SESSION_ENGINE = "session_code.RedisSession"
""" @staticmethod
def get_session(self):
import settings
engine = settings.SESSION_ENGINE
import importlib
module_path,cls_name = engine.split('.',maxsplit=1)
md = importlib.import_module(module_path)
cls = getattr(md,cls_name)
return cls

  

Tornado基于内存和redis的session组件实现示例

app.py

from tornado import ioloop
from tornado.web import RequestHandler,Application
from session_code import SessionFactory settings = {
'template_path':'templates',
} class SessionHandler(object):
def initialize(self):
cls = SessionFactory.get_session()
self.session = cls(self) class IndexHandler(SessionHandler,RequestHandler): def get(self):
user = self.session['user']
if user:
self.write("首页欢迎你")
else:
self.redirect('/login') class LoginHandler(SessionHandler,RequestHandler): def get(self):
self.render('login.html',msg="")
def post(self, *args, **kwargs):
name = self.get_argument('name')
pwd = self.get_argument('pwd')
if 'alex'==name and '123' == pwd:
self.session['user'] = name
self.redirect('/index')
else:
self.render('login.html',msg="用户名或密码错误") application = Application([
(r"/index", IndexHandler,{},'alias_name1'),
(r"/login", LoginHandler,{},'alias_name2'),
],**settings) if __name__ == "__main__":
application.listen(8090)
ioloop.IOLoop.instance().start()

session_code.py

import json
import time
import hashlib
import settings def gen_random_str():
md5 = hashlib.md5()
md5.update(str(time.time()).encode('utf-8'))
return md5.hexdigest() class MemSession(object):
''' 静态字段在类里只保存一份(不管实例化多少次,用的都是同一份) '''
container = {} def __init__(self,handler):
self.handler = handler
self.session_id = settings.SESSION_ID
self.expires = settings.EXPIRES
self.initial() def initial(self):
client_random_str = self.handler.get_cookie(self.session_id)
if client_random_str and client_random_str in self.container:
self.random_str = client_random_str
else:
self.random_str = gen_random_str()
self.container[self.random_str] = {} # 给服务端设置session
expires = time.time() + self.expires
self.handler.set_cookie(self.session_id,self.random_str,expires=expires) # 给客户端设置cookie def __getitem__(self, item):
return self.container[self.random_str].get(item) def __setitem__(self, key, value):
self.container[self.random_str][key] = value
print(self.container) def __delitem__(self, key):
if key in self.container[self.random_str]:
del self.container[self.random_str][key] class RedisSession(object): def __init__(self,handler):
self.handler = handler
self.session_id = settings.SESSION_ID
self.expires = settings.EXPIRES
self.initial() def get_redis_conn(self):
import redis
conn = redis.Redis(host='8.8.8.8', port=6379)
return conn def initial(self):
self.redis_conn = self.get_redis_conn()
client_random_str = self.handler.get_cookie(self.session_id)
if client_random_str and self.redis_conn.exists(client_random_str):
self.random_str = client_random_str
else:
self.random_str = gen_random_str() expires = time.time() + self.expires
self.handler.set_cookie(self.session_id,self.random_str,expires=expires) # 给客户端设置cookie
self.redis_conn.expire(self.random_str,self.expires) # 给redis设置超时时间 def __getitem__(self, item):
# redis 返回的是byte类型,所以需要decode
data_str = self.redis_conn.hget(self.random_str,item)
if data_str:
return json.loads(data_str.decode('utf-8')) def __setitem__(self, key, value):
# val = {'type':type(value).__name__,'value':value}
self.redis_conn.hset(self.random_str,key,json.dumps(value)) def __delitem__(self, key):
self.redis_conn.hdel(self.random_str,key) class SessionFactory(object):
"""
工厂模式
"""
@staticmethod
def get_session():
import settings
engine = settings.SESSION_ENGINE
import importlib
module_path,cls_name = engine.split('.',maxsplit=1)
md = importlib.import_module(module_path)
cls = getattr(md,cls_name)
return cls

settings.py

SESSION_ENGINE = "session_code.RedisSession"
SESSION_ID = "__session_id__"
EXPIRES = 300

login.html

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body> <h1 class="c1">Login</h1>
<form action="" method="post">
<input type="text" name="name">
<input type="text" name="pwd">
<input type="submit" value="提交"> {{ msg }}
</form> </body>
</html>

  

自定义Tornado的session组件的更多相关文章

  1. Python web框架 Tornado(三)自定义session组件

    我们在学习Django框架的过程中,内部封装了session组件,以方便于我们使用进行验证.但是Tornado框架是没有session的,所以如果想使用session的话,就需要我们自己定制相对应的组 ...

  2. 第二百六十九节,Tornado框架-Session登录判断

    Tornado框架-Session登录判断 Session需要结合cookie来实现 Session的理解 1.用户登录系统时,服务器端获取系统当前时间,进行nd5加密,得到加密后的密串 2.将密串作 ...

  3. Web框架之Django_08 重要组件(form组件、cookie和session组件)

    摘要: form组件 cookie组件 session组件 一.form组件 form介绍我们之前在html页面中利用form表单向后端提交数据时候,都需要对用户的输入进行校验,比如校验用户是否输入正 ...

  4. Combo( 自定义下拉框) 组件

    本节课重点了解 EasyUI 中 Combo(自定义下拉框)组件的使用方法,这个组件依赖于ValidateBox(验证框)组件 一. 加载方式自定义下拉框不能通过标签的方式进行创建.<input ...

  5. 第二百一十二节,jQuery EasyUI,Combo(自定义下拉框)组件

    jQuery EasyUI,Combo(自定义下拉框)组件 学习要点: 1.加载方式 2.属性列表 3.事件列表 4.方法列表 本节课重点了解 EasyUI 中 Combo(自定义下拉框)组件的使用方 ...

  6. MongoDB实践-自定义ASP.NET Session Store

    Session由来 由于HTTP协议是无状态的,客户端与服务器端进行“请求-响应”操作后,建立的连接就释放了,服务器端根本不知道刚才是哪个客户端访问的.但是有些场景是需要知道客户端的状态的,最典型的就 ...

  7. Vue 使用use、prototype自定义自己的全局组件

    使用Vue.use()写一个自己的全局组件. 目录如下: 然后在Loading.vue里面定义自己的组件模板 <template> <div v-if="loadFlag& ...

  8. Flask框架【七】—session组件详解

    一.flask session简介 flask中session组件可分为内置的session组件还有第三方flask-session组件,内置的session组件缺点: 功能单一 session是保存 ...

  9. Django框架(十六)—— cookie和session组件

    目录 cookie和session组件 一.cookie 1.cookie的由来 2.什么是cookie 3.cookie的原理 4.cookie的覆盖 5.在浏览器中查看cookie 6.cooki ...

随机推荐

  1. [CF976E]Well played!

    题目描述 Recently Max has got himself into popular CCG "BrainStone". As "BrainStone" ...

  2. Nifi 模板

        Acqusition_and_Processing.xml Moving templates to own directory to make repo cleaner   CADF_Pars ...

  3. CSS垂直翻转与水平翻转

    /*水平翻转*/ .flipx { -moz-transform:scaleX(-1); -webkit-transform:scaleX(-1); -o-transform:scaleX(-1); ...

  4. centos7搭建ELK Cluster集群日志分析平台(二):Logstash

    续  centos7搭建ELK Cluster集群日志分析平台(一) 已经安装完Elasticsearch 5.4 集群. 安装Logstash步骤 . 安装Java 8 官方说明:需要安装Java ...

  5. 【洛谷P1637】三元上升子序列

    题目大意:给定一个长度为 N 的序列,求有多少个三元组满足 \(i<j<k,a_i<a_j<a_k\). 题解:这是一类二维偏序问题,与逆序对问题类似. 对于序列中每个点来说, ...

  6. [bzoj1717][Milk Patterns 产奶的模式]

    题目链接 思路 先求出后缀数组,并且求出LCP.二分一下长度len.check的时候就是看有没有连续的k个后缀的LCP大于len.也就是判断是不是有连续的k-1个height大于len. 代码 #in ...

  7. 跟我一起写一个hello-world react组件并发布到npm

    第一步:初始化我们的配置 $ mkdir react-hello-world $ cd react-hello-world/ $ npm init -y 修改我们的package.json文件 //p ...

  8. 3 字节的 UTF-8 序列的字节 2 无效

    由于目前写完了[消息队列]模块,想做个单元测试,所以就利用spring的import标签,将mq的配置文件加入了配置.结果出现了<3 字节的 UTF-8 序列的字节 2 无效>这个问题. ...

  9. Luogu P4009 汽车加油行驶问题

    题目链接 \(Click\) \(Here\) 分层图..好长时间没写差点要忘了\(hhhhh\),其实思路还是很明了的. 注意需要强制消费. #include <bits/stdc++.h&g ...

  10. lucene和solr的区别(六)

    Lucene是一个开放源代码的全文检索引擎工具包,即它不是一个完整的全文检索引擎,而是一个全文检索引擎的架构,提供了完整的查询引擎和索引引擎,部分文本分析引擎(英文与德文两种西方语言).Lucene的 ...