实现高质量视频通话的javascript技巧与方法
.markdown-body { line-height: 1.75; font-weight: 400; font-size: 15px; overflow-x: hidden; color: rgba(43, 43, 43, 1); font-family: -apple-system, system-ui, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Hiragino Sans GB, Microsoft YaHei, Arial, sans-serif; background-image: linear-gradient(90deg, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0), linear-gradient(1turn, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0); background-size: 20px 20px; background-position: center }
.markdown-body h1, .markdown-body h2, .markdown-body h3, .markdown-body h4, .markdown-body h5, .markdown-body h6 { padding: 30px 0; margin-top: 35px; margin-bottom: 10px; color: rgba(77, 208, 225, 1) }
.markdown-body h1 { font-size: 30px; text-align: center; position: relative; width: max-content; margin: 0 auto }
.markdown-body h1:before { position: absolute; content: ""; z-index: -1; top: -20px; height: 100%; width: 100px; left: 0; right: 0; margin: 0 auto; background: url("") center / 64px 64px no-repeat; opacity: 0.84 }
.markdown-body h1:after { position: absolute; content: ""; width: 150%; left: -25%; height: 50%; bottom: 12px; border-radius: 50%; background: linear-gradient(rgba(0, 0, 0, 0) 80%, rgba(77, 208, 225, 0.8)); opacity: 0.6; animation: 6s linear infinite h1animate }
@keyframes h1Animate { 0% { background-position: right bottom } 50% { background-position: right } 100% { background-position: right bottom } }
.markdown-body h2 { display: block; border-bottom: 4px solid rgba(77, 208, 225, 1); position: relative; font-size: 24px; padding: 12px 32px; margin: 30px 0 }
.markdown-body h2:before { width: 24px; height: 24px; left: 0; top: 0; margin: auto; background-size: 24px 24px; background-image: url("") }
.markdown-body h2:after, .markdown-body h2:before { content: ""; display: block; position: absolute; bottom: 0 }
.markdown-body h2:after { right: 0; width: 400px; height: 10px; border-top-right-radius: 24px; background: linear-gradient(90deg, rgba(255, 255, 255, 1), rgba(77, 208, 225, 1)); max-width: 50vw }
.markdown-body h3 { margin: 30px 0; font-size: 18px; position: relative; padding: 4px 32px; width: max-content }
.markdown-body h3:before { border-bottom: 2px solid rgba(77, 208, 225, 1); width: 100%; content: ""; display: block; height: 28px; position: absolute; left: 0; top: 0; bottom: -2px; margin: auto; background-size: 28px 28px; background-image: url(""); background-repeat: no-repeat; animation: 2s infinite alternate h3animationbefore }
@keyframes h3AnimationBefore { 0% { width: 28px } 25% { width: 100% } 50% { width: 100% } 100% { width: 100% } }
.markdown-body h3:after { content: ""; display: block; width: 28px; height: 28px; position: absolute; border: 2px solid rgba(77, 208, 225, 1); border-radius: 50%; right: -15px; top: 0; bottom: 0; margin: auto; background-size: 28px 28px; background-image: url(""); animation: 2s infinite alternate h3animationafter }
@keyframes h3AnimationAfter { 0% { } 10% { } 50% { transform: rotate(-1turn) } 100% { transform: rotate(-1turn) } }
.markdown-body h4 { font-size: 16px }
.markdown-body h5 { font-size: 15px }
.markdown-body h6 { margin-top: 5px }
.markdown-body p { line-height: inherit; margin: 22px 0; letter-spacing: 2px; font-size: 14px; word-spacing: 2px }
.markdown-body img { max-width: 80%; border-radius: 6px; display: block; margin: 20px auto !important; object-fit: contain; box-shadow: 0 0 16px rgba(110, 110, 110, 0.45) }
.markdown-body figcaption { display: block; font-size: 13px; color: rgba(43, 43, 43, 1) }
.markdown-body figcaption:before { content: ""; background-image: url(""); display: inline-block; width: 18px; height: 18px; background-size: 18px; background-repeat: no-repeat; background-position: center; margin-right: 5px; margin-bottom: -5px }
.markdown-body hr { border-top: 1px solid rgba(77, 208, 225, 1); border-right: none; border-bottom: none; border-left: none; margin-top: 32px; margin-bottom: 32px }
.markdown-body del { color: rgba(77, 208, 225, 1) }
.markdown-body code { border-radius: 2px; overflow-x: auto; background-color: rgba(77, 208, 225, 0.08); color: rgba(38, 198, 218, 1); padding: 0.195em 0.4em }
.markdown-body pre { font-family: Menlo, Monaco, Consolas, Courier New, monospace; overflow: auto; position: relative; line-height: 1.75; box-shadow: 0 0 8px rgba(110, 110, 110, 0.45); border-radius: 4px; margin: 16px }
.markdown-body pre:before { content: ""; display: block; height: 30px; width: 100%; margin-bottom: -7px; background: url("") 10px 10px / 40px no-repeat }
.markdown-body pre>code { font-size: 12px; padding: 15px 12px; margin: 0; word-break: normal; display: block; overflow-x: auto; color: rgba(51, 51, 51, 1); background: rgba(248, 248, 248, 1) }
.markdown-body a { color: rgba(77, 208, 225, 1); border-bottom: 1px solid rgba(77, 208, 225, 1); font-weight: 400; text-decoration: none; margin: 0 4px }
.markdown-body a:active, .markdown-body a:hover { background-color: rgba(77, 208, 225, 0.1) }
.markdown-body strong { color: rgba(38, 198, 218, 1) }
.markdown-body strong:before { content: "「" }
.markdown-body strong:after { content: "」" }
.markdown-body em { font-style: normal; color: rgba(77, 208, 225, 1); font-weight: 700 }
.markdown-body table { display: inline-block !important; font-size: 12px; width: auto; max-width: 100%; overflow: auto; border: 1px solid rgba(246, 246, 246, 1) }
.markdown-body thead { background: rgba(246, 246, 246, 1); color: rgba(0, 0, 0, 1); text-align: left }
.markdown-body tr:nth-child(2n) { background-color: rgba(77, 208, 225, 0.05) }
.markdown-body td, .markdown-body th { padding: 12px 7px; line-height: 24px }
.markdown-body td { min-width: 120px }
.markdown-body blockquote { margin: 2em 0; padding: 24px 32px; border-left: 4px solid rgba(38, 198, 218, 1); background: rgba(77, 208, 225, 0.15); position: relative }
.markdown-body blockquote:before { content: "❝"; top: 8px; left: 8px; color: rgba(77, 208, 225, 1); font-size: 30px; line-height: 1; font-weight: 700; position: absolute; opacity: 0.7 }
.markdown-body blockquote:after { content: "❞"; font-size: 30px; position: absolute; right: 8px; bottom: 0; color: rgba(77, 208, 225, 1); opacity: 0.7 }
.markdown-body blockquote p { color: rgba(89, 89, 89, 1); line-height: 2 }
.markdown-body ol, .markdown-body ul { color: rgba(89, 89, 89, 1); padding-left: 28px }
.markdown-body ol li, .markdown-body ul li { margin-bottom: 0; list-style: inherit }
.markdown-body ol li .task-list-item, .markdown-body ul li .task-list-item { list-style: none }
.markdown-body ol li .task-list-item ol, .markdown-body ol li .task-list-item ul, .markdown-body ul li .task-list-item ol, .markdown-body ul li .task-list-item ul { margin-top: 0 }
.markdown-body ol ol, .markdown-body ol ul, .markdown-body ul ol, .markdown-body ul ul { margin-top: 3px }
.markdown-body ol li { padding-left: 6px }
@media (max-width: 720px) { .markdown-body h1 { font-size: 24px } .markdown-body h2 { font-size: 20px } .markdown-body h3 { font-size: 18px } }.hljs-comment, .hljs-quote { color: rgba(212, 208, 171, 1) }
.hljs-deletion, .hljs-name, .hljs-regexp, .hljs-selector-class, .hljs-selector-id, .hljs-tag, .hljs-template-variable, .hljs-variable { color: rgba(255, 160, 122, 1) }
.hljs-built_in, .hljs-builtin-name, .hljs-link, .hljs-literal, .hljs-meta, .hljs-number, .hljs-params, .hljs-type { color: rgba(245, 171, 53, 1) }
.hljs-attribute { color: rgba(255, 215, 0, 1) }
.hljs-addition, .hljs-bullet, .hljs-string, .hljs-symbol { color: rgba(171, 227, 56, 1) }
.hljs-section, .hljs-title { color: rgba(0, 224, 224, 1) }
.hljs-keyword, .hljs-selector-tag { color: rgba(220, 198, 224, 1) }
.markdown-body pre, .markdown-body pre>code.hljs { background: rgba(43, 43, 43, 1); color: rgba(248, 248, 242, 1) }
.hljs-emphasis { font-style: italic }
.hljs-strong { font-weight: 700 }
@media screen and (-ms-high-contrast: active) { .hljs-addition, .hljs-attribute, .hljs-built_in, .hljs-builtin-name, .hljs-bullet, .hljs-comment, .hljs-link, .hljs-literal, .hljs-meta, .hljs-number, .hljs-params, .hljs-quote, .hljs-string, .hljs-symbol, .hljs-type { color: rgba(181, 213, 255, 1) } .hljs-keyword, .hljs-selector-tag { font-weight: 700 } }
一、javascript在实现视频通话中的优势
随着互联网技术的不断发展,视频通话正在成为人们日常生活中不可或缺的一部分。为了实现高质量的视频通话,很多开发者选择使用javascript进行开发。javascript作为一种脚本语言,具有灵活、跨平台、易于学习和使用等优势,非常适合用于实现视频通话功能。
二、选择合适的javascript库和开发环境
在开始实现视频通话之前,需要选择合适的javascript库和开发环境。以下是一些流行的javascript库和开发环境:
- WebRTC:WebRTC是一种基于web浏览器的实时通信技术,提供了用于音频、视频和数据通信的API。它是实现视频通话的主要技术之一。
- Socket.io:Socket.io是一种实现WebSocket通信的javascript库,它提供了跨浏览器和平台的实时通信。
- Node.js:Node.js是一种基于javascript的服务器端开发技术,它提供了一些有用的模块和API,例如HTTP模块、WebSocket模块等。
三、搭建视频通话基础架构:建立信令服务器和视频流传输通道
在开始实现视频通话之前,需要搭建视频通话的基础架构,包括信令服务器和视频流传输通道。以下是一些实现视频通话基础架构的代码示例:
1.搭建信令服务器
使用Node.js和Socket.io可以快速地搭建一个简单的信令服务器。以下是一个简单的Node.js服务器代码示例:
const app = require('express')();
const http = require('http').Server(app);
const io = require('socket.io')(http);
io.on('connection', function(socket){
  console.log('a user connected');
});
http.listen(3000, function(){
  console.log('listening on *:3000');
});
2.建立视频流传输通道
使用WebRTC API可以建立视频流传输通道。以下是一个简单的WebRTC代码示例:
const peerConnection = new RTCPeerConnection();
navigator.mediaDevices.getUserMedia({video: true, audio: true})
  .then(stream => {
    stream.getTracks().forEach(track => peerConnection.addTrack(track, stream));
  });
