Python 基础(一):入门必备知识
10-30 阅读数 10万+
目录1 标识符2 关键字3 引号4 编码5 输入输出6 缩进7 多行8 注释9 数据类型10 运算符10.1 常用运算符10.2 运算符优先级 1 标识符 标识符是编程时使用的名字,用于给变量、函数、... 博文
BeanPostProcessor是Spring IOC容器给我们提供的一个扩展接口。接口声明如下:
-
public
interface BeanPostProcessor {
-
//bean初始化方法调用前被调用
-
Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException;
-
//bean初始化方法调用后被调用
-
Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException;
-
-
}
如上接口声明所示,BeanPostProcessor接口有两个回调方法。当一个BeanPostProcessor的实现类注册到Spring IOC容器后,对于该Spring IOC容器所创建的每个bean实例在初始化方法(如afterPropertiesSet和任意已声明的init方法)调用前,将会调用BeanPostProcessor中的postProcessBeforeInitialization方法,而在bean实例初始化方法调用完成后,则会调用BeanPostProcessor中的postProcessAfterInitialization方法,整个调用顺序可以简单示意如下:
--> Spring IOC容器实例化Bean
--> 调用BeanPostProcessor的postProcessBeforeInitialization方法
--> 调用bean实例的初始化方法
--> 调用BeanPostProcessor的postProcessAfterInitialization方法
可以看到,Spring容器通过BeanPostProcessor给了我们一个机会对Spring管理的bean进行再加工。比如:我们可以修改bean的属性,可以给bean生成一个动态代理实例等等。一些Spring AOP的底层处理也是通过实现BeanPostProcessor来执行代理包装逻辑的。
了解了BeanPostProcessor的相关知识后,下面我们来通过项目中的一个具体例子来体验一下它的神奇功效吧。
先介绍一下我们的项目背景吧:我们项目中经常会涉及AB 测试,这就会遇到同一套接口会存在两种不同实现。实验版本与对照版本需要在运行时同时存在。下面用一些简单的类来做一个示意:
-
public
class HelloService{
-
void sayHello();
-
void sayHi();
-
}
HelloService有以下两个版本的实现:
-
@Service
-
public
class HelloServiceImplV1 implements HelloService{
-
public void sayHello(){
-
System.out.println(
"Hello from V1");
-
}
-
public void sayHi(){
-
System.out.println(
"Hi from V1");
-
}
-
}
-
@Service
-
public
class HelloServiceImplV2 implements HelloService{
-
public void sayHello(){
-
System.out.println(
"Hello from V2");
-
}
-
public void sayHi(){
-
System.out.println(
"Hi from V2");
-
}
-
}
做AB测试的话,在使用BeanPostProcessor封装前,我们的调用代码大概是像下面这样子的:
-
@Controller
-
public
class HelloController{
-
@Autowird
-
private HelloServiceImplV1 helloServiceImplV1;
-
@Autowird
-
private HelloServiceImplV2 helloServiceImplV2;
-
-
public void sayHello(){
-
if(getHelloVersion()==
"A"){
-
helloServiceImplV1.sayHello();
-
}
else{
-
helloServiceImplV2.sayHello();
-
}
-
}
-
public void sayHi(){
-
if(getHiVersion()==
"A"){
-
helloServiceImplV1.sayHi();
-
}
else{
-
helloServiceImplV2.sayHi();
-
}
-
}
-
}
可以看到,这样的代码看起来十分不优雅,并且如果AB测试的功能点很多的话,那项目中就会充斥着大量的这种重复性分支判断,看到代码就想死有木有!!!维护代码也将会是个噩梦。比如某个功能点AB测试完毕,需要把全部功能切换到V2版本,V1版本不再需要维护,那么处理方式有两种:
怎么解决这个问题呢,我们先看代码,后文再给出解释:
-
@Target({ElementType.FIELD})
-
@Retention(RetentionPolicy.RUNTIME)
-
@Documented
-
@Component
-
public
@interface RoutingInjected{
-
}
-
@Target({ElementType.FIELD,ElementType.METHOD})
-
@Retention(RetentionPolicy.RUNTIME)
-
@Documented
-
@Component
-
public
@interface RoutingSwitch{
-
/**
-
* 在配置系统中开关的属性名称,应用系统将会实时读取配置系统中对应开关的值来决定是调用哪个版本
-
* @return
-
*/
-
String value()
default
"";
-
}
-
-
@RoutingSwitch(
"hello.switch")
-
public
class HelloService{
-
-
@RoutingSwitch(
"A")
-
void sayHello();
-
-
void sayHi();
-
}
-
@Controller
-
public
class HelloController{
-
-
@RoutingInjected
-
private HelloService helloService;
-
-
public void sayHello(){
-
this.helloService.sayHello();
-
}
-
-
public void sayHi(){
-
this.helloService.sayHi();
-
}
-
}
现在我们可以停下来对比一下封装前后调用代码了,是不是感觉改造后的代码优雅很多呢?那么这是怎么实现的呢,我们一起来揭开它的神秘面纱吧,请看代码:
-
@Component
-
public
class RoutingBeanPostProcessor implements BeanPostProcessor {
-
-
@Autowired
-
private ApplicationContext applicationContext;
-
-
@Override
-
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
-
return bean;
-
}
-
-
@Override
-
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
-
Class clazz = bean.getClass();
-
Field[] fields = clazz.getDeclaredFields();
-
for (Field f : fields) {
-
if (f.isAnnotationPresent(RoutingInjected.class)) {
-
if (!f.getType().isInterface()) {
-
throw
new BeanCreationException(
"RoutingInjected field must be declared as an interface:" + f.getName()
-
+
" @Class " + clazz.getName());
-
}
-
try {
-
this.handleRoutingInjected(f, bean, f.getType());
-
}
catch (IllegalAccessException e) {
-
throw
new BeanCreationException(
"Exception thrown when handleAutowiredRouting", e);
-
}
-
}
-
}
-
return bean;
-
}
-
-
private void handleRoutingInjected(Field field, Object bean, Class type) throws IllegalAccessException {
-
Map
candidates =
this.applicationContext.getBeansOfType(type);
-
field.setAccessible(
true);
-
if (candidates.size() ==
1) {
-
field.set(bean, candidates.values().iterator().next());
-
}
else
if (candidates.size() ==
2) {
-
Object proxy = RoutingBeanProxyFactory.createProxy(type, candidates);
-
field.set(bean, proxy);
-
}
else {
-
throw
new IllegalArgumentException(
"Find more than 2 beans for type: " + type);
-
}
-
}
-
}
-
-
public
class RoutingBeanProxyFactory {
-
-
public static Object createProxy(Class targetClass, Map
beans) {
-
ProxyFactory proxyFactory = new ProxyFactory();
-
proxyFactory.setInterfaces(targetClass);
-
proxyFactory.addAdvice(new VersionRoutingMethodInterceptor(targetClass, beans));
-
return proxyFactory.getProxy();
-
}
-
static
class VersionRoutingMethodInterceptor implements MethodInterceptor {
-
private String classSwitch;
-
private Object beanOfSwitchOn;
-
private Object beanOfSwitchOff;
-
-
public VersionRoutingMethodInterceptor(Class targetClass, Map
beans) {
-
String interfaceName = StringUtils.uncapitalize(targetClass.getSimpleName());
-
if(targetClass.isAnnotationPresent(RoutingSwitch.
class)){
-
this.classSwitch =((RoutingSwitch)targetClass.getAnnotation(RoutingSwitch.
class)).value();
-
}
-
this.beanOfSwitchOn = beans.
get(
this.buildBeanName(interfaceName,
true));
-
this.beanOfSwitchOff = beans.
get(
this.buildBeanName(interfaceName,
false));
-
}
-
-
private String buildBeanName(String interfaceName, boolean isSwitchOn) {
-
return interfaceName +
"Impl" + (isSwitchOn ?
"V2" :
"V1");
-
}
-
-
@Override
-
public Object invoke(MethodInvocation invocation) throws Throwable {
-
Method method = invocation.getMethod();
-
String switchName =
this.classSwitch;
-
if (method.isAnnotationPresent(RoutingSwitch.
class)) {
-
switchName = method.getAnnotation(RoutingSwitch.
class).value();
-
}
-
if (StringUtils.isBlank(switchName)) {
-
throw new IllegalStateException(
"RoutingSwitch's value is blank, method:" + method.getName());
-
}
-
return invocation.getMethod().invoke(getTargetBean(switchName), invocation.getArguments());
-
}
-
-
public Object getTargetBean(String switchName) {
-
boolean switchOn;
-
if (RoutingVersion.A.equals(switchName)) {
-
switchOn =
false;
-
}
else
if (RoutingVersion.B.equals(switchName)) {
-
switchOn =
true;
-
}
else {
-
switchOn = FunctionSwitch.isSwitchOpened(switchName);
-
}
-
return switchOn ? beanOfSwitchOn : beanOfSwitchOff;
-
}
-
}
-
}
-
我简要解释一下思路:
好了,BeanPostProcessor的介绍就到这里了。不知道看过后大家有没有得到一些启发呢?欢迎大家留言交流反馈。今天就到这里,我们下期再见吧 哈哈哈
10-30 阅读数 10万+
目录1 标识符2 关键字3 引号4 编码5 输入输出6 缩进7 多行8 注释9 数据类型10 运算符10.1 常用运算符10.2 运算符优先级 1 标识符 标识符是编程时使用的名字,用于给变量、函数、... 博文
01-19 阅读数 4万+
此博客仅为我业余记录文章所用,发布到此,仅供网友阅读参考,如有侵权,请通知我,我会删掉。
补充
有不少读者留言说本文章没有用,因为天气预报直接打开手机就可以收到了,为何要多此一举发送到邮箱呢!!!…
博文
11-04 阅读数 2万+
文章目录博客说明:基础知识 001.Hello,WorldPython2.7能够正常输出py2、py3Python3.7无法正常输出py2,版本不兼容 0... 博文
11-18 阅读数 7万+
11月8日,由中国信息通信研究院、中国通信标准化协会、中国互联网协会、可信区块链推进计划联合主办,科技行者协办的2019可信区块链峰会将在北京悠唐皇冠假日酒店开幕。
区块链技术被认为…
博文
11-18 阅读数 8921
不知觉已中码龄已突破五年,一路走来从起初铁憨憨到现在的十九线程序员,一路成长,虽然不能成为高工,但是也能挡下一面,从15年很火的android开始入坑,走过java、.Net、QT,目前仍处于... 博文
11-19 阅读数 11万+
作者 | 胡书敏
责编 | 刘静
出品 | CSDN(ID:CSDNnews)
本人目前在一家知名外企担任架构师,而且最近八年来,在多家外企和互联网公司担任Java技术面试官…
博文
11-20 阅读数 7万+
知乎高赞:中国有什么拿得出手的开源软件产品? 在知乎上,有个问题问“中国有什么拿得出手的开源软件产品(在 GitHub 等社区受欢迎度较好的)?” 事实上,还不少呢~ 本人于2019.7.6进行了较为... 博文
11-25 阅读数 8万+
文章目录一、数据库简介二、MySQL数据类型(5.5版本)三、Sql语句(1)Sql语句简介(2)数据定义语言DDLcreate,alter,drop(3)数据操纵语言DMLupdate,insert... 博文
11-21 阅读数 14万+
引言 王者荣耀大家都玩过吧,没玩过的也应该听说过,作为时下最火的手机MOBA游戏,咳咳,好像跑题了。我们今天的重点是爬取王者荣耀所有英雄的所有皮肤,而且仅仅使用20行Python代码即可完成。 准备工... 博文
11-21 阅读数 1万+
作者 | 许向武
责编 | 屠敏
出品 | CSDN 博客
前言
在 Python 进阶的过程中,相信很多同学应该大致上学习了很多 Python 的基础知识,也正在努力成长。在此…
博文
11-22 阅读数 835
欢迎关注文章系列 ,关注我 《提升能力,涨薪可待》 《面试知识,工作可待》 《实战演练,拒绝996》 欢迎关注我博客,原创技术文章第一时间推出 也欢迎关注公 众 号【Ccww笔记】,同时推出 如果此文... 博文
11-22 阅读数 1万+
今天咱们第一课,来讲讲大家一直很关注的数据中台。其实,数据中台也是企业数据管理的一部分,甚至可以说是很重要的一部分。
一、什么是中台?
这其实是一个老生常谈的概念了,中台,顾名思义,就是在起中间作…
博文
11-24 阅读数 4万+
我清晰的记得,刚买的macbook pro回到家,开机后第一件事情,就是上了淘宝网,花了500元钱,找了一个上门维修电脑的师傅,上门给我装了一个windows系统。。。。。。 表砍我。。。 当时买ma... 博文
11-25 阅读数 11万+
二哥,你好,我想知道一般程序猿都如何接私活,我也想接,能告诉我一些方法吗?
上面是一个读者“烦不烦”问我的一个问题。其实不止是“烦不烦”,还有很多读者问过我类似这样的问题。
我接的私活不算多,挣到…
博文
11-25 阅读数 7万+
今年正式步入了大四,离毕业也只剩半年多的时间,回想一下大学四年,感觉自己走了不少弯路,今天就来分享一下自己大学的学习经历,也希望其他人能不要走我走错的路。
(一)初进校园
刚进入大学的时候自己完全…
博文
02-02 阅读数 1681
目录
一、前言
二、redis基础知识
2.1 从“处理器-缓存-内存”到“后台-redis-数据库”
2.2 不使用缓存与使用缓存(读操作+写操作)
2.3 redis典型问题:缓存穿透…
博文
12-08 阅读数 9万+
本文十天后设置为粉丝可见,喜欢的提前关注
不要白嫖请点赞
不要白嫖请点赞
不要白嫖请点赞
文中提到的书我都有电子版,可以评论邮箱发给你。
文中提到的书我都有电子版,可以评论邮箱发给你。
文…
博文
02-03 阅读数 1214
第8章 IO库 部分IO库设施:
istream:输入流类型,提供输入操作。
ostream:输出流类型,提供输出操作。
cin:istream对象,从标准输入读取数据。
cout:ostream对…
博文
01-07 阅读数 5万+
每天都会收到很多读者的私信,问我:“二哥,有什么推荐的学习网站吗?最近很浮躁,手头的一些网站都看烦了,想看看二哥这里有什么新鲜货。” 今天一早做了个恶梦,梦到被老板辞退了。虽然说在我们公司,只有我辞退... 博文
01-08 阅读数 1万+
软件工程师花费大量时间通过练习leet code问题和完善简历来获得更好的面试通过可能。一旦他们最终被谷歌、亚马逊或其他公司录用,他们可能会发现:过去用来得到这份工作的技能与他们日常工作中需要的技能并... 博文
01-08 阅读数 4万+
依稀记得,毕业那天,我们导员发给我毕业证的时候对我说“你可是咱们系的风云人物啊”,哎呀,别提当时多开心啦????,嗯,我们导员是所有导员中最帅的一个,真的???? 不过,导员说的是实话,很多人都叫我大... 博文
01-09 阅读数 1万+
作者 | 胡巍巍
出品 | CSDN(ID:CSDNnews)
世界500强中,30%的掌舵人,都是印度人。
是的,你没看错。这是近日《哈佛商业评论》的研究结果。
其中又以微软CEO萨提亚·纳…
博文
02-03 阅读数 1108
栈
在邂逅了完线性结构的数组和队列后, 我们便偶遇了栈这个东东, 他到底是个啥?
就让我们慢慢揭开它的神秘面纱吧~~~
需求介绍
栈的介绍
栈的英文为(stack)
栈是一个先入后出(FILO…
博文
01-14 阅读数 1万+
前奏: 今天2B哥和大家分享一位前几天面试的一位应聘者,工作4年26岁,统招本科。 以下就是他的简历和面试情况。
基本情况:
专业技能:
1、 熟悉Sping了解SpringMVC、S…
博文
01-17 阅读数 1万+
专栏 | 九章算法 网址 | www.jiuzhang.com/?utm_source=sc-csdn-fks
HelloGitHub
star:19k
Python,Java,PHP,C++,go…
博文
01-21 阅读数 5662
目录
前言:
禅道
Jenkins
sonarqube
4.showdoc
5.swgger
6.分布式配置中心apollo
8.项目开发…
博文
01-21 阅读数 7926
CPU对每个程序员来说,是个既熟悉又陌生的东西? 如果你只知道CPU是中央处理器的话,那可能对你并没有什么用,那么作为程序员的我们,必须要搞懂的就是CPU这家伙是如何运行的,尤其要搞懂它里面的寄存器是... 博文
02-03 阅读数 720
提示1:锐米所有 LoRa 产品严格遵循国标标准的 LoRaWAN 协议。 提示2:您可以免费复制,修改和商用本项目,请注明锐米原创。 提示3:如果您有其他 LoRa 需求或建议,欢迎联系锐米 sup... 博文
01-24 阅读数 5053
请允许我用22种编程语言,祝大家新年快乐 C语言:printf(“祝大家新年快乐”); C++ : cout<<“祝大家新年快乐”; OC: NSLog(@“祝大家新年快乐”) Q... 博文
01-27 阅读数 2万+
很遗憾,这个春节注定是刻骨铭心的,新型冠状病毒让每个人的神经都是紧绷的。那些处在武汉的白衣天使们,尤其值得我们的尊敬。而我们这些窝在家里的程序员,能不外出就不外出,就是对社会做出的最大的贡献。
有些…
博文
02-01 阅读数 4716
man: Manual 意思是手册,可以用这个命令查询其他命令的用法。 pwd:Print working directory 意思是密码。 su:Swith user 切换用户,切换到ro... 博文
01-27 阅读数 2万+
今天,群里白垩老师问如何用python画武汉肺炎疫情地图。白垩老师是研究海洋生态与地球生物的学者,国家重点实验室成员,于不惑之年学习python,实为我等学习楷模。先前我并没有关注武汉肺炎的具体数据,... 博文
01-27 阅读数 4414
受疫情影响,道路几乎都被封闭,短时间回京是没有希望了。整天待在家里无所事事,倒不如早些开工。今天初三,早上起来便开始着手工作。写这篇文章只是聊聊对最近发生的一些事情的看法,锻炼一下思辨能力。 学习思辨... 博文
01-30 阅读数 1万+
返回json示例 { "errcode":0,//0标识接口正常 "data":{ "date":"2020-01-30 07:47:23",//实时更新时间 ... 博文
01-30 阅读数 8325
python是当下很热门的语言,我在入门python时花的时间特别长,一方面是自学,另一方面是这个东西对新人很不友好。因此,我写下这篇文章,希望能对想从零开始学python的同学有一些帮助。 首先给大... 博文
01-30 阅读数 6029
地图绘制 数据源 腾讯疫情实时追踪 网站结构比较简单,可以直接获取json格式的数据
抓取每个城市的当前感染数据
导入相关模块
import time
import json
import re…
博文
没有更多推荐了,返回首页
你好! 这是你第一次使用 Markdown编辑器 所展示的欢迎页。如果你想学习如何使用Markdown编辑器, 可以仔细阅读这篇文章,了解一下Markdown的基本语法知识。
我们对Markdown编辑器进行了一些功能拓展与语法支持,除了标准的Markdown编辑器功能,我们增加了如下几点新功能,帮助你用它写博客:
撤销:Ctrl/Command + Z
重做:Ctrl/Command + Y
加粗:Ctrl/Command + B
斜体:Ctrl/Command + I
标题:Ctrl/Command + Shift + H
无序列表:Ctrl/Command + Shift + U
有序列表:Ctrl/Command + Shift + O
检查列表:Ctrl/Command + Shift + C
插入代码:Ctrl/Command + Shift + K
插入链接:Ctrl/Command + Shift + L
插入图片:Ctrl/Command + Shift + G
查找:Ctrl/Command + F
替换:Ctrl/Command + G
直接输入1次#,并按下space后,将生成1级标题。
输入2次#,并按下space后,将生成2级标题。
以此类推,我们支持6级标题。有助于使用TOC
语法后生成一个完美的目录。
强调文本 强调文本
加粗文本 加粗文本
标记文本
删除文本
引用文本
H2O is是液体。
210 运算结果是 1024.
链接: link.
图片:
带尺寸的图片:
居中的图片:
居中并且带尺寸的图片:
当然,我们为了让用户更加便捷,我们增加了图片拖拽功能。
去博客设置页面,选择一款你喜欢的代码片高亮样式,下面展示同样高亮的 代码片
.
// An highlighted block
var foo = 'bar';
一个简单的表格是这么创建的:
项目 | Value |
---|---|
电脑 | $1600 |
手机 | $12 |
导管 | $1 |
使用:---------:
居中
使用:----------
居左
使用----------:
居右
第一列 | 第二列 | 第三列 |
---|---|---|
第一列文本居中 | 第二列文本居右 | 第三列文本居左 |
SmartyPants将ASCII标点字符转换为“智能”印刷标点HTML实体。例如:
TYPE | ASCII | HTML |
---|---|---|
Single backticks | 'Isn't this fun?' |
‘Isn’t this fun?’ |
Quotes | "Isn't this fun?" |
“Isn’t this fun?” |
Dashes | -- is en-dash, --- is em-dash |
– is en-dash, — is em-dash |
一个具有注脚的文本。2
Markdown将文本转换为 HTML。
您可以使用渲染LaTeX数学表达式 KaTeX:
Gamma公式展示 Γ ( n ) = ( n − 1 ) ! ∀ n ∈ N \Gamma(n) = (n-1)!\quad\forall n\in\mathbb N Γ(n)=(n−1)!∀n∈N 是通过欧拉积分
Γ ( z ) = ∫ 0 ∞ t z − 1 e − t d t . \Gamma(z) = \int_0^\infty t^{z-1}e^{-t}dt\,. Γ(z)=∫0∞tz−1e−tdt.
你可以找到更多关于的信息 LaTeX 数学表达式here.
可以使用UML图表进行渲染。 Mermaid. 例如下面产生的一个序列图:
这将产生一个流程图。:
我们依旧会支持flowchart的流程图:
如果你想尝试使用此编辑器, 你可以在此篇文章任意编辑。当你完成了一篇文章的写作, 在上方工具栏找到 文章导出 ,生成一个.md文件或者.html文件进行本地保存。
如果你想加载一篇你写过的.md文件,在上方工具栏可以选择导入功能进行对应扩展名的文件导入,
继续你的创作。
mermaid语法说明 ↩︎
注脚的解释 ↩︎