1.探索TP6验证场景的源码only、remove、分析append规则
2.记一次源码追踪分析,源码从Java到JNI,分析再到JVM的源码C++:fileChannel.map()为什么快;源码分析map方法,put方法
3.laravelåthinkphpçåºå«ï¼
4.宝塔安装ThinkPHP6 详细过程
5.超详细通达信主图指标安装教程,分析auto.js脚本源码意简言赅过程简单!源码有惊喜源码
6.2024路由器推荐,分析路由器怎么选不踩雷,源码路由器哪个牌子好用,分析6款实用路由器排行
探索TP6验证场景的源码only、remove、分析append规则
在探讨 TP6 验证场景的源码 only、remove、分析append 规则时,源码我们首先需要关注源码,以便准确理解其运作机制。在 Validate.php 文件中,这些方法的执行顺序和交互逻辑为我们提供了关键线索。让我们逐步解析这些规则及其相互作用,以便在实际应用中灵活运用。
首先,明确的是,only、remove、append 方法之间不存在特定的执行顺序要求。它们各自完成的任务如下:
1. **only**: 指定仅需验证的规则。这一步是筛选过程,从所有可能的验证规则中挑选出特定的几个用于后续验证。
2. **remove**: 从筛选后的规则中移除特定的规则。这一步允许在 only 筛选之后,进一步排除不需要的验证规则。
3. **append**: 向筛选后的规则集中添加新的规则。这意味着在验证流程中引入额外的验证标准。
接下来,无纸化审批 源码根据实际操作和源码分析,我们发现以下几个关键规则:
1. **remove 和 append 需要 only 的配合**:只有在 only 指定了验证字段后,remove 和 append 才能对该字段执行操作。如果缺少 only 的声明,remove 和 append 将直接跳过对应的验证逻辑。
2. **append 需要明确规则内容**:在使用 append 时,需要提供具体的规则描述,例如 `append('date', 'dateFormat:Y-m-d')`。而 remove 则主要通过规则名称进行操作,如 `remove('date', 'dateFormat')`。
3. **remove 支持 true 参数**:remove 方法允许传入 true 参数来移除该字段的所有验证规则。然而,这种方法不能与 append 同时用于同一规则,因为两者的目的与逻辑不兼容。
4. **remove 和 append 的交互规则**:当 remove 和 append 作用于同一规则时,只会移除原有的规则,不会添加新的规则。这一规则限制了在某些特定场景下的灵活应用。
最后,我们注意到一个在源码中影响验证逻辑的关键细节:在执行 append 后,系统会尝试移除先前添加的条件,这导致了在某些情况下,验证逻辑无法按预期执行。为了解决这一问题,我们需要对源码进行适当的修改:
- **注释掉相关代码行**:在源码中,找到并注释掉那些用于移除条件的代码,避免重复添加和移除条件。
- **调整验证流程**:在确保注释行不影响整体逻辑的前提下,调整验证流程以确保 append 的条件不会被意外移除。
通过以上解析和调整,我们可以更好地理解和利用 TP6 验证器中的 only、remove、append 方法,以适应复杂多变的沸点指标源码验证场景需求。
记一次源码追踪分析,从Java到JNI,再到JVM的C++:fileChannel.map()为什么快;源码分析map方法,put方法
前言
在系统IO相关的系统调用有read/write,mmap,sendfile等这些。
其中read/write是普通的读写,每次都需要将buffer从用户空间拷贝到内核空间;
而mmap使用的是内存映射,会将磁盘文件对应的页映射(拷贝)到内核空间的page cache,并记录到用户进程的页表中,使得用户空间也可以像操作用户空间一样操作该文件的映射,最后再由操作系统来讲该映射(脏页)回写到磁盘;
sendfile则使用的是零拷贝技术,在mmap的基础上,当发送数据的时候只拷贝fd和offset等元数据信息,而将数据主体直接拷贝至protocol buffer,实现了内核数据零冗余的零拷贝技术
本文地址:/post//
问题/目的问题1Java中哪些API使用到了mmap问题2怎么知道该API使用到了mmap,如何追踪程序的系统调用目的1源码中分析验证,从Java到JNI,再到C++:fileChannel.map()使用的是系统调用mmap目的2源码验证分析:调用mmapedByteBuffer.put(Byte[])时JVM在搞些什么?mmap比普通的read/write快在哪?揭晓答案1mmap在Java NIO中的体现/使用看一个例子
// 1GBpublic static final int _GB = 1**;File file = new File("filename");FileChannel fileChannel = new RandomAccessFile(file, "rw").getChannel();MappedByteBuffer mmapedByteBuffer = fileChannel.map(FileChannel.MapMode.READ_WRITE, 0, _GB);for (int i = 0; i < _GB; i++) { count++;mmapedByteBuffer.put((byte)0);}其中fileChannel.map()底层使用的就是系统调用mmap,函数签名为: public abstract MappedByteBuffer map(MapMode mode,long position, long size)throws IOException
答案2程序执行的系统调用追踪/** * @author Tptogiar * @description * @date /5/ - : */public class TestMappedByteBuffer{ public static final int _4kb = 4*;public static final int _GB= 1**;public static void main(String[] args) throws IOException, InterruptedException { // 为了方便在日志中找到本段代码的开始位置和结束位置,这里利用文件io来打开始标记FileInputStream startInput = null;try { startInput = new FileInputStream("start1.txt");startInput.read();} catch (IOException e) { e.printStackTrace();}File file = new File("filename");FileChannel fileChannel = new RandomAccessFile(file, "rw").getChannel();MappedByteBuffer map = fileChannel.map(FileChannel.MapMode.READ_WRITE, 0, _GB); //我们想分析的语句问题2for (int i = 0; i < _GB; i++) { map.put((byte)0); // 下文中需要分析的语句目的2}// 打结束标记FileInputStream endInput = null;try { endInput = new FileInputStream("end.txt");endInput.read();} catch (IOException e) { e.printStackTrace();}}}把上面这段代码编译后把“.class”文件拉到linux执行,并用linux上的strace工具记录其系统调用日志,拿到日志文件我们可以在日志中看到以下信息(关于怎么拿到日志可以参照我的博文:无(代写)):
注:日志有多行,这里只选取我们关注的
// ...// 看到了我们打的开始标志openat(AT_FDCWD, "start1.txt", O_RDONLY) = -1 ENOENT (No such file or directory)// ... // 打开文件,文件描述符fd为6openat(AT_FDCWD, "filename", O_RDWR|O_CREAT, ) = 6// 判断文件状态fstat(6, { st_mode=S_IFREG|, st_size=, ...}) = 0// ... // 判断文件状态fstat(6, { st_mode=S_IFREG|, st_size=, ...}) = 0// 进行内存映射mmap(NULL, , PROT_READ|PROT_WRITE, MAP_SHARED, 6, 0) = 0x7f2fd6cd// ...// 程序退出exit(0)// 看到了我们打的结束标志openat(AT_FDCWD, "end.txt", O_RDONLY) = -1 ENOENT (No such file or directory)在上面程序的系统调用日志中我们确实看到了我们打的开始标志,结束标志。在开始标志和结束标志之间我们看到了我们的文件"filename"确实被打开了,文件描述符fd = 6;在打开文件后紧接着又执行了系统调用mmap,这一点我们Java代码一致,这样,我们就验证了我们答案1中的结论,可以开始我们的下文了
源码追踪分析,从Java到JNI,再到JVM的保护 javascript源码C++目的1寻源之旅:fileChannel.map()我们知道我们执行Java代码fileChannel.map()确实会在底层调用系统调用,那怎么在源码中得到验证呢?怎么落脚于源码进行分析呢?下面开始我们的寻源之旅
FileChannelImpl.map() 注:由于代码较长,这里代码中略去了一些我们不关注的,比如异常捕获等
public MappedByteBuffer map(MapMode mode, long position, long size)throws IOException{ // ...try { // ...synchronized (positionLock) { // ...long mapPosition = position - pagePosition;mapSize = size + pagePosition;try { // !我们要找的语句就在这!addr = map0(imode, mapPosition, mapSize);} catch (OutOfMemoryError x) { // 如果内存不足,先尝试进行GCSystem.gc();try { Thread.sleep();} catch (InterruptedException y) { Thread.currentThread().interrupt();}try { // 再次试着mmapaddr = map0(imode, mapPosition, mapSize);} catch (OutOfMemoryError y) { // After a second OOME, failthrow new IOException("Map failed", y);}}} // ...} finally { // ...}}上面函数源码中真正执行mmap的语句是在addr = map0(imode, mapPosition, mapSize),于是我们寻着这里继续追踪
FileChannelImpl.map0()
// Creates a new mappingprivate native long map0(int prot, long position, long length)throws IOException;可以看到,该方法是一个native方法,所以后面的源码我们需要到这个FileChannelImpl.class对应的fileChannelImpl.c中去看,所以我们需要去找到JDK的源码
在JDK源码中我们找到fileChannelImpl.c文件
fileChannelImpl.c 根据JNI的对应规则,我们找到该文件内对应的Java_sun_nio_ch_FileChannelImpl_map0方法,其源码如下:
JNIEXPORT jlong JNICALLJava_sun_nio_ch_FileChannelImpl_map0(JNIEnv *env, jobject this, jint prot, jlong off, jlong len){ void *mapAddress = 0;jobject fdo = (*env)->GetObjectField(env, this, chan_fd);jint fd = fdval(env, fdo);int protections = 0;int flags = 0;if (prot == sun_nio_ch_FileChannelImpl_MAP_RO) { protections = PROT_READ;flags = MAP_SHARED;} else if (prot == sun_nio_ch_FileChannelImpl_MAP_RW) { protections = PROT_WRITE | PROT_READ;flags = MAP_SHARED;} else if (prot == sun_nio_ch_FileChannelImpl_MAP_PV) { protections =PROT_WRITE | PROT_READ;flags = MAP_PRIVATE;}// !我们要找的语句就在这里!mapAddress = mmap(0,/* Let OS decide location */len,/* Number of bytes to map */protections,/* File permissions */flags,/* Changes are shared */fd, /* File descriptor of mapped file */off); /* Offset into file */if (mapAddress == MAP_FAILED) { if (errno == ENOMEM) { JNU_ThrowOutOfMemoryError(env, "Map failed");return IOS_THROWN;}return handle(env, -1, "Map failed");}return ((jlong) (unsigned long) mapAddress);}我们要找的语句就上面代码中的mapAddress = mmap(0,len,protections,flags,fd,off),至于为什么不是直接的mmap,而是mmap,是因为这里的mmap是一个宏,在文件上方有其定义,如下:
#define mmap mmap至此,我们就在源码中得到验证了我们问题2中的结论:fileChannelImpl.map()底层使用的是mmap系统调用
目的2寻源之旅:mmapedByteBuffer.put(Byte[ ])接着我们来看看当我们调用mmapedByteBuffer.put(Byte[])JVM底层在搞些什么动作
MappedByteBuffer ?首先我们得知道,当我们执行MappedByteBuffer map = fileChannel.map(FileChannel.MapMode.READ_WRITE, 0, _GB)时,实际返回的对象是DirectByteBuffer类的实例,因为MappedByteBuffer为抽象类,且只有DirectByteBuffer继承了它,看下面两图就明白了
DirectByteBuffer 于是我们找到DirectByteBuffer内的put(Byte[ ])方法
public ByteBuffer put(byte x) { unsafe.putByte(ix(nextPutIndex()), ((x)));return this;}可以看到该方法内实际是调用Unsafe类内的putByte方法来实现功能的,所以我们还得去看Unsafe类
Unsafe.class
public native voidputByte(long address, byte x);该方法在Unsafe内是一个native方法,所以所以我们还得去看unsafe.cpp文件内对应的实现
unsafe.cpp
在JDK源码中,我们找到unsafe.cpp
在这份源码内,没有使用JNI内普通加前缀的方法来形成对应关系
不过我们还是能顺着源码的蛛丝轨迹找到我们要找的方法
注意到源码中有这样的注册机制,所以我们可以知道我们要找的代码就是上图中标注的代码
顺藤摸瓜,我们就找到了该方法的定义
UNSAFE_ENTRY(void, Unsafe_SetNative##Type(JNIEnv *env, jobject unsafe, jlong addr, java_type x)) \UnsafeWrapper("Unsafe_SetNative"#Type); \JavaThread* t = JavaThread::current(); \t->set_doing_unsafe_access(true); \void* p = addr_from_java(addr); \*(volatile native_type*)p = x; \t->set_doing_unsafe_access(false); \UNSAFE_END \该方法内主要的逻辑语句就是以下两句:
/** * @author Tptogiar * @description * @date /5/ - : */public class TestMappedByteBuffer{ public static final int _4kb = 4*;public static final int _GB= 1**;public static void main(String[] args) throws IOException, InterruptedException { // 为了方便在日志中找到本段代码的开始位置和结束位置,这里利用文件io来打开始标记FileInputStream startInput = null;try { startInput = new FileInputStream("start1.txt");startInput.read();} catch (IOException e) { e.printStackTrace();}File file = new File("filename");FileChannel fileChannel = new RandomAccessFile(file, "rw").getChannel();MappedByteBuffer map = fileChannel.map(FileChannel.MapMode.READ_WRITE, 0, _GB); //我们想分析的语句问题2for (int i = 0; i < _GB; i++) { map.put((byte)0); // 下文中需要分析的语句目的2}// 打结束标记FileInputStream endInput = null;try { endInput = new FileInputStream("end.txt");endInput.read();} catch (IOException e) { e.printStackTrace();}}}0至此,我们就知道:其实我们调用mmapedByteBuffer.put(Byte[ ])时,JVM底层并不需要涉及到系统调用(这里也可以用strace工具追踪从而得到验证)。也就是说通过mmap映射的空间在内核空间和用户空间是共享的,我们在用户空间只需要像平时使用用户空间那样就行了————获取地址,设置值,源码git 安装而不涉及用户态,内核态的切换
总结fileChannelImpl.map()底层用调用系统函数mmap
fileChannelImpl.map()返回的其实不是MappedByteBuffer类对象,而是DirectByteBuffer类对象
在linux上可以通过strace来追踪系统调用
JNI中“.class”文件内方法与“.cpp”文件内函数的对应关系不止是前缀对应的方法,还可以是注册的方式,这一点的追寻代码的时候有很大帮助
directByteBuffer.put()方法底层并没有涉及系统调用,也就不需要涉及切态的性能开销(其底层知识执行获取地址,设置值的操作),所以mmap的性能就比普通读写read/write好
...
原文:/post/laravelåthinkphpçåºå«ï¼
thinkphp主è¦æä»ä¹ç¨
1ãèä¸å¯ä»¥å¾æ¹ä¾¿çæ§å¶æ³¨å ¥çåºæ¬å®å ¨æªæ½ãåç§å¤åçæ¹æ³ä¹é常好ç¨ï¼æ¯å¦ajaxãcookieçç2åç§åè½é½å ¨ãæ³è¦é«å¤§ä¸ï¼æ²¡é®é¢ï¼thinkphpçåç§åè½é½é常é½å ¨ï¼ä»ä¹ä¸ä¼ ãéªè¯ç ãå页ãé«éç¼åã模åéªè¯balabalaå ¨é½æã
2ãæ¨å¥½ï¼ææ¥ä¸ºæ¨è§£ThinkPHPæ¯ä¸ä¸ªå è´¹å¼æºçãå¿«éç®åçãé¢å对象çãMVCç»æçè½»é级PHPå¼åæ¡æ¶ï¼æ¯ä¸ºäºææ·Webåºç¨å¼ååç®åä¼ä¸åºç¨å¼åèè¯ççã项ç®åç«äºå¹´åï¼éµå¾ªApache2å¼æºåè®®åå¸ã
3ãthinkphpæ¯ç¼ç¨è¯è¨phpçä¸ä¸ªæ¡æ¶ï¼æ¡æ¶å°±æ¯å·¥å ·éåä½ï¼æé«å¼åæççã
4ãThinkphp模åç±»ç主è¦ä½ç¨æ¯ï¼å®æä¸å¡é»è¾å¤çï¼å æ¬å¯¹æ°æ®è¡¨çå¢å æ¹æ¥ï¼CUEDï¼æä½ã对å¤ççæ°æ®è¿è¡å°è£ ï¼å¯¹å段åå±æ§è¿è¡éªè¯ï¼å®æ对象åå±æ§çè¿æ»¤çåè½ã
5ãä¸è¬æ åµä¸æ¯ä¸ä¸ªè¡¨å¯¹åºä¸ä¸ªmodelç±»ï¼ä¸ç¨ä¹æ¯å¯ä»¥çã
6ãè¿ä¸ªæ¯ä¸ä¸ªactionç±»ï¼æåä¸å¥$this-display(Publicï¼text)ï¼æ¯æ¾ç¤ºpublicä¸çtext模æ¿ãä¹å°±æ¯è¯´å æ§è¡è¿ä¸ªç±»çè¿ä¸ªæ¹æ³ï¼ç¶åæ¾ç¤ºæ¨¡æ¿ã模æ¿ç¨å°çåéå°±ä»è¿éè°ç¨assignåé è¿å»ã
thinkphpä¸phpçåºå«
æ»çæ¥è¯´ï¼ç»æ¢¦ï¼dedecmsï¼ãphpcmsé½æ¯cmså 容管çç³»ç»ï¼thinkPHPæ¯PHPçä¸ç§å¼åæ¡æ¶ï¼cmsç¸å¯¹ç®åï¼è®°ä½å¯¹åºçè°ç¨æ ç¾ï¼å°±æ¯ç®åçå¥æ¨¡æ¿ï¼ä¸ç¨å¼ååå°ï¼äºæ¬¡å¼åé¤å¤ï¼ï¼æ¨¡ååªè½æ¯cmsæä¾ç模åã
åºæ¬ä¸å¤æ°cmsé½æ¯åºäºæ¡æ¶å¼åçï¼å ¶ä¸æäºcmsç³»ç»å°±æ¯åºäºthinkPHPæ¡æ¶å¼åèµ·æ¥çï¼æ以å¯ä»¥è¯´ç¨äºå¼å项ç®çæ¡æ¶ï¼æ¯é¡¹ç®çå身ã--就好æ¯ç³ç åæ¿å±ã
æä¸å¾ä¸æ¿è®¤ThinkPHPæ¯ä¸ä¸ªæ¯å½å æ¡æ¶è¿è¥æ¹é¢çæ¦æ ·ï¼å½FleaPHP/QeePHPæç«çé£éµï¼æ说è¿FleaPHP/QeePHPä¼åçï¼ã
æ人说tpæ¯è¾èè¿ï¼æè§å¾è¿æ¯å®çä¼ç¹ï¼å 为强大æ以æèè¿ï¼å¦ææ¯é常å°ç项ç®ï¼æè§å¾ç¡®å®ä¸å¤ªéåï¼åºè¯¥ä½¿ç¨æ´è½»é级çæ¡æ¶ã
ç®åphpå¼åç½ç«,æ为æµè¡çæ¡æ¶æ¯åªå 个?CakePHPCakePHPæ¯ä¸ä¸ªå¿«éå¼åPHPçæ¡æ¶ï¼å ¶ä¸ä½¿ç¨äºä¸äºå¸¸è§ç设计模å¼å¦ï¼AssociationDataMappingï¼FrontController以åMVCã
ä¸é¢æ们èä¸èæ¯è¾æµè¡çä¸ä¸ªphpå¼åæ¡æ¶ãLavarelï¼è¿æ¯ä¸æ´å¥è¯æ³ç®æ´ï¼å¥½ç解çwebå¼åæ¡æ¶ã主è¦çä¼å¿éä¸ä½ç°å¨åæçæ¯æãææ¡£çå ¨å¤ï¼è¿ææ´»è·ç社åºä¸ã
Canphpæ¯ä¸ä¸ªä¸å½å¼æºçphpæ¡æ¶ï¼ä¸»è¦ç设计ç念就æ¯ä¸ºäºè¿½æ±ç®åï¼åå°phpåå¦è çå¦ä¹ ææ¬ï¼èä¸è¿½æ±é«æçï¼æ¨¡çç±»ä¸æä¾å ¶ä»æ ç¾ï¼ç´æ¥åºç¨phpçæºä»£ç ï¼è¿æå®æ¯å¾®å æ ¸è®¾è®¡ï¼ç§»æ¤æ§å¼ºï¼æ¾æ£è¦åçç¹ç¹ï¼è½éæå°ä½ ä»»ä½ä¸ä¸ªç³»ç»éé¢ã
CakePHPå¼åæ¡æ¶å¦æä½ ä»ç¶éè¦ç¼åé¢åPHP4å ¼å®¹ç代ç ï¼CakePHPå°æ¯ä¸ä¸ªé常ä¸éçéæ©ï¼å¨PHP4&5çMVCå¼æ¡æ¶å表éé¢ï¼CakePHPé½æ¾ç»æ¯ææµè¡çã
PHP常ç¨æ¡æ¶æï¼thinkphpï¼å½äººå¼åï¼ææ¡£æ¯è¾å ¨ï¼ä¸æææ¡£ãå ¥é¨æ¯è¾ç®åãåæ³å¿«ãlaravelï¼å½å¤äººåçï¼å ¥é¨é¨æ§é«ãè±è¯ææ¡£ï¼ç¤¾åºæ¯è¾æ´»è·ãsymfonyï¼å½å¤äººåçãé¨æ§æ¯è¾é«ï¼è±è¯ææ¡£ã
ææµè¡çPHPæ¡æ¶ï¼Symfonyï¼LaravelåYiiãSymfonySymfonyæ¯ä¸å¥å¯éå¤ä½¿ç¨çPHPç»ä»¶ï¼å®å 许å¼åè 人åå建å¯æ©å±çï¼é«æè½çåºç¨ç¨åºãå®æ个ç»ä»¶å¯ä»¥éæ©ï¼å¼å人åå¯ä»¥æ足å¤çèªç±å¨RADç¯å¢ä¸è¿è¡è¯éªåå·¥ä½ã
ä¼ä¸å¼åç¨tp6è¿æ¯hyperf1ãç¼ç¨è¯è¨ç±»åä¸åï¼TP6æ¯ä¸ç§åºäºPrologè¯è¨çé»è¾ç¼ç¨è¯è¨ï¼èJavaæ¯ä¸ç§åºäºé¢å对象ç¼ç¨çéç¨ç¼ç¨è¯è¨ãåºç¨åºæ¯ä¸åï¼TP6主è¦ç¨äºäººå·¥æºè½ãèªç¶è¯è¨å¤çåä¸å®¶ç³»ç»çé¢åï¼èJavaå广æ³åºç¨äºä¼ä¸çº§åºç¨åWebåºç¨å¼åã
2ãæ²¡å¿ è¦ãtp6æ²¡å¿ è¦ç¨å¤åºç¨ï¼å 为æ¬å°±æ¯åå ¥å£èªå¨å¤åºç¨æ¨¡å¼ï¼å¹¶ä¸ä¸éè¦ä½¿ç¨composeræ¥å¼å¯å¤åºç¨æ¨¡å¼æ©å±ï¼tp6åå¸å¼é¨ç½²å¤ä¸ªæ°æ®åºï¼å®ç°è¯»åå离ãtp6çåå¸å¼é¨ç½²è¯»ååä»ç¶æ¯ä¸ä¸ªç³»ç»ã
3ãtp6loadåimportæ¯PHPå¼åä¸é常常ç¨ç两个å½æ°ãtp6loadç¨äºå 载类æ件ï¼èimportåæ¯éè¿åå空é´å 载类æ件ã
4ãå¯ä»¥æ¯å¯ä»¥ï¼ä½æ¯ä¸æ¨èï¼commonéé¢æ¾çæ¹æ³æ´å¤çæ¯å ¬ç¨çæ¹æ³ï¼æ¯å¦å¾å¤æ¨¡åé½ä¼è°ç¨çæ¹æ³ãå æ¤ï¼å¦æä¸ä¸ªæ¹æ³åªæ¯ä¸ªå«æ¨¡å使ç¨çè¯ï¼å»ºè®®å°å®æ¾å°æ¨¡å对åºçå é¨å³å¯ï¼è¿æ ·æ´æ¹ä¾¿ä»£ç çéæåè¿ä»£ï¼æ´åºè²ã
5ãè¿å 个çåéçè¯åºè¯¥æ¯åç¡çè·¯ç±å¨æ好ï¼è ¾è¾¾åTPæ¯å ¥é¨çº§å®¶ç¨è·¯ç±å¨ãå¦ææ¯å¯¹ç½éæå¾é«è¦æ±çè¯å¯ä»¥èèæç§æè 软路ç±ãå¦ææ¯ä½å¸¦å®½ç¯å¢çè¯å 个çåçè·¯ç±å¨é½å·®ä¸å¤çã
phpæ¡æ¶åªä¸ªå¥½ç¨,容æä¸æ1ãSpeedPHPæ¡æ¶æ¯ä»å®é è¿è¡çåä¸ç³»ç»ä¸åå ¶ç²¾åèæçï¼å¨ç¨³å®æ§åè¿è¡é度ä¸é½é常åºè²ï¼åæ¶æçæ¸ æ°çæ¶æï¼æ´æå©äºæé«å¢éå¼åæçï¼æç¨ä¼å¤ï¼å ¥é¨å®¹æï¼å·ç§°æéååå¦è çPHPæ¡æ¶ï¼å¿«éå¸¦ä½ è¿å ¥PHPé«æçè¡åã
2ãLavarelï¼è¿æ¯ä¸æ´å¥è¯æ³ç®æ´ï¼å¥½ç解çwebå¼åæ¡æ¶ã主è¦çä¼å¿éä¸ä½ç°å¨åæçæ¯æãææ¡£çå ¨å¤ï¼è¿ææ´»è·ç社åºä¸ãå¦ä¹ Lavarelï¼éè¦å 对PHPè¯è¨ãé¢å对象ãæ°æ®åºçæä½æä¸ä¸ªåºæ¬çå¦ä¹ åºç¡ã
3ãæ¨è使ç¨thinkphpï¼è¿ä¸ªæ¯è¾ç®åï¼ä¸ææ¯è¾å¿«ThinkPHPæ¯ä¸ä¸ªå¿«éãå ¼å®¹èä¸ç®åçè½»é级å½äº§PHPå¼åæ¡æ¶ï¼è¯çäºå¹´åï¼ååFCSï¼å¹´å æ¦æ£å¼æ´å为ThinkPHPï¼ä»¥ä¸ç®ç§°TPï¼ãä½ä¸ºå½äº§æ¡æ¶ï¼å¨å½å ã
4ãé¢ä¸»å¦ææ¯ä¸ä¸ªåæ¥è§¦PHPçæ°äººçè¯ï¼å¯è½ä¼è§å¾æ¡æ¶å¾å¤ï¼å个æ¡æ¶é½æåèªçç¹ç¹ï¼é¾ä»¥ææ¡ï¼å ¶å®è½ç¶æ¯ä¸ªæ¡æ¶çç¹ç¹ä¸ä¸æ ·ï¼ä½æ¯æ¡æ¶çæè·¯é½æ¯å·®ä¸å¤çï¼å¦æä½ ç¨çäºï¼å°±ä¼è§å¾ä»ä¹æ¡æ¶é½å·®ä¸å¤ã
5ãä»MVCï¼DAO/ï¼widgetsï¼cachingï¼ç级å¼RBACï¼Webæå¡ï¼å°ä¸»ä½åï¼INåLNï¼Yiiæä¾äºä»æ¥Web0åºç¨å¼åæéè¦çå ä¹ä¸ååè½ãèä¸è¿ä¸ªæ¡æ¶çä»·æ ¼ä¹å¹¶ä¸å¤ªé«ãäºå®ä¸ï¼Yiiæ¯æææççPHPæ¡æ¶ä¹ä¸ã
6ãä½ å¥½ï¼å ¶å®ç°å¨ä¸»æµçPHPæ¡æ¶é½åæä¼å¿ï¼èä¸é½å¯ä»¥å®æphpå¼åè çåºæ¬è¦æ±ãä½æ¯å ¶å®å¨ä½¿ç¨çä¸ï¼æ¯ä¸ªåºåé½æ¯ä¸ä¸æ ·çãå¨ä¸å½ï¼thinkphpç使ç¨çè¯å®æ¯æé«çï¼èå¨å¤å½å个æ¡æ¶ç使ç¨çä¹ä¸ä¸æ ·ã
宝塔安装ThinkPHP6 详细过程
最近我开始学习ThinkPHP,因此打开了Centos7系统,以下内容将详细记录我在宝塔面板安装TP6过程中遇到的问题及解决方法。
首先,由于TP6只能通过composer安装,因此我在宝塔中先安装了composer。关于安装步骤,网上有很多教程,这里仅作简要说明。
在安装过程中,需要删除以下禁用函数:php管理 ——> 禁用函数 ——> 删除函数,putenv()。
我采用的是局部安装方式,得到了composer.phar文件。随后,我将该文件移动到全局目录下,并去除了后缀。现在,该文件已经位于/usr/bin/目录下。执行composer -v命令后,显示执行成功。
接下来,我修改了镜像源,使用阿里的源,也可以选择其他源。成功更换为阿里源后,我切换到/www/wwwroot/目录下,执行以下命令下载TP6的源码:composer TP6的源码。这里,你可以将tp目录名更改为任意名称,这个目录将成为我们后续操作的应用根目录。我将它修改为TP6。
执行完毕后,可以看到ThinkPHP6的源码已经下载到本地。
接下来,我使用宝塔创建站点及其数据库,并将网站根目录设置为存放源码的文件夹。然后,我修改网站设置,将网站目录和运行目录都设置为public。
最后,直接访问网站,即可看到ThinkPHP6的首页。
关于开启调试模式,只需要将根目录下的.example.env重命名为.env即可。在文件中,你可以进行控制,true代表开启调试,false代表关闭调试。
超详细通达信主图指标安装教程,意简言赅过程简单!有惊喜源码
安装通达信主图指标的步骤简洁明了,只需遵循以下步骤即可。
首先,找寻你心仪的主图指标公式。进入通达信软件,找到“功能”菜单,点击“公示系统”,然后选择“公式管理器”。打开公式管理器后,转至“技术指标公式”部分,选择“其他类型”,点击右侧的“新建”按钮,此时将打开指标公式编辑器。
在编辑器中,输入公式名称,选择画线方法,并粘贴已选中的公式。确认无误后,点击确定按钮完成公式创建。
接下来,预览新建的指标。返回至公式管理器界面,找到新增的指标,点击右侧的“预览”按钮,预览图将随即呈现。
源码如下,供参考学习与应用。代码包含复杂的指标逻辑,用于辅助交易决策。
VAR1:=CLOSE-LOW;VAR2:=HIGH-LOW;VAR3:=CLOSE-HIGH;VAR4:=IF(HIGH>LOW,(VAR1/VAR2+VAR3/VAR2)*VOL,0);HPTP:=SUM(VAR4,)/,COLORSTICK;TKXL:=-1;XVYO:=UPNDAY(TKXL,1),NODRAW;G:=MA(C,5);D:=MA(C,);HH:=REF(H,5)=HHV(H,);LL:=REF(L,5)=LLV(L,);FG:=BACKSET(HH,6)>BACKSET(HH,5);FD:=BACKSET(LL,6)>BACKSET(LL,5);FG:=IF(BARSLAST(FG)=BARSLAST(FD) AND G>D,FG,IF(BARSLAST(FD)>BARSLAST(FG),FG,0));FD:=IF(BARSLAST(FG)=BARSLAST(FD) AND D>G,FD,IF(BARSLAST(FG)>BARSLAST(FD),FD,0));FG0:=FG AND H=HHV(H,BARSLAST(FD));FD0:=FD AND L=LLV(L,BARSLAST(FG));GQ:=L>REF(H,1) AND DAY!=REF(DAY,1);DQ:=H>=REF(H,1) AND L<=REF(L,1);BHG:=COUNT(BH0,BARSLAST(FD0));BHD:=COUNT(BH0,BARSLAST(FG0));BGQ:=COUNT(GQ,BARSLAST(FD0));BDQ:=COUNT(DQ,BARSLAST(FG0));BK0:=IF(BHG>0,BHG+2,IF(BHD,BHD+2,3));BK:=IF(BGQ,BK0-BGQ,IF(BDQ,BK0-BDQ,BK0));G1X:=(FG AND BARSLAST(FD)>BK),NODRAW;D1X:=(FD AND BARSLAST(FG)>BK),NODRAW;G1:=(FG0 AND REF(H,BARSLAST(FG0))>=REF(H,BARSLAST(G1X)) AND BARSLAST(D1X)>BARSLAST(G1X))OR (FG1 AND COUNT(GQ,BARSLAST(FD1))>0 AND REF(H,BARSLAST(FG1))>REF(H,BARSLAST(G1X)));{ W:=IF(DATE<,C,DRAWNULL);}D1:=(FD0 AND REF(L,BARSLAST(FD0))<=REF(L,BARSLAST(D1X)) AND BARSLAST(G1X)>BARSLAST(D1X))OR (FD1 AND COUNT(DQ,BARSLAST(FG1))>0 AND REF(L,BARSLAST(FD1))>REF(L,BARSLAST(G1X)));G2:=G1 AND H=HHV(G1H,BARSLAST(D1)+1) AND H>REF(H,1) AND BARSLAST(D1)>BARSLAST(G1);D2:=D1 AND L=LLV(D1L,BARSLAST(G1)+1)
为了感谢各位粉丝的支持,我承诺提供此技术与指标的免费分享。如需使用或获取全套源码,欢迎私信联系我。
路由器推荐,路由器怎么选不踩雷,路由器哪个牌子好用,6款实用路由器排行
在智能化时代,家用路由器已成为连接家庭与数字世界的桥梁。它不仅关乎网络的稳定与流畅,更是家庭智能设备的中枢。一款优质的家用路由器,能够满足高清视频播放、在线游戏、远程办公等需求,提供强大网络支持。现代路由器融合了智能管理、安全防护等功能,使家庭网络使用更为便捷、安全。
选购家用路由器时,需综合考虑以下关键因素:
1. **Wi-Fi标准与协议**:
- **Wi-Fi 6(.ax)**:提供更快的速度、更高的设备容量和更低的延迟,适合多设备家庭或办公环境。
- **Wi-Fi 5(.ac)**:新一代标准,支持较高带宽,适用于大多数家庭。
- **Wi-Fi 4(.n)及更早**:适合基本需求,但可能不支持高速和大量设备同时连接。
2. **无线频段**:
- **双频或多频路由器**:支持2.4GHz和5GHz频段,甚至更高频段,5GHz频段速度快,干扰少;2.4GHz覆盖范围广。
3. **无线速度**:
- 无线速度表示为理论最大传输速率,如AC(Wi-Fi 5,Mbps)、AX(Wi-Fi 6,Mbps)。实际速度会受环境影响。
4. **天线技术**:
- **内置与外置天线**:内置天线外观简洁,外置天线可灵活调整方向以优化信号。
- **MU-MIMO**:允许路由器同时与多个设备通信,提高网络效率。
- **波束成形**:定向增强信号,提供更强的连接质量。
5. **端口配置**:
- **WAN/LAN端口**:至少1个WAN口用于连接互联网,LAN口数量(通常4个或更多)根据家庭有线设备数量决定。
- **Gigabit Ethernet**:确保所有以太网端口支持千兆速度。
- **USB端口**:用于连接存储设备或打印机,实现网络共享。
6. **安全性**:
- **WPA3**:最新的Wi-Fi安全协议,提供更强的数据保护。
- **内置硬件防火墙**:保护网络安全。
- **家长控制**:限制特定设备的上网时间和访问内容。
7. **处理器与内存**:
- **高性能CPU**:如高通、博通等品牌,支持大量数据流并发。
- **更高内存**:如MB或MB,提升处理复杂任务的能力。
8. **QoS(Quality of Service)**:
- **允许用户自定义数据流优先级**,如优先保证视频或游戏流量,确保流畅体验。
9. **管理界面**:
- **易用性**:Web界面或手机APP,支持远程管理和固件升级。
- **开放源代码固件支持**:如DD-WRT、OpenWrt,为高级用户提供更多定制化选项。
以下推荐的路由器,旨在满足不同家庭网络需求:
1. **华为Q6**:采用凌霄子母路由系统,覆盖全面,支持Wi-Fi6+,大幅提升信号质量与稳定性,适合大平层家庭。
2. **夏新CPE**:采用烟囱散热系统,支持5G极速上网,提供全配套,包括散热、特增加大金属散热模块和多矩阵散热片,确保稳定性能输出。
3. **中兴U Pro**:全球超长续航5G随身WiFi,高性能骁龙X芯片平台,支持SA+NSA网络环境,适配全球频段,提供全球漫游功能,适合经常出国的家庭。
4. **华硕RT-AXU Pro**:搭载博通芯片四核2.0GHZ处理器,支持Wi-Fi6,全场景网络加速,提供易展无缝漫游组网,保障网络安全防护。
5. **TP-LINK K**:分布式子母路由,无需区分主、子,全屋Wi-Fi6覆盖,易展MESH技术组网,适合大户型和复杂户型。
6. **京东云无线宝BE**:高通四核Wi-Fi7,支持两频三频自由切换,提供全2.5G疾速网口,自研Mesh组网2.0,适合追求极致网络速度的家庭。
选购家用路由器时,应根据个人家庭网络实际需求进行选择,确保网络稳定、高速,让智能设备之间的通信更加流畅。通过综合评估路由器的关键参数,如发射功率、频段支持、内存和闪存等,将能为家庭找到最适合的解决方案。