peerConnection.addEventListener('track', event => {
  document.querySelector('video').srcObject = event.streams[0];
});
peerConnection.createOffer().then(offer => peerConnection.setLocalDescription(offer));
四、实现视频流采集和处理:使用WebRTC API获取摄像头视频流并进行加密、解密和压缩
在实现视频通话时,需要实现视频流的采集、加密、解密和压缩等功能。WebRTC API提供了一系列用于实现这些功能的接口和方法。以下是一些实现视频流采集和处理的javascript代码示例:
1.获取视频流
使用WebRTC API可以获取摄像头的视频流。以下是一个简单的获取视频流的代码示例:
navigator.mediaDevices.getUserMedia({video: true, audio: true})
  .then(stream => {
    document.querySelector('video').srcObject = stream;
  });
2.加密和解密视频流
使用WebRTC API可以实现视频流的加密和解密。以下是一个简单的加密和解密视频流的代码示例:
const peerConnection = new RTCPeerConnection();
navigator.mediaDevices.getUserMedia({video: true, audio: true})
  .then(stream => {
    stream.getTracks().forEach(track => peerConnection.addTrack(track, stream));
  });
peerConnection.addEventListener('track', event => {
  const decryptedStream = new MediaStream();
  event.streams[0].getTracks().forEach(track => {
    const encryptedTrack = peerConnection.createEncodedVideoStreamTrack(track);
    const decryptedTrack = peerConnection.createDecodedVideoStreamTrack(track, encryptedTrack);
    decryptedStream.addTrack(decryptedTrack); });
    document.querySelector('video').srcObject = decryptedStream; });
    peerConnection.createOffer().then(offer => peerConnection.setLocalDescription(offer));
