之前有一篇博文写到关于购物车的业务逻辑,分别运用cookie和redis存储未登录和登录用户的购物车数据,虽然已经很好的完成了业务逻辑,但是会发现代码的冗余很严重,也不够具有python特色,今天就让我来用python特色,高逼格的来改写一下购物车逻辑把。

回看之前代码,我们会发现,会经常取出购物车数据,然后也会修改数据,我们能否对读和写的操作进行封装,第二,cookie中和redis中存储的我们对数据存储的格式不一样,能否考虑将两者存储为一种格式,第三,考虑数据通过网络传输,我们能否尽量存储少的数据。

之前我们在redis中是将商品数量和商品勾选状态分别存为hash和set,而在cookie中则是存成一个大字典,然后将大字典转化为字符串,在redis中,我们也可以采用相同的存储方式,而相对于字典中嵌套字典,我们的小字典中只有两组数据,我们可以用一个列表直接存储商品数量和勾选状态,记住对应的下标即可,这样我们就不用花空间去存储冗余的key值。确定好存储方式,我们需要想一下怎么去进行封装。在python是面向对象语言,同时具有多继承的特色,所以我们可以用拓展类的方式去封装读写操作,在对应的视图类中采取多继承的方式,引入读写的方法。

class CartMixin(object):
"""购物车扩展类"""
str_to_dict_class = (str.encode, base64.b64decode, pickle.loads)
dict_to_dict_class = (pickle.dumps, base64.b64encode, bytes.decode) def conversion_and_encryption(self, key):
     # 将数据转化为需要格式的方法
if isinstance(key, dict):
for func in self.dict_to_dict_class:
key = func(key)
else:
for func in self.str_to_dict_class:
key = func(key)
return key def get_cart_dict(self, request):
     # 获取购物车的方法
try:
user = request.user
except Exception:
user = None
if user and user.is_authenticated:
return self.get_cart_from_redis(request)
else:
return self.get_cart_from_cookie(request) def write_cart_dict(self, cart_dict, request, response):
     # 写入购物车数据的方法
try:
user = request.user
except Exception:
user = None
if user and user.is_authenticated:
self.write_cart_to_redis(cart_dict, user)
return response
else:
response = self.write_cart_to_cookie(cart_dict, response)
return response def get_cart_from_redis(self,request):
     # 从redis中获取购物车
redis_conn = get_redis_connection('cart')
redis_cart = redis_conn.get('cart:%s'%request.user.id)
if not redis_cart:
return {}
# redis_cart = pickle.loads(base64.b64decode(redis_cart))
cart_dict = self.conversion_and_encryption(redis_cart.decode())
return cart_dict def get_cart_from_cookie(self,request):
     # 从cookie中获取购物车的方法
cart_dict = request.COOKIES.get('cart')
if cart_dict:
# cart_dict = pickle.loads(base64.b64decode(cart_dict.encode()))
cart_dict = self.conversion_and_encryption(cart_dict)
else:
cart_dict = {}
return cart_dict def write_cart_to_redis(self, cart_dict, user):
     # 写入购物车数据到redis的方法
redis_conn = get_redis_connection('cart')
# cart_dict = base64.b64encode(pickle.dumps(cart_dict)).decode()
cart_dict = self.conversion_and_encryption(cart_dict)
redis_conn.set('cart:%s'%user.id, cart_dict) def write_cart_to_cookie(self, cart_dict, response):
     # 写入购物车数据到cookie的方法
# cart_cookie = base64.b64encode(pickle.dumps(cart_dict)).decode()
cart_cookie = self.conversion_and_encryption(cart_dict)
response.set_cookie('cart', cart_cookie, max_age=constants.CART_COOKIE_EXPIRES)
return response

为了保证数据的可靠性和安全性,我们会对数据进行一些处理在进行存储,考虑到有多个地方可能用到数据处理的操作,我们将数据处理的操作也封装成对应的方法,同时,我们也可能用不同的方式处理数据,所以将处理数据的方法放到一个列表中,用类属性方法进行存储,需要的更改的时候就只需要更改对应的类属性即可。这样我们就完成了读写操作的封装,在需要进行读写操作的地方调用对应的方法即可。这样能减少代码的冗余。

而在对应的业务逻辑中,我们应该只需要考虑业务方面的需要,无需考虑用户的登录状态,同时,购物车数据的读取和写入也是与我们的业务逻辑无关。关于登录状态的判断,已经封装到扩展类当中了,而对于读写,我们可以不必在视图的业务逻辑中进行操作,在drf框架中在请求进来之前会进行一个初始化操作,而在响应返回,传到前端之前,也会进行最后的处理,我们可以在视图类中,重写这两个方法,继承父类的操作,在加上我们需要进行的读写操作即可,这样就实现了读写操作和业务逻辑的分离。

class CartView(CartMixin, APIView):
'''购物车'''
permission_classes = [IsAuthenticated] def perform_authentication(self, request):
"""
重写父类的用户验证方法,不在进入视图前就检查JWT
"""
pass def initial(self, request, *args, **kwargs):
super().initial(request, *args, **kwargs)
self.cart_dict = self.get_cart_dict(request) def finalize_response(self, request, response, *args, **kwargs):
response = super().finalize_response(request, response, *args, **kwargs)
self.response = self.write_cart_dict(self.cart_dict, request, response)
return self.response

至此,代码改写完成。

新人写博客锻炼自己,想一起交流的可以加我的qq595395786

最新文章

  1. 【JavaScript】--重点解析之跨域请求
  2. Ajax的使用
  3. 开发板支持wifi
  4. Code Simplicity–The Science of Software Development 书摘
  5. Visual Studio 2010配置Opencv2.4.9
  6. 1从零开始学习Xamarin.iOS安装篇
  7. Maximum Product Subarray
  8. Linux异步IO【转】
  9. POJ 3468 A Simple Problem with Integers (伸展树区间更新求和操作 , 模板)
  10. C#中Action和Func的使用
  11. LCD 和 LED 的区别?
  12. 【转】国内用户如何加快App Store的访问速度
  13. iOS 项目中将 http 改成 https 后需要改动的地方(密钥验证)
  14. strus2与spring3 mvc的差别
  15. Vagrant常用命令
  16. Git 入门 ---- Git 与 SVN 区别
  17. C语言_了解下结构体指针
  18. mybatis自定义代码生成器(Generator)——自动生成model&dao代码
  19. event、fly.js、购物车特效
  20. EntityFramework优化:第一次启动优化

热门文章

  1. NetCore AutoMapper的封装
  2. Java核心技术中的程序片段
  3. Dijkstra算法堆优化
  4. ForkJoinPool分支/合并框架工程使用的工作窃取
  5. Spring集成Shiro使用小结
  6. Gin框架 - 自定义错误处理
  7. linux初学者-磁盘配额篇
  8. 学习LayUI时自研的表单参数校验框架
  9. Initialization failed for 'https://start.spring.io' Please check URL
  10. Ubuntu下python安装mysqldb(驱动)