strerror() 和perror()函数
阅读目录
楔子
我在之前的文章《flask源码解析之上下文》中对flask上下文流程进行了详细的说明,但是在学习的过程中我一直在思考flask上下文中为什么要使用栈完成对请求上下文和应用上下文的入栈和出栈操作,而且栈所维护的无非不就是一个列表,我直接用一个列表去存储请求上下文和应用上下文不可以吗?或者说我用一个变量、字典其他任何可存储数据的数据类型不行吗?对于这个问题的解答,是我在理解离线脚本和 flask多app应用中才理解flask上下文中使用栈的精髓。对于为什么使用栈进行存储上下文,请耐心看我之前对离线脚本和flask多app应用的铺垫。
离线脚本
在项目的实际应用中,我们需要使用离线脚本完成不能作为后台功能的操作,例如:
1. 每天凌晨对还款用户进行短信提醒
2. 每天凌晨对数据库进行特定操作
3. 在项目开发完成交付他人进行测试之前,使用一个脚本完成对数据库的初始化操作
对于以上的所要实现的功能,他们都不能作为后台代码开发的一部分,只能使用一个离线的脚本完成上述操作。我们以需求2为例编写一个离线脚本:
from sansa import db
from sansa.models import Users
db.session.add(Users(name="大萨达所"))
db.session.commit()
在项目未启动情况下,运行此脚本抛出以下错误:
"D:\Program Files\Python36\python.exe" D:/Demo/s8/demo/sansa/数据插入的离线脚本.py
Traceback (most recent call last):
File "D:\Program Files\Python36\lib\site-packages\sqlalchemy\util\_collections.py", line 999, in __call__
return self.registry[key]
KeyError: <greenlet.greenlet object at 0x0000000003723930> During handling of the above exception, another exception occurred: Traceback (most recent call last):
File "D:/Demo/s8/demo/sansa/数据插入的离线脚本.py", line 3, in <module>
db.session.add(Users(name="大萨达所"))
File "D:\Program Files\Python36\lib\site-packages\sqlalchemy\orm\scoping.py", line 158, in do
return getattr(self.registry(), name)(*args, **kwargs)
File "D:\Program Files\Python36\lib\site-packages\sqlalchemy\util\_collections.py", line 1001, in __call__
return self.registry.setdefault(key, self.createfunc())
File "D:\Program Files\Python36\lib\site-packages\sqlalchemy\orm\session.py", line 2950, in __call__
return self.class_(**local_kw)
File "D:\Program Files\Python36\lib\site-packages\flask_sqlalchemy\__init__.py", line 141, in __init__
self.app = app = db.get_app()
File "D:\Program Files\Python36\lib\site-packages\flask_sqlalchemy\__init__.py", line 912, in get_app
'No application found. Either work inside a view function or push'
RuntimeError: No application found. Either work inside a view function or push an application context. See http://flask-sqlalchemy.pocoo.org/contexts/. Process finished with exit code 1
提示错误的原因是没有找到应用上下文对象的入栈操作,出现此问题的原因是:我们只是导入了db,但是项目原项目没有启动,更没有请求到来一说,我们执行的此离线脚本从头到尾就没有出现应用上下文,那么就更不会有应用上下文的入栈和出栈操作,所以db找到不到应用上下文,找不到应用上下文就无法导入连接数据库的配置信息。现在既然我们知道了出现问题的原因,解决此问题的方法就是我们手动执行应用上下文的创建、入栈和出栈操作,即要执行以下步骤:
1. app_ctx = self.app.app_context()
2. app_ctx.push()
3. app_ctx.pop(exc)
因此,编写的离线脚本为:
from sansa import db,create_app
from sansa.models import Users # 获取到生成app
app = create_app().app
# 创建app_ctx
app_ctx = app.app_context()
# app_ctx 入栈
app_ctx.push()
# 对数据库进行操作
db.session.add(Users(name="大萨达所"))
# 提交
db.session.commit()
app_ctx.pop()
由于 app_ctx 有 __enter__ 和 __exit__ 方法:
def __enter__(self):
self.push()
return self
def __exit__(self, exc_type, exc_value, tb):
self.pop(exc_value) if BROKEN_PYPY_CTXMGR_EXIT and exc_type is not None:
reraise(exc_type, exc_value, tb)
根据以上方法,离线脚本还可以这样写:
from sansa import db,create_app
from sansa.models import Users
app = create_app().app with app.app_context():
# 对数据库进行操作
db.session.add(Users(name="大萨达所"))
# 提交
db.session.commit()
# 释放当前的连接
db.session.remove()
这样,此脚本可实现在项目离线的情况下,按照指定的时间点执行该离线脚本完成对数据库的操作。
flask的多app应用
博主有在《flask源码解析之DispatcherMiddleware》一文中对flask多app应用的使用和源码流程进行过详细阐述,flask多app所实现的功能与蓝图相同,就是完成路由分发的功能。例如:
from werkzeug.wsgi import DispatcherMiddleware
from werkzeug.serving import run_simple
from flask import Flask, current_app,request app1 = Flask('app01') app2 = Flask('app02') @app1.route('/index/')
def index():
return "app01" @app2.route('/index2/')
def index2():
"""
存在的问题:
def __call__(self, environ, start_response):
# 获取当前请求的URL,script == '/index/'
script = environ.get('PATH_INFO', '')
path_info = ''
# '/' 在 '/index/' 中
while '/' in script:
# self.mounts == {'/sec': app2,}, '/' 不在 self.mounts 中
if script in self.mounts:
app = self.mounts[script]
break
script, last_item = script.rsplit('/', 1)
path_info = '/%s%s' % (last_item, path_info)
else:
app = self.mounts.get(script, self.app)
original_script_name = environ.get('SCRIPT_NAME', '')
environ['SCRIPT_NAME'] = original_script_name + script
environ['PATH_INFO'] = path_info # 这里得到的知识不包含前缀的url,那这样不是就丢失了吗
return app(environ, start_response)
:return:
"""
print(request.full_path) # 得到的是 /index2/? 且没有request.path_info
return "app2" # http://www.oldboyedu.com/index
# http://www.oldboyedu.com/sec/index2
dm = DispatcherMiddleware(app1, {
'/sec': app2,
}) if __name__ == "__main__":
run_simple('localhost', 5000, dm)
注意事项:
在DispatcherMiddleware类的实例参数中,第一个参数不能是一个字典,即第一个参数只能是一个app实例,即没有前缀的。但是后面的APP实例都可以是有前缀的app,存放至字典中
源码分析
请求每次进来执行DispatcherMiddleware的__call__方法,实现分发的原理是:获取本次请求的url,对url进行分割,直到分割出来的部分url(前缀)在保存了分发路由关系字典的self.mounts中时,分割结束,根据前缀找到要执行的函数,并做后续的操作。
为什么用栈
在 执行离线脚本 + 多app应用下,执行如下脚本程序:
from sansa import db,create_app
from sansa.models import Users
app1 = create_app1().app
app2 = create_app2().app with app1.app_context():
# 对数据库进行操作
db.session.add(Users(name="大萨达所"))
# 提交
db.session.commit()
# 释放当前的连接
db.session.remove()
with app2.app_context():
db.session.add(Users(name="xxx"))
# 提交
db.session.commit()
# 释放当前的连接
db.session.remove()
对于上述离线脚本程序,入栈顺序为: app1.app_context() ---> app2.app_context() ,由于执行当前脚本程序只开了一个线程,因此在两个应用上下文入栈的时候会存放至同一个线程id所维护的栈中;出栈的时候按照当前线程id去匹配此线程id所对维护的栈,每一个应用上下文在出栈的时候都会获取栈顶元素,即出栈顺序为: app2.app_context() ---> app1.app_context() ,完成了后进先出的栈特点,因此需要栈的数据结构。
https://www.cnblogs.com/liuyinzhou/p/9779518.html#_label0
strerror() 和perror()函数的更多相关文章
- strerror和perror函数详解
/*#include <string.h> char *strerror(int errnum); 它返回errnum的值所对应的错误提示信息,例如errnum等于12的话,它就会返回&q ...
- Linux/Unix C编程之的perror函数,strerror函数,errno
#include <stdio.h> // void perror(const char *msg); #include <string.h> // char *strerro ...
- 文件IO之——阻塞和非阻塞及perror函数
读常规文件是不会阻塞的,不管读多少字节,read一定会在有限的时间内返回.从终端设备或网络读则不一定,如果从终端输入的数据没有换行符,调用read读终端设备就会阻塞,如果网络上没有接收到数据包,调用r ...
- c中perror函数
写代码这么久,竟然很少用到perror函数,忘记了其强大的功能. 所在头文件: #include<stdio.h> 函数定义: void perror(const char *str); ...
- perror()函数的使用
perror()函数的功能是打印一个系统错误信息. perror()函数在Linux系统中属于库函数,在头文件中有如下定义: #include <stdio.h> ...
- C/C++错误分析errno,perror,strerror和GetLastError()函数返回的错误代码的意义
在C语言编译中,经常会出现一些系统的错误,这些错误如果在编译的时候不能很好的“预见”,会使系统“崩溃”,常见的捕获错误函数有: errno #include<errno.h> 这个变量是程 ...
- 【转载++】C/C++错误分析errno,perror,strerror和GetLastError()函数返回的错误代码的意义
本文是上一篇“fopen返回0(空指针NULL)且GetLastError是0”的侧面回应.听赶来多么地正确和不容置疑,返回NULL时调用GetLastError来看看报错啊,但当时却返回了0,大家都 ...
- vim命令以及gcc编译器的常用cmd
Gcc常用命令: -c 仅对源文件进行编译,不链接生成可执行文件.常用于查错和只生成目标文件. -o 经过gcc处理过后的结果保存在-o后面的文件中,可以是多种文件 ...
- 信号处理篇alarm ferror kill mkfifo pause pclose perror pipe popen sigaction sigaddset sigdelset sigemptyset signal sleep strerror
alarm(设置信号传送闹钟) 相关函数 signal,sleep 表头文件 #include<unistd.h> 定义函数 unsigned int alarm(unsigned int ...
随机推荐
- 微信小程序之分享,动态添加分享数据
1.效果: 2..js代码: page({ /** * 用户点击分享按钮或右上角分享 */ onShareAppMessage: function (res) { var that = this; r ...
- windows安装tensorflow GPU
一.安装Anaconda Anaconda是Python发行包,包含了很多Python科学计算库.它是比直接安装Python更好的选择. 二.安装Tensorflow 如果安装了tensorflow, ...
- 深入浅出Android开发之Surface介绍
一 目的 本节的目的就是为了讲清楚Android中的Surface系统,大家耳熟能详的SurfaceFlinger到底是个什么东西,它的工作流程又是怎样的.当然,鉴于SurfaceFlinger的复杂 ...
- python学习笔记013——推导式
1 推导式简介 推导式comprehensions(又称解析式),是Python的一种独有特性. 推导式是可以从一个数据序列构建另一个新的数据序列的结构体. 推导式有三种形式: 1)列表推导式 (li ...
- linux下core dump
1.前言 一直在从事linux下后台开发,经常与core文件打交道.还记得刚开始从事linux下开发时,程序突然崩溃了,也没有任何日志.我不知所措,同事叫我看看core,我却问什么是core,怎么看. ...
- malloc的内存分配原理
0 堆内存的在计算机内存中的形式 根据<The C Programming language>推测得到堆内存,图中的Heap区域即为堆内存块(Heap区域的数目不代表计算机堆内存的真实数目 ...
- Intelligence System (hdu 3072 强联通缩点+贪心)
Intelligence System Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Othe ...
- Android利用Fiddler进行网络数据抓包【怎么跟踪微信请求】
主要介绍Android及IPhone手机上如何利用Fiddler进行网络数据抓包,比如我们想抓某个应用(微博.微信.墨迹天气)的网络通信请求就可以利用这个方法. Mac 下请使用 Charles 代替 ...
- Model層資料驗證
概述 上节我们学习了Model的数据在界面之间的传递,但是很多时候,我们在数据传递的时候为了确保数据的有效性,不得不给Model的相关属性做基本的数据验证. 本节我们就学习如何使用 System.Co ...
- C 多级指针
C多级指针 *p -----> &p1 *(*p) ----->*(&p1) = &p ...