3.压缩视频流
使用WebRTC API可以实现视频流的压缩。以下是一个简单的压缩视频流的代码示例:
const peerConnection = new RTCPeerConnection();
navigator.mediaDevices.getUserMedia({video: true, audio: true})
  .then(stream => {
    const videoTrack = stream.getVideoTracks()[0];
    const videoSender = peerConnection.addTrack(videoTrack, stream);
    const videoCodec = RTCRtpSender.getCapabilities('video').codecs.find(codec => codec.mimeType.indexOf('VP9') !== -1);
    const videoParameters = {
      encodings: [{ rid: 'h', maxBitrate: 1000000 }, { rid: 'm', maxBitrate: 300000 }, { rid: 'l', maxBitrate: 100000 }],
      codecPayloadType: videoCodec.payloadType,
      codec: videoCodec.mimeType,
      rtcpFeedback: [{ type: 'nack' }, { type: 'nack', parameter: 'pli' }, { type: 'goog-remb' }]
    };
    videoSender.setParameters({ encodings: videoParameters.encodings });
  });
peerConnection.addEventListener('track', event => {
  document.querySelector('video').srcObject = event.streams[0];
});
peerConnection.createOffer().then(offer => peerConnection.setLocalDescription(offer));
五、实现视频通话界面:使用HTML和CSS创建简洁美观的视频通话界面
为了提供更好的用户体验,需要创建一个简洁美观的视频通话界面。以下是一个简单的HTML和CSS代码示例:
<div class="video-chat">
  <div class="remote-video">
    <video autoplay></video>
  </div>
  <div class="local-video">
    <video autoplay muted></video>
  </div>
