28、使用cookie、iframe进行客户端与服务器端的通信
关键字: cookie iframe
cookie
cookie是第一个JavaScript可以利用客户端——服务器端之间的交互手段。浏览器向服务器发送请求时,为这个服务器存储的cookie会与其他信息一起发送到服务器。这使得JavaScript 可以在客户端设置一个cookie,之后服务器端就能够读取它了。
cookie的组成
?名称——每个cookie由一个唯一的名称表示。名称不区分大小写,但最好是区分,因为有些服务器端软件是这样的。
?值——保存在cookie中的字符串值。这个值在存储之前必须用encodeURIComponent()对其进行编码,以免丢失数据或占用了cookie。注:名称和值一起的字节数不能超4KB。
?域——出于安全考虑,网站不能访问由其他域创建的cookie。创建cookie后,域的信息会作为cookie的一部分存储起来。不过,虽然这不常见,还是可以覆盖这个设置以允许另一个网站访问这个cookie的。
?路径——另一个cookie的安全特征,路径限制了对WEB服务器上的特定目录的访问。例如:可指定cookie只能从https://www.doczj.com/doc/1c11832027.html,/books中访问,这样就不能访问https://www.doczj.com/doc/1c11832027.html,/上的网页了,尽管都在同一个域中。
?失效日期——cookie何时应该被删除。默认情况下,关闭浏览器时,即将cookie删除;不过,也可以自己设置删除时间。这个值是个GMT格式的日期(可以使用Date 对象的toGMTString()方法),用于指定应该删除cookie的准确时间。因此,cookie可在浏览器关闭后依然保存在用户的机上。如果你设置的失效日期是以前时间,则cookie 被立即删除。
?安全标志——一个true/false值,用于表示cookie是否只能从安全网站(使用SSL 和https协议的网站)中访问。可以将这个值设置为true以提供加强的保护,进而确保cookie不被其他网站访问。
其他安全限制
为确保cookie不能恶意使用,浏览器还对cookie的使用进行了一些限制:?每个域最多只能在一台用户的机器上存储20个cookie。
?每个cookie的总大小不能超过4096字节。
?一台用户的机器上的cookie的总数不能超过300个。
另外,一些新的浏览器还对cookie进行了严格控制,可以让用户阻止所有的cookie,阻止某些未知网站的cookie或者在创建cookie时进行提示。
JavaScript中的cookie
document对象有个cookie属性,是包含所有给定页面可访问的cookie的字符串。cookie属性很特别,因为将这个cookie属性设置为新值只会改变对页面可访问的cookie,并不会真
正改变cookie(属性)本身。
要创建一个cookie,必须按照下面的格式创建字符串:
cookie_name=cookie_value; expires=expiration_time; path=domain_path; domain=domain_name;secure
只有第一部分对设置cookie是必需的,其他部分都可以省。然后将这个字符串复制给document.cookie属性,即可创建cookie。如:
document.cookie='name=Nicholas';
document.cookie='book=' + encodeURIComponent('Professional JavaScript');
读取document.cookie的值即可以访问这些cookie,以及所有其他可以从给定页面访问的cookie。如果在运行上面两行代码后显示document.cookie 的值,则出现'name=Nicholas;book=Professional%20JavaScript'。注,即使创建了其他的cookie属性,如失效时间,document.cookie也只返回每个cookie的名称和值,并用分号来分隔这此cookie。因为创建和读取cookie均需记住它的格式,大部分开发人员用函数来处理这些细节。cookie 操作函数:
Js代码
1//创建cookie
2function setCookie(sName, sV alue, oExpires, sPath, sDomain, bSecure) {
3var sCookie = sName + "=" + encodeURIComponent(sV alue);
4//除sName, sV alue外,其他参考可选,所以使用前要判断是否传入
5if (oExpires) {
6//时间要是GMT格式
7sCookie += "; expires=" + oExpires.toGMTString();
8}
9if (sPath) {
10sCookie += "; path=" + sPath;
11}
12if (sDomain) {
13sCookie += "; domain=" + sDomain;
14}
15if (bSecure) {
16sCookie += "; secure";
17}
18document.cookie = sCookie;
19}
20//通过传入的名字获取cookie的值
21function getCookie(sName) {
22var sRE = "(?:; )?" + sName + "=([^;]*);?";
23var oRE = new RegExp(sRE);
24if (oRE.test(document.cookie)) {
25return decodeURIComponent(RegExp["$1"]);
26} else {
27return null;
28}
29}
30//删除cookie
31function deleteCookie(sName, sPath, sDomain) {
32//删除cookie必须给出与创建它时一样的路径和域信息。
33var sCookie = sName + "=; expires=" + (new Date(0)).toGMTString();
34if (sPath) {
35sCookie += "; path=" + sPath;
36}
37if (sDomain) {
38sCookie += "; domain=" + sDomain;
39}
40document.cookie = sCookie;
41}
Html代码
42
43
44
45
46//引用上面脚本...
47
48
49
50
51alert("Setting cookies...");
52setCookie("name", "Nicholas");
53setCookie("book", "Professional JavaScript");
54
55
56alert("The value of cookie 'name' is " + getCookie("name"));
57alert("The value of cookie 'book' is " + getCookie("book"));
58
59alert("Deleting cookies...");
60deleteCookie("name");
61deleteCookie("book");
62
63alert("The value of cookie 'name' is " + getCookie("name"));
64alert("The value of cookie 'book' is " + getCookie("book"));
65
66
67
68
69
jsp服务器端的cookie
先可以通过request对象的getCookies()方法,返回一组Cookies对象,每个对象有以下方法:?getComment()——返回描述cookie目的的注释,如果cookie没有注释则为null
?getDomain()——返回cookie设置的域名称
?getMaxAge()——返回cookie的最大寿命,单位为称;默认情况下为-1,表示cookie 会持续到浏览器关闭
?getName()——返回cookie的名称
?getPath()——返回服务器为浏览器提供的返回cookie的路径
?getSecure()——如果浏览器只通过安全协议发送cookie则为true,否则为false
?getV alue()——返回cookie的值
?getV ersion()——返回与cookie相关的协议的版本
?setComment(String perpose)——指定描述cookie目的的注释
?setDomain(String pattern)——指定给出cookie的域
?setMaxAge(int expiry)——指定cookie的最大寿命,单位为秒
?setPath(String uri)——指定浏览器返回cookie的路径
?setSecure(boolean flag)——指示浏览器cookie是否只能通过安全协议,如HTTPS或者SSL来传送
?setV alue(String new V alue)——创建cookie后,为cookie指定一个值
?setV ersion(int v)——指定与cookie相关的协议的版本
Java代码
70<%!//读取cookie
71public static Cookie getCookie(HttpServletRequest request, String name) {
72Cookie[] cookies = request.getCookies();
73if (cookies != null) {
74for (int i = 0; i < cookies.length; i++) {
75if (cookies[i].getName().equals(name)) {
76return cookies[i];
77}
78}
79}
80return null;
81}%>
82
83<%
84//创建cookie
85Cookie nameCookie = new Cookie("name", "Nicholas");
86nameCookie.setDomain("https://www.doczj.com/doc/1c11832027.html,");
87nameCookie.setPath("/books");
88response.addCookie(nameCookie);
89%>
90
91<%
92//删除cookie
93Cookie cookieToDel = getCookie(request, "name");
94cookieToDel.setMaxAge(0);
95response.addCookie(cookieToDel);
96%>
在客户端与服务器端之间传递cookie
与Javascript的交互,常常是服务器在发送响应前创建一个cookie(也许在servlet或者其他运行在服务器上的应用中)。然后带有胜于获取cookie值的Javascript的页面就会载入。
例如,假设你在网站上创建一个反馈表单,上面要求用户输入名字、电子邮件地址及反馈消息。你可以让用户将他们的名字与电子邮件地址保存下来,以便他们下次访问并要留下反馈时,可以无需再输入这两个字段。思路如下:
首先通过在服务器端创建相应的Cookie,并通过response回送给客户端保存下来(如果浏览器支持Cookie会自动保存),然后,将来某天用户进入上面那个面页时,通过JavaScript 调用Cookie读取相应的值。
其实创建与读取Cookie过程完全可以在服务上全部实现,不需要由客户端读取。iframe
可以通过iframe来模拟ajax的后台请求,即在不刷新面页的情况下能与服务器进行通信。主页面main.htm:
Html代码
97
98
99
100
101
102//全局变量,在不刷新情况下可以重复请求而不用多次创建
103var oHiddenFrame = null;
104function getServerInfo() {
105//如果还没有创建过,则创建
106if (oHiddenFrame == null) {
107//使用DOM的createElement创建框架对象
108oHiddenFrame = document.createElement("iframe");
109/*必须同时将name与id设置为hiddenFrame,以保证在大部分浏览
110中都能正常工作(有些使用name,有些使用id)*/
https://www.doczj.com/doc/1c11832027.html, = "hiddenFrame";
112oHiddenFrame.id = "hiddenFrame";
113//高度与宽度都必须为0
114oHiddenFrame.style.height = "0px";
115oHiddenFrame.style.width = "0px";
116//位置设置为绝对
117oHiddenFrame.style.position = "absolute";
118//设置为不可见
119oHiddenFrame.style.visibility = "hidden";
120//最后将创建的框架添加到body对象中去
121document.body.appendChild(oHiddenFrame);
122}
123/*iframe创建好并添加后,大部分浏览器(尤其是Mozilla与Opera)会需要一小段时间(毫秒级)
124来认出新的框架并将其添加到帧集合中。考虑此问题,应该用setTimeout()函数使发送请求的
125操作等待10毫秒。*/
126setTimeout(function () {
127//为document中的hiddenFrame框架设置要请求的URL
128frames["hiddenFrame"].location.href = "iframeResponse.jsp";
129}, 10);
130}
131//处理服务器响应消息,供响应页面HiddenFrameExampleCom2.htm中脚本回调
132function handleResponse(sResponseText) {
133alert("服务器回应的响应信息: " + sResponseText);
134}
135
136
137
138
139
140
点击按键通过iframe隐藏框架向服务器发送请求.
141 142
143
响应页面iframeResponse.jsp实现:
Java代码
144
145
146
147
148/*在服务器处理完后,将处理的数据送到此页面中,再由此页面将数据149进一步的传送给主页面创建iframe框架的页面),这一过程通过触发window的onload事件调用主页面的handleResponse函数来完成*/
150window.onload = function () {
151parent.handleResponse('
<%=request.getAttribute("responseData")%>');
152};
153
154
155
156
157
上传文件进度条实现
上传文件时如果要实现真实的进度条,则要借助于iframe框架。因为普通的文件上传方式会在提交后,页面会变白且处于等待服务器相应,所以上传时不能采用普通方式。可以使用iframe在上传时不出现白页面,且可以在上传文件的当前页面实现进度条。
上传页面代码大体如下:
Html代码
158...
159
160
161
162
168
169...
170
171
175
209 |
212...
设置进度条样式相关js:
Js代码
213//...
214//设置进度条宽度
215$("percent_div").style.width = percent + "%";
216//设置进度条值
217$("percent").innerText = percent + "%";
218
219//进度条样式设置
220if (percent != "0") {
221setBarStyle();
222} else {
223intialBarStyle();
224}
225//...
226/**
227* method:初始化进度条样式
228*/
229function intialBarStyle() {
230var percent_div = $("percent_div");
231percent_div.style.border = "0px solid #000";
232percent_div.style.margin = "2px";
233percent_div.style.fontSize = "11pt";
234percent_div.style.width = "0%";
235}
236/**
237* method:进度非零时进度条样式
238*/
239function setBarStyle() {
240var percent_div = $("percent_div");
241percent_div.style.heigth = "11px";
242percent_div.style.background = "#178399";
243percent_div.style.border = "1px solid #000";
244percent_div.style.margin = "2px";
245percent_div.style.padding = "0";
246percent_div.style.fontSize = "10pt";
247}
其他的工作是就是用ajax与服务器进行异步通信,获取进度值而已。
既然这里说到了上传,下面随便贴一下下载的方面要注意的:
Java代码
248<%
249response.setContentType("application/octet-stream;charset=UTF-8");
250
251String logType = request.getParameter(Const.PARA_LOGTYPE);
252
253String currDateStr = Util.getDateTimeStr(Util.getCurrDate(), Const.YYYYMMDDHHMMSS);
254String filePrefix = null;
255if (Const.SUCC_LOG_DETAIL.equals(logType)) {
256filePrefix = "SUCC_LOG_";
257} else if (Const.UNSUCC_LOG_DETAIL.equals(logType)) {
258filePrefix = "FAIL_LOG_";
259} else if (Const.UNDONE_LOG_DETAIL.equals(logType)) {
260filePrefix = "UNDONE_LOG_";
261}
262
263String paramLook = request.getParameter("look");
264String paramSave = request.getParameter("save");
265
266String method = null;
267if (paramLook != null) {
268method = "inline";
269} else {
270method = "attachment";
271}
272response.setHeader("Content-Disposition", method + ";filename=" + filePrefix + currDateStr + ".txt");
273String logStr = (String) request.getAttribute(Const.KEY_OPRNFOSTR);
274response.setHeader("Accept-Ranges", "bytes");
275response.setHeader("Accept-Length", (new Integer(logStr.getBytes().length)).toString());
276System.out.println(logStr);
277PrintWriter os = response.getWriter();
278os.print(logStr);
279os.flush();
280os.close();
281%>
#include "" #include <> #include
package cn pab import java.util.List; import java.util.Scanner; import cn.pab.dao.PersonDao; import cn.pab.dao.TypeDao; import https://www.doczj.com/doc/1c11832027.html,erDao; import cn.pab.dao.impl.PersonDaoImpl; import cn.pab.dao.impl.TypeDaoImpl; import https://www.doczj.com/doc/1c11832027.html,erDaoImpl; import cn.pab.entity.Person; import cn.pab.entity.Type; /** * 业务类 */ public class PABmanager { /** * 系统启动 */ public static void main(String[] args) { Scanner input = new Scanner(System.in); UserDao userDao = new UserDaoImpl(); TypeDao typeDao = new TypeDaoImpl(); PersonDao personDao = new PersonDaoImpl(); System.out.println( " System.out.print("\ n 欢迎使用个人通讯录管理系统**********"); 请选择操作(1. 系统登录 2. 密码修改 3. 取消):"); String in = input.next(); if ("1".equals(in)) { boolean islogin = userDao.login(); if(islogin){ System.out.println("******** *** 成功登录个人通讯录管理系统 nm\、\ ? //System.out.print(" \n 退出系统):"); }else{ System.out.println(" System.exit(-1); } }else if ("2".equals(in)) { 请选择操作(1. 类别管理 2. 联系人管理 3. 用户名或密码错误,不能登录!");
关于客户端与数据库服务器端的时间同步问题 这是一个做C/S的管理软件开发时经常被忽略的问题,客户端的时间与服务器的时间如果有偏差,数据统计、报表等等肯定会有“意外”的情况发生。 意图很简单:从数据库服务器获取到时间,根据这个时间修改当前客户端电脑时间。 用Sql的函数getdate(),是比较容易的。 我们是基于dotnet4.0、EntityFramework开发软件,所以希望用ESQL的方式获取数据库服务器的时间,但昨天折腾了半天,还没搞定。 如果有哪位同学已经解决了这个问题,希望能指点一下! 暂时解决,之所以说是暂时,是因为并没有用Esql的方式,而是用T-Sql的方式。 以下是我的过程: System.Data.EntityClient.EntityConnection 这个是实体概念模型与数据源的连接,继承自DbConnection 在这个连接下CreateCommand(),就需要写Esql语句,我的语句是"SELECT VALUE CurrentDateTime()",却是语法错误。翻遍了手册和网络查询,没有任何有用的结果。 但在这个连接对象下有一个属性StoreConnection,返回的是Sql方式的连接,在这个下面CreateCommand(),可以写T-Sql语句,我的语句是"SELECT getdate()",运行成功。
以上是程序代码例子: //与数据库服务器的时间进行同步 System.Data.EntityClient.EntityConnection conn = (System.D ata.EntityClient.EntityConnection)Blemployee.myData.Conne ction ; IDbConnection conn0=conn.StoreConnection; IDbCommand comm =conn0.CreateCommand(); //https://www.doczj.com/doc/1c11832027.html,mandText = "SELECT VALUE CurrentDateTime()"; https://www.doczj.com/doc/1c11832027.html,mandText = "SELECT getdate()"; https://www.doczj.com/doc/1c11832027.html,mandType = CommandType.Text; if (comm.Connection.State != ConnectionState.Open) comm.Connection.Open(); object tt= comm.ExecuteScalar(); DateTime sqlDT = Convert.ToDateTime(tt); SetLocalTime(sqlDT); //设置本机时间
客户端与服务器端交互原理 经常看到HTTP客户端与服务器端交互原理的各种版本的文章,但是专业术语太多,且流程过于复杂,不容易消化。于是就按照在Servlet 里面的内容大致做了一些穿插。本来连Tomcat容器和Servlet的生命周期也准备在这里一起写的,但怕过于庞大,于是就简单的引用了一些Servlet对象。这样的一个整个流程看下来,相信至少在理解HTTP协议和request和response是如何完成从请求到生成响应结果回发的。在后续的一些文章里会专门讲一讲Tomcat和Servlet 是如何处理请求和完成响应的,更多的是说明Servlet的生命周期。 HTTP介绍 1. HTTP是一种超文本传送协议(HyperText Transfer Protocol),是一套计算机在网络中通信的一种规则。在TCP/IP体系结构中,HTTP属于应用层协议,位于TCP/IP协议的顶层。 2. HTTP是一种无状态的协议,意思是指在Web浏览器(客户端)和Web 服务器之间不需要建立持久的连接。整个过程就是当一个客户端向服务器端发送一个请求(request),然后Web服务器返回一个响应(respo nse),之后连接就关闭了,在服务端此时是没有保留连接的信息。 3. HTTP遵循请求/响应(request/response)模型的,所有的通信交互都被构造在一套请求和响应模型中。 4. 浏览Web时,浏览器通过HTTP协议与Web服务器交换信息,Web服务器向Web 浏览器返回的文件都有与之相关的类型,这些信息类型的格式由 MIME 定义。 HTTP定义的事务处理由以下四步组成: 1. 建立连接。 2?客户端发送HTTP请求头。 3. 服务器端响应生成结果回发。 4. 服务器端关闭连接,客户端解析回发响应头,恢复页面。
源代码: #include "stdio.h" #include "stdlib.h" #include "string.h" #include "conio.h" #include "stdlib.h" #define null 0 struct record { char name[20]; char phone[20]; char adress[40]; char postcode[10]; char e_mail[30]; }student[500]; struct LinkList { struct record US; struct LinkList *next; }a; struct LinkList *head=null; int num=0; FILE *fp; int menu_select(); int adduser(); int list(); int search(); int display(); int add(); int listbyname(); int dele(); int save(); int exit(); void main() {
system("cls"); for(;;) { switch(menu_select()) { case 0:adduser();break; case 1:list();break; case 2:search();break; case 3:display();break; case 4:add();break; case 5:listbyname();break; case 6:dele();break; case 7:save();break; case 8:exit(0); } } } menu_select() { char s[80]; int a; printf("*_* press any key enter menu! *_* \n"); getch(); system("cls"); printf("\t\t********************MENU*********************\n\n"); printf("\t\t 0. 输入记录\n"); printf("\t\t 1. 显示记录\n"); printf("\t\t 2. 按姓名查找\n"); printf("\t\t 3. 按电话号码查找\n"); printf("\t\t 4. 插入记录\n"); printf("\t\t 5. 按姓名排序\n"); printf("\t\t 6. 删除记录\n"); printf("\t\t 7. 记录保存文件\n"); printf("\t\t 8. Quit\n"); printf("\t\t***********************************************\n"); do{ printf("\n Enter you choice(0~11):"); scanf("%s",s); a=atoi(s); } while (a<0||a>11); return a;
计算机科学与技术系 实验(项目)报告 一、基本信息 二、目的与要求 目的: 通过实验掌握Linux环境下基于TCP套接字编程的基本步骤和方法。 要求: 1.掌握网络编程的基本框架; 2.了解客户端、服务器的基本概念; 3.掌握TCP套接字编程的基本流程及程序的调试方法。 三、完成实验(项目)过程和效果 内容: 1.程序结构设计; 2.程序数据设计; 3.程序基本流程; 4.编码、调试及运行。
server代码: client代码:
步骤: 1.程序结构设计 1)服务器端结构 先启动,进行socket、bind、listen、accept、一系列准备工作,然后如果没有建立连接,则服务器处于阻塞状态,等待客户端连接。 2)客户端结构。 客户端启动后,首先向服务器端发起建立连接的请求,得到同意后,连接建立成功,客户端和服务器端开始进行数据通信,客户机完成通信工程后释放链接,关闭套接字。
2.程序数据设计 1)服务器端相关数据: 监听套接字描述符、连接套接字描述符int sockfd,client_fd; 服务器端地址struct sockaddr_in my_addr 客户端地址struct sockaddr_in remote_addr 2)客户端相关数据 套接字描述符int sockfd; 服务器端地址结构struct sockaddr_in serv_addr; 保存接收到的数据char buf[MAXDATASIZE]; 接收的数据字符int recvbytes; 3.程序基本流程图。 四、知识应用以及经验总结 1.编写UDP通信程序?
服务器端代码: using https://www.doczj.com/doc/1c11832027.html,; using https://www.doczj.com/doc/1c11832027.html,.Sockets; Static void Main(string[] args){ Socket serverSocket=new Socket(AddressFamily.InterNetWork,SocketType.Stream,ProtocalTy pe.TCP); //new一个Socket对象,注意这里用的是流式Socket(针对于面向连接的TCP服务应用)而不是数据报式Socket(针对于面向无连接的UDP服务应用)。 IPAddress serverIP=IPAddress.Parse("127.0.0.1"); int port=2112; IPEndPoint ipEndPoint=new IPEndPoint(serverIP,port);//网络节点对象 serverSocket.Bind(ipEndPoint);//将结点绑定到套接字上 serverSocket.Listen(10);//设置连接队列的最大长度,可根据服务器的性能,可以设置更大程度。 Console.WriteLine("服务器已就绪准备客户端连接。。。。"); while(true){//循环监听端口,得到客户端连接 Socket socket=serverSocket.Accept();//当有客户端连接时,就产生一个socket实例 SessionServer sserver=new SessionServer(socket);//将socket实例传入到消息处理类中 Thread t=new Thread(sserver.GetClientMsg);//当有一个客户端连接,就启动一个线程来处理此客户端的消息 t.Start();
实验六基于TCP/IP的网络编程 1 实验目的 MFC提供的关于网络应用的类CSocket是一个比较高级的封装,使用它编制出属于自己的网络应用程序,可以编一个属于自己的网络通讯软件。通过这个实验,同学们也可以增进对于TCP/IP协议的理解。 2 实验内容 基于TCP/IP的通信基本上都是利用SOCKET套接字进行数据通讯,程序一般分为服务器端和用户端两部分。设计思路(VC6.0下): 第一部分服务器端 一、创建服务器套接字(create)。 二、服务器套接字进行信息绑定(bind),并开始监听连接(listen)。 三、接受来自用户端的连接请求(accept)。 四、开始数据传输(send/receive)。 五、关闭套接字(closesocket)。 第二部分客户端 一、创建客户套接字(create)。 二、与远程服务器进行连接(connect),如被接受则创建接收进程。 三、开始数据传输(send/receive)。 四、关闭套接字(closesocket)。 CSocket的编程步骤:(注意我们一定要在创建MFC程序第二步的时候选上Windows Socket 选项,其中ServerSocket是服务器端用到的,ClientSocket是客户端用的。) (1)构造CSocket对象,如下例: CSocket ServerSocket; CSocket ClientSocket; (2)CSocket对象的Create函数用来创建Windows Socket,Create()函数会自行调用Bind()函数将此Socket绑定到指定的地址上面。如下例: ServerSocket.Create(823); //服务器端需要指定一个端口号,我们用823。ClientSocket.Create(); //客户端不用指定端口号。 (3)现在已经创建完基本的Socket对象了,现在我们来启动它,对于服务器端,我们需要这个Socket不停的监听是否有来自于网络上的连接请求,如下例: ServerSocket.Listen(5);//参数5是表示我们的待处理Socket队列中最多能有几个Socket。(4)对于客户端我们就要实行连接了,具体实现如下例: ClientSocket.Connect(CString SerAddress,Unsinged int SerPort);//其中SerAddress是服务器的IP地址,SerPort是端口号。 (5)服务器是怎么来接受这份连接的呢?它会进一步调用Accept(ReceiveSocket)来接收它,而此时服务器端还须建立一个新的CSocket对象,用它来和客户端进行交流。如下例:CSocket ReceiveSocket; ServerSocket.Accept(ReceiveSocket); (6)如果想在两个程序之间接收或发送信息,MFC也提供了相应的函数。 (7)代码 package test.socket3; import java.awt.event.ActionEvent; import java.awt.event.ActionListener;
服务器与移动客户端通信设计 软件的通信方式是开发过程中的重要一环。智能手机的快速发展,使得手机不仅作为一般通讯工具,更进一步成为一款便携式移动互联网终端。通常来说,Android操作系统的手机使用Android系统自身集成的HttpClient直接访问网络资源[35]。 服务器MySQL 图4.7 客户端与数据库通信方式示意图 Fig.4.7 Communication mode between client and database HttpClient是一种HTTP协议的支撑工具包,它能够为客户端提供一系列高效、便捷、多功能的编程工具,且能够支持最新的HTTP协议,操作简单。对于HTTP连接中的各种复杂问题都能够予以有效的解决。如上图4.7所示,HttpClient 实现HTTP协议的方法,主要是GET与POST两种方法。 1.GET方法。HTTP协议的GET方法即利用HttpClient向客户端发送GET 请求,这一过程一般用来进行客户端的信息查询操作,例如,在本次客户端中, 其可以用于 检修故障信息、零部件信息以及检修工单信息的查询。具体的实现步骤有以下几 步[36]: 1) 创建HttpClient实例;2) 创建HttpPost实例。 3) 将需要发送的GET请求参数直接连接至URL地址中,并用“?”将参 数与地址隔开,每个参数之间用“&”隔开,若有需要额外添加的参数,可以选 择调用setParams()的方式来进行添加。 4) 调用第一步创建的HttpClient实例中的execute()方法来执行第二步创建 的HttpGet实例,并读取Response对象。 5) 采取调用getAllHeaders()、getHeaders(String name)等方式获取服务器响应,并释放连接,无论上述第四步的执行过程是否成功,都必须释放连接,允许 用户获得服务器的响应内容。 2.POST方法。HTTP协议的POST方法即利用HttpClient向客户端发送POST 请求,该请求过程一般用来进行客户端的信息修改操作,例如,在本课题所设计 的客户端中,其可以用于对登录、密码等修改等操作。其具体的实现过程也分为 五个步骤:
服务器端界面 服务器端代码: using System; using System.Collections.Generic; using https://www.doczj.com/doc/1c11832027.html,ponentModel; using System.Data; using System.Drawing; using System.Text; using System.Windows.Forms; using https://www.doczj.com/doc/1c11832027.html,.Sockets; using System.Threading; using System.IO; using https://www.doczj.com/doc/1c11832027.html,; using System.Collections; namespace IMS.Server { public partial class Server : Form { TcpListener myListener; TcpClient tcpClient = new TcpClient(); Thread mythread; NetworkStream ns;
public Server() { InitializeComponent(); } private void Server_Load(object sender, EventArgs e) { Control.CheckForIllegalCrossThreadCalls = false; mythread = new Thread(new ThreadStart(receive)); mythread.IsBackground = true; mythread.Start(); } private void receive() { myListener = new TcpListener(IPAddress.Parse("192.168.1.106"), 8080); myListener.Start(); tcpClient = myListener.AcceptTcpClient(); while (true) { string rec = ""; ns = tcpClient.GetStream(); byte[] bytes = new byte[1024]; ns.Read(bytes,0,bytes.Length); rec = Encoding.Unicode.GetString(bytes); richTextBox1.Text = rec; ns.Flush(); } } private void btnSend_Click(object sender, EventArgs e) { try { ns = tcpClient.GetStream(); byte[] bytes = new byte[1024]; // bytes = Encoding.Unicode.GetBytes(sendmsg); bytes = Encoding.Unicode.GetBytes(richTextBox1.Text +"\r\n" + "服务器说:" + richTextBox2.Text);
Linux网络编程-基础知识(1) 1. Linux网络知识介绍 1.1 客户端程序和服务端程序 网络程序和普通的程序有一个最大的区别是网络程序是由两个部分组成的--客户端和服务器端. 网络程序是先有服务器程序启动,等待客户端的程序运行并建立连接. 一般的来说是服务端的程序在一个端口上监听,直到有一个客户端的程序发来了请求. 1.2 常用的命令 由于网络程序是有两个部分组成,所以在调试的时候比较麻烦,为此我们有必要知道一些常用的网络命令 netstat 命令netstat是用来显示网络的连接,路由表和接口统计等网络的信息. netstat有许多的选项我们常用的选项是-an 用来显示详细的网络状态.至于其它的选项我们可以使用帮助手册获得详细的情况. telnet telnet是一个用来远程控制的程序,但是我们完全可以用这个程序来调试我们的服务端程序的. 比如我们的服务器程序在监听8888端口,我们可以用telnet localhost 8888来查看服务端的状况. 1.3 TCP/UDP介绍 TCP(Transfer Control Protocol)传输控制协议是一种面向连接的协议, 当我们的网络程序使用这个协议的时候,网络可以保证我们的客户端和服务端的连接是可靠的,安全的. UDP(User Datagram Protocol)用户数据报协议是一种非面向连接的协议, 这种协议并不能保证我们的网络程序的连接是可靠的,所以我们现在编写的程序一般是采用TCP协议的. Linux网络编程-简单的客户端和服务器通讯程序开发入门(2)简介: 本文详细介绍了Linux下B/S结构的客户端服务器通讯程序的开发入门, 其中对重要的网络函数和结构体作了详细的说明和分析, 最后给出一个简单的客户端和服务器通讯程序示例以加深理解。 2. 初等网络函数介绍(TCP) Linux系统是通过提供套接字(socket)来进行网络编程的.网络程序通过socket和其它几个函数的调用, 会返回一个通讯的文件描述符,我们可以将这个描述符看成普通的文件的描述符来操作, 这就是linux的设备无关性的好处.我们可以通过向描述符读写操作实现网络之间的数据交流. 2.1 socket
SimpleChatServer.java package test.chatclient; import java.io.*; import https://www.doczj.com/doc/1c11832027.html,.*; import java.util.*; public class SimpleChatServer { ArrayList
个人通讯录管理系统C语言源程序优秀版 Last revision date: 13 December 2020.
#i n c l u d e/*头文件*/ #include
import https://www.doczj.com/doc/1c11832027.html,.*; import java.io.*; import java.util.Calendar; import java.awt.*; import java.awt.event.*; class EchoClient extends Frame implements ActionListener,Runnable{ Thread t; static Calendar T; static int H; static int M; static int S; static String l; private Button b1 = new Button("发送"); private Button b2 = new Button("关闭"); private TextField t1 = new TextField(30); private TextArea t2 = new TextArea(); private int m; private String n; Socket connection; DataInputStream in; DataOutputStream out; private class window extends WindowAdapter{ public void windowClosing(WindowEvent e){ System.exit(0); } } private void focusEvt(java.awt.event.WindowEvent evt) { t1.requestFocus(); } public void clock(){ T=Calendar.getInstance(); H=T.get(Calendar.HOUR_OF_DAY); M=T.get(Calendar.MINUTE); S=T.get(Calendar.SECOND); l=String.valueOf(H)+":"+String.valueOf(M)+":"+String.valueOf(S); } public EchoClient(String i,int j){ super("客户端"); t = new Thread(this); m=j; n=i; t.start(); }
2009.17 网络与通信 NETWORK&COMMUNICATION 1引言 大部分网络协议的实现都由客户端(Client)和服务器端 (Server)来协作完成。这种模型本质上涉及两个不同的程序, 通常这两个程序在不同机器上运行。这些机器之间都有网络连接。服务器端程序提供服务并对来自客户程序的请求作成响应。而客户端程序则是在使用者和服务器端程序之间建立某种沟通的渠道,或者是作为使用服务器端提供的某种网络服务的工具。 一个典型的服务器与客户机之间的交互可能如下所示:(1)客户机提出一个请求; (2)服务器收到客户机的请求,进行分析处理;(3)服务器将运行处理的结果返回给客户机。 通常一个服务器需要向多个客户机提供服务。因此对服务器来说,还需要考虑如何有效地处理多个客户的请求。 2服务器与客户端的Socket 通信类型 Socket 的连接类型可以分为两种,分别是面向连接的字节 流类型(Sock_stream)和面向无连接数据报类型(Sock_dgram)。 面向无连接数据报类型的Socket 工作流程比较简单,双方不需要进行太多的沟通与交互。客户机直接将用户的请求打包发送到服务器端,省略了建立一个固定信息通道的过程。服务器端也是直接将处理的结果发送给客户端。其工作流程如图1所示。 面向连接的字节流类型的Socket 工作中有比较严格的操作次序,工作的原理也比较复杂。在这种类型的Socket 的工作过程中,必须首先启动服务器端,通过调用Socket ()函数建立一个Socket 对象,然后调用Bind ()函数将该Socket 对象和本地网络地址绑定到一起,再调用Listen ()函数使该Socket 对象处于侦听状态,并规定它的最大请求的数量。其工作流程如图2所示。 总的来说,无连接和面向连接的通信方式各有长处和短处。在仅仅涉及少量的信息传递的场合可以使用无连接操作;如果涉及大量信息传递的场合可以采用面向连接操作。 3Delphi 的Socket 组件 ClientSocket 组件为客户端组件。它是通信的请求方,也 就是说,它是主动地与服务器端建立连接。 客户端与服务器端的Socket 通信 夏 玲 摘 要:介绍有关Socket 通讯应用的基本知识,并通过客户端和服务器端的Delphi 编程实 例,说明两者是如何进行通信的。 关键词:Socket ;Delphi ;通信;客户端;服务器端 图1 无连接Socket 操作流程 图2 面向连接Socket 操作流程 49
一、课程设计题目及要求 题目通讯录管理系统 任务:自学C语言中相关知识,设计出通讯录管理系统。要求如下所述: ◆建立通讯录信息,信息至少包含编号、姓名、年龄、电话、通讯地址、电子 邮箱等;; ◆能够提供添加、删除和修改通讯录信息的功能; ◆能够提供按姓名或电话等查询; ◆将通讯录保存在文件中; ◆能够按表格方式输出通讯录信息。 二、系统设计方案 (一)总体框架图: 通讯录管理系统:1、录入通讯录信息 2、修改通讯录信息 3、查询通讯录信息 4、浏览通讯录信息 5、增加通讯录信息 6、推出系通讯录统 (二)模块设计 模块一:头文件,变量定义,函数的声明 对系统所使用的变量进行定义,对函数进行声明 模块二:录入通讯录信息 声明函数void readfile(),说明一个文件指针FILE *fp 打开文件"student.txt" 模块三:通讯录信息的查找 声明void seek()为查找函数,通过switch(item)设定用学号查找,用姓名查找两个分支 模块四:通讯录信息的修改
声明void modify()为学生信息修改函数,通过switch(item)设定所要修改的项目模块五:通讯录信息按学号排序 声明void sort()将录入通讯录信息系按升序排列,用的是“冒泡排序法”实现排序模块六:加通讯录信息 声明void insert()插入通讯录信息,先通过判断通讯录是否存在,若否则继续输入,若是跳出,重新循环 模块七:通讯录信息 声明void del()实现通讯录信息删除,通过学号对比确定要删除的信息,然后用后一个替换掉。 模块八:示学生信息 通过display()函数输出通讯录信息 模块九;存信息 通过fp=fopen("student.txt","w");写入信息 模块十:界面菜单 通过switch(num)调用,以上各函数,实现功能 三、设计详情 1.主函数 主函数设计要求简洁,只提供部分提示语和函数的调用 【程序】
Socket服务器与客户端双向通信实例 using System; using System.Collections.Generic; using https://www.doczj.com/doc/1c11832027.html,ponentModel; using System.Data; using System.Drawing; using System.Linq; using System.Text; using System.Windows.Forms; using https://www.doczj.com/doc/1c11832027.html,; using https://www.doczj.com/doc/1c11832027.html,.Sockets;//添加命名空间 using System.Threading;//添加命名空间 namespace WFAsynSocket { public partial class Form1 : Form { Thread LisThread;
Socket LisSocket; Socket newSocket; EndPoint point; string strmes = String.Empty; int port = 8000;//定义侦听端口号 public Form1() { InitializeComponent(); } private void btn_Listen_Click(object sender, EventArgs e) { LisThread = new Thread(new ThreadStart(BeginListern));//开线程执行BeginListern方法 LisThread.Start();//线程开始执行 } public IPAddress GetIP() { /*获取本地服务器的ip地址 */ IPHostEntry iep = Dns.GetHostEntry(Dns.GetHostName()); IPAddress ip = iep.AddressList[0]; return ip; } public void BeginListern() { LisSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, Proto colType.Tcp);//实例化Socket IPAddress ServerIp = GetIP();/*获取本地服务器的ip地址 */ IPEndPoint iep = new IPEndPoint(ServerIp, port); LisSocket.Bind(iep); /*将Socket绑定ip */ toolStripStatusLabel1.Text = iep.ToString() + "正在监听"; LisSocket.Listen(50); //Socket开始监听 newSocket = LisSocket.Accept();//获取连接请求的Socket /*接收客户端Socket所发的信息 */ while (true) { try {