</div>
.video-chat {
  display: flex;
  flex-direction: column;
  justify-content: center;
  align-items: center;
  height: 100vh;
}
.remote-video {
  width: 50%;
  margin-bottom: 20px;
}
.local-video {
  width: 25%;
  margin-top: 20px;
}
六、优化视频通话体验:提高视频质量、延迟和稳定性的方法
为了提高视频通话体验,需要采取一些措施来优化视频质量、延迟和稳定性。以下是一些优化视频通话体验的javascript代码示例:
1.自适应码率控制
使用WebRTC API可以实现自适应码率控制,调整视频质量和带宽使用。以下是一个简单的自适应码率控制的代码示例:
const peerConnection = new RTCPeerConnection();
navigator.mediaDevices.getUserMedia({video: true, audio: true})
  .then(stream => {
    const videoTrack = stream.getVideoTracks()[0];
    const videoSender = peerConnection.addTrack(videoTrack, stream);
    const videoCodec = RTCRtpSender.getCapabilities('video').codecs.find(codec => codec.mimeType.indexOf('VP9') !== -1);
    const videoParameters = {
      encodings: [{ rid: 'h', maxBitrate: 1000000 }, { rid: 'm', maxBitrate: 300000 }, { rid: 'l', maxBitrate: 100000 }],
      codecPayloadType: videoCodec.payloadType,
      codec: videoCodec.mimeType,
      rtcpFeedback: [{ type: 'nack' }, { type: 'nack', parameter: 'pli' }, { type: 'goog-remb' }]
    };
    videoSender.setParameters({ encodings: videoParameters.encodings });
  });
peerConnection.addEventListener('track', event => {
  document.querySelector('video').srcObject = event.streams[0];
});
const bandwidth = 1000;
peerConnection.getSenders().forEach(sender => {
  const parameters = sender.getParameters();
  if (!parameters.encodings) {
    parameters.encodings = [{}];
  }
  parameters.encodings[0].maxBitrate = bandwidth * 1000;
  sender.setParameters(parameters);
});
peerConnection.createOffer().then(offer => peerConnection.setLocalDescription(offer));
2.处理网络延迟
使用WebRTC API可以实现网络延迟控制,减少视频通话中的延迟。以下是一个简单的网络延迟控制的代码示例:
const peerConnection = new RTCPeerConnection();
navigator.mediaDevices.getUserMedia({video: true, audio: true})
  .then(stream => {
    const videoTrack = stream.getVideoTracks()[0];
    const videoSender = peerConnection.addTrack(videoTrack, stream);
    const videoCodec = RTCRtpSender.getCapabilities('video').codecs.find(codec => codec.mimeType.indexOf('VP9') !== -1);
    const videoParameters = {
      encodings: [{ rid: 'h', maxBitrate: 1000000 }, { rid: 'm', maxBitrate: 300000 }, { rid: 'l', maxBitrate: 100000 }],
      codecPayloadType: videoCodec.payloadType,
      codec: videoCodec.mimeType,
      rtcpFeedback: [{ type: 'nack' }, { type: 'nack', parameter: 'pli' }, { type: 'goog-remb' }]
    };
    videoSender.setParameters({ encodings: videoParameters.encodings });
  });
peerConnection.addEventListener('track', event => {
  document.querySelector('video').srcObject = event.streams[0];
});
const bandwidth = 1000;
peerConnection.getSenders().forEach(sender => {
  const parameters = sender.getParameters();
  if (!parameters.encodings) {
    parameters.encodings = [{}];
  }
  parameters.encodings[0].maxBitrate = bandwidth * 1000;
  sender.setParameters(parameters);
});
peerConnection.setParameters({
  degradationPreference: 'maintain-framerate',
  transactionId: 'network-conditions'
});
peerConnection.createOffer().then(offer => peerConnection.setLocalDescription(offer));
3.处理网络抖动
使用WebRTC API可以实现网络抖动控制,减少视频通话中的抖动。以下是一个简单的网络抖动控制的代码示例:
const peerConnection = new RTCPeerConnection();
navigator.mediaDevices.getUserMedia({video: true, audio: true})
  .then(stream => {
    const videoTrack = stream.getVideoTracks()[0];
    const videoSender = peerConnection.addTrack(videoTrack, stream);
    const videoCodec = RTCRtpSender.getCapabilities('video').codecs.find(codec => codec.mimeType.indexOf('VP9') !== -1);
    const videoParameters = {
      encodings: [{ rid: 'h', maxBitrate: 1000000 }, { rid: 'm', maxBitrate: 300000 }, { rid: 'l', maxBitrate: 100000 }],
      codecPayloadType: videoCodec.payloadType,
      codec: videoCodec.mimeType,
      rtcpFeedback: [{ type: 'nack' }, { type: 'nack', parameter: 'pli' }, { type: 'goog-remb' }]
    };
    videoSender.setParameters({ encodings: videoParameters.encodings });
  });
peerConnection.addEventListener('track', event => {
  document.querySelector('video').srcObject = event.streams[0];
});
const bandwidth = 1000;
peerConnection.getSenders().forEach(sender => {
  const parameters = sender.getParameters();
  if (!parameters.encodings) {
    parameters.encodings = [{}];
  }
  parameters.encodings[0].maxBitrate = bandwidth * 1000;
  sender.setParameters(parameters);
});
peerConnection.setParameters({
  transactionId: 'network-conditions'
});
peerConnection.createOffer().then(offer => peerConnection.setLocalDescription(offer));
七、总结
在本文中,我们介绍了如何使用javascript实现高质量视频通话的技巧和方法。通过选择合适的javascript库和开发环境、搭建视频通话基础架构、实现视频流采集和处理、创建简洁美观的视频通话界面、优化视频通话体验等步骤,可以实现高质量、稳定和流畅的视频通话;希望本文对各位掘友有所帮助。
实现高质量视频通话的javascript技巧与方法的更多相关文章
- 《编写高质量代码改善JavaScript程序的188个建议》读书笔记
		逗号运算符比较怪异.如 var a =(1,2,3,4);alert(a);// 4 var a = 1,2,3,4;//报错 注意a++和++a的差别,变量在参与运算中不断地变化.v ... 
- [已读]编写高质量代码 改善JavaScript程序的188个建议
		吐槽一万遍,买的最后悔的一本,没有之一,大量篇幅抄袭<高性能javascript>,我记得还有部分抄袭<javascript精粹>,<javascript模式>有没 ... 
- (第一章)改善JavaScript,编写高质量代码。
		根据<编写高质量代码改善JavaScript程序的188个建议>这本书,来记录我目前所了解的建议方式. 建议1:警惕Unicode乱码 根据ECMA标准规定JavaScript语言可以使用 ... 
- 每周一书-编写高质量代码:改善C程序代码的125个建议
		首先说明,本周活动有效时间为2016年8月28日到2016年9月4日.本周为大家送出的书是由机械工业出版社出版,马伟编著的<编写高质量代码:改善C程序代码的125个建议>. 编辑推荐 10 ... 
- 第二章 设计高质量的React组件
		第二章 设计高质量的React组件 高质量React组件的原则和方法: 划分组件边界的原则: React组件的数据种类: React组件的生命周期. 2.1 易于维护组件的设计要素 1.高内聚:指的是 ... 
- 编写高质量JavaScript代码的68个有效方法
		简介: <Effective JavaScript:编写高质量JavaScript代码的68个有效方法>共分为7章,分别涵盖JavaScript的不同主题.第1章主要讲述最基本的主题,如版 ... 
- 如何写出高质量的JavaScript代码
		优秀的Stoyan Stefanov在他的新书中(<Javascript Patterns>)介绍了很多编写高质量代码的技巧,比如避免使用全局变量,使用单一的var关键字,循环式预存长度等 ... 
- 深入理解JavaScript系列(1):编写高质量JavaScript代码的基本要点
		深入理解JavaScript系列(1):编写高质量JavaScript代码的基本要点 2011-12-28 23:00 by 汤姆大叔, 139489 阅读, 119 评论, 收藏, 编辑 才华横溢的 ... 
- JavaScript手札:《编写高质量JS代码的68个有效方法》(一)(1~5)
		编写高质量JS代码的68个有效方法(一) *:first-child { margin-top: 0 !important; } body>*:last-child { margin-botto ... 
- 高质量JavaScript代码书写基本要点
		翻译-高质量JavaScript代码书写基本要点 by zhangxinxu from http://www.zhangxinxu.com本文地址:http://www.zhangxinxu.com/ ... 
随机推荐
- [THUSC2015] 异或运算 题解
			学到新思路了:求解 \(k\) 大值时,可以将所有元素放一块一起跑. 考虑到 \(n,q\) 奇小无匹,我们便可以制造一个 \(O(qn\log V)\) 的代码. 那么对于我们不想在时间复杂度中出现 ... 
- MySQL - [17] Oracle、SQLServer、MySQL数据类型对比
			题记部分 一.数据类型对比 对应关系 (1)整数类型 Oracle的NUMBER(*,0) 对应 SQL Server的INT 和 MySQL的INT Oracle的BIGINT 可能需要映射到SQL ... 
- 帝国CMS下iframe标签无法引入视频,ueditor编辑器中html标签无法显示问题,设置ueditor默认行高为1.75
			问题描述: 1.帝国cms后台添加优酷视频,使用到iframe,富文本编辑器中使用iframe引入视频后检查发现html代码未出现iframe字样,排查后发现为ueditor限制过滤了部分html代码 ... 
- 学习理论:单阶段代理损失的(H, R) - 一致界证明
			1 导引 我们在上一篇博客<学习理论:预测器-拒绝器多分类弃权学习>中介绍了弃权学习的基本概念和方法,其中包括了下列针对多分类问题的单阶段预测器-拒绝器弃权损失\(L_{\text{abs ... 
- 【CIM信息整合】关于三维建筑模型
			还是无暇细细检索并总结列出有逻辑的明确表述,以下很多地方都是人云亦云的复制,自己也没太搞清 1.5 三维建筑模型 CIM中三维建筑模型主要表达建(构)筑物的空间位置.几何形态及外观效果等. 在建筑相关 ... 
- 如何解决ubuntu安装第三方deb出现的read unknown VMA问题(SUID sandbox配置问题)
			前言 众所不周知,ubuntu的snap有些让人无语凝噎的问题,比方说如果你在Ubuntu 24.04 LTS通过它安装vscode,恭喜你,你无法在vscode中输入中文,缘由不明,又或者对于str ... 
- mac强制关闭程序
			使用快捷键:Command+Option+Esc 来打开"强制退出应用程序"的窗口,然后选中你需要退出的程序,再点右下方的"强制退出"即可. 
- centos安装php环境
			安装 PHP 所需扩展 yum install libxml2 libxml2-devel openssl openssl-devel bzip2 bzip2-devel libcurl libcur ... 
- phpstudy安装redis扩展 windows安装redis扩展
			1.首先查看自己本地环境的php版本信息: 执行phpinfo(),显示php相关信息 2.下载所需的扩展 php redis扩展下载地址 windows上,php开启扩展需要下载ddl 这里要注意的 ... 
- Thinkphp3.2 PHPMailer 发送邮件
			第一步 :下载附件PHPMailer解压到ThinkPHP\Library\Vendor 第二步:在Common文件夹中的公共函数function.php中写一个发送邮件的函数, 这样可以在项目任意位 ... 
