你得学会并且学得会的Socket编程基础知识(续)——Silverlight客户端

本文将在这个案例的基础上,加入一个特殊场景,利用Silverlight来实现客户端。有的朋友可能会说,其实是一样的吧。请不要急于下结论,有用过Silverlight的朋友都有这种体会,很多在标准.NET Framework编程中能用的技术,到了Silverlight里面,或多或少会有些限制。不幸的是,Socket这个功能就是其中一个。这本身没有什么好不好的问题,Silverlight首先是运行在一个特殊的沙盒中,受到一些限制也是意料之中的,毕竟安全第一嘛

我总结Silverlight中应用Socket的几点特殊之处

1.所有的操作都必须的异步的,包括连接,发送和接收消息

2.Silverlight只能做客户端,不能做服务器(虽然这句看起来说的有点多余,不过确实有朋友想这么做呢)

3.Silverlight的Socket只能访问如下端口,4502-4530,只能用TCP。

4.Silverlight的Socket收到访问策略的限制,服务端必须监听,并提供ClientAccessPolicy的支持。通常是在943端口(TCP)进行监听,也可以在HTTP 80端口监听。

本文完整代码如下 http://files.cnblogs.com/chenxizhang/SocketWorkshop(with-silverlight).rar

那么,我们就来通过例子学习一下在Silverlight中如何使用Socket技术与服务端通讯吧

第一步:创建Silverlight项目

第二步:设计Silverlight界面

<UserControl
    x:Class="SocketSilverlightClient.MainPage"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
    xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
    mc:Ignorable="d"
    d:DesignHeight="300"
    d:DesignWidth="400">

    <Grid
        x:Name="LayoutRoot"
        Background="White"
        Margin="20">

        <Grid.Resources>
            <Style
                TargetType="Button">
                <Setter
                    Property="Width"
                    Value="100"></Setter>
                <Setter
                    Property="HorizontalAlignment"
                    Value="Left"></Setter>
                <Setter
                    Property="Margin"
                    Value="5"></Setter>
            </Style>

            <Style
                TargetType="TextBlock">
                <Setter
                    Property="Margin"
                    Value="5"></Setter>
                <Setter
                    Property="HorizontalAlignment"
                    Value="Left"></Setter>
                <Setter
                    Property="TextWrapping"
                    Value="Wrap"></Setter>
            </Style>
        </Grid.Resources>

        <StackPanel>
            <Button
                Content="Connect"
                x:Name="btConnect"
                Click="btConnect_Click"></Button>

            <TextBlock
                Text="Type your message"></TextBlock>

            <StackPanel
                Margin="5"
                Orientation="Horizontal">
                <TextBox
                    x:Name="txtInput"
                    Width="200"></TextBox>
                <Button
                    Content="Send"
                    x:Name="btSend"
                    Click="btSend_Click"></Button>
            </StackPanel>

            <TextBlock
                Text="Messages from server"></TextBlock>

            <ItemsControl
                Margin="5"
                x:Name="icResult">
            </ItemsControl>
        </StackPanel>

    </Grid>
</UserControl>

这个界面看起来像下面这样

第三步:编写基本的客户端代码

using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Animation;
using System.Windows.Shapes;

//导入命名空间
using System.Net.Sockets;

namespace SocketSilverlightClient
{
    /// <summary>
    /// 演示如何在Silverlight中使用Socket技术
    /// 作者:陈希章
    /// </summary>
    public partial class MainPage : UserControl
    {
        public MainPage()
        {
            InitializeComponent();
        }

        /// <summary>
        /// 尝试连接到服务器
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void btConnect_Click(object sender, RoutedEventArgs e)
        {
            //创建一个套接字
            var socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
            //准备一个异步参数(这是特有的)
            var args = new SocketAsyncEventArgs();
            //设置远程服务器地址,这里用DnsSafeHost,可以获取到宿主远程服务器的主机名称
            args.RemoteEndPoint = new DnsEndPoint(App.Current.Host.Source.DnsSafeHost, 4530);
            //注册Completed事件处理程序
            args.Completed += (o, a) =>
            {
                if(a.SocketError > 0)//0表示成功,其他的表示有错误
                {
                    //注意,因为Completed方法是在工作线程触发的,所以要对主线程进行访问,必须使用Dispatcher机制
                    this.Dispatcher.BeginInvoke(() =>
                    {
                        MessageBox.Show("Connect fail:" + a.SocketError.ToString());
                    });
                }
                else
                {

                    this.Dispatcher.BeginInvoke(() =>
                    {
                        MessageBox.Show("Connect success");
                    });
                }

            };

            //发起异步的连接请求
            socket.ConnectAsync(args);
        }

        private void btSend_Click(object sender, RoutedEventArgs e)
        {

        }
    }
}

【注意】在Silverlight中使用Socket的代码,与一般的客户端还是不同的。最主要的区别在于异步模型。

目前,我这里只编写了Connect的代码,是因为这里就会遇到连接问题,其他代码先不着急写出来。我们可以运行起来看看

点击“Connect”之后,我们发现有一个错误,是AccessDenied。这就是说,Silverlight遇到了权限问题无法直接访问到服务器。

我们都知道,Silverlight是运行在一个沙盒里面的,它要访问宿主网站之外的资源,是受到很多限制的。它会先尝试检查目标资源是否有一个ClientAccessPolicy的设置。

这里有一篇详细的介绍 http://msdn.microsoft.com/zh-cn/library/cc197955(VS.95).aspx

第四步:为服务器添加PolicyServer。

已经有不少先进同学在这方面有研究了。这个PolicyServer是负责向Silverlight发送策略信息的,也就是说,Silverlight的Socket,在连接之前,会默认去连接目标主机的943端口,请求ClientAccessPolicy的认证,只有通过了,则可以继续访问其他的Socket。

这个PolicyServer的设计,不是我的原创,但我稍做了修改。请将下面的代码保存为一个独立的文件,放在SocketServer这个项目里面

using System;
using System.Configuration;
using System.Diagnostics;
using System.IO;
using System.Net;
using System.Net.Sockets;
using System.Reflection;
using System.Text;

namespace SocketServerService
{
    /// <summary>
    /// This is a silverlight socket client access policy file server.
    ///
    /// Background:
    /// When a socket connection open attempt to some server is made in Silverlight 2.0
    /// Silverlight automatically makes a request to the server in question on port 943 for a policy file
    /// The policy file served includes the valid ports and valid clients for the socket server
    ///
    /// Outcomes:
    /// The socket request will result in success if the client access policy file served by the socket
    /// server permits access to the requested port and the client URI is in the <allow-from> element
    /// See ClientAccessPolicy.xml & http://msdn.microsoft.com/en-us/library/cc645032(VS.95).aspx for further details
    ///
    /// The socket request will be denied if the client access policy file is not served or if the client /
    /// port is denied in the client access policy file
    /// </summary>
    class SL_SocketPortPolicyListener
    {
        TcpListener _Listener = null;
        TcpClient _Client = null;
        const string _PolicyRequestString = "<policy-file-request/>";
        int _ReceivedLength = 0;
        byte[] _Policy = null;
        byte[] _ReceiveBuffer = null;
        EventLog eventLog;

        /// <summary>
        /// Initializes a new instance of the <see cref="SL_SocketPortPolicyListener"/> class.
        /// </summary>
        /// <param name="serviceEventLog">The service event log.</param>
        public SL_SocketPortPolicyListener(EventLog serviceEventLog)
        {
            eventLog = serviceEventLog;
            Start();
        }
        /// <summary>
        /// 增加的代码
        /// </summary>
        public SL_SocketPortPolicyListener()
            : this(new EventLog("Application"))
        {

        }

        /// <summary>
        /// Starts this instance.
        /// </summary>
        void Start()
        {
            try
            {
                //增加的代码
                var policyConfig =
                    "<?xml version=\"1.0\" encoding =\"utf-8\"?>" +
                        "<access-policy>" +
                          "<cross-domain-access>" +
                            "<policy>" +
                              "<allow-from>" +
                                "<domain uri=\"*\" />" +
                              "</allow-from>" +
                              "<grant-to>" +
                                "<socket-resource port=\"4502-4530\" protocol=\"tcp\" />" +
                              "</grant-to>" +
                            "</policy>" +
                          "</cross-domain-access>" +
                        "</access-policy>";

                //删除的代码
                //string executionLocation = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location);
                //string policyFile = ConfigurationManager.AppSettings["PolicyFilePath"];
                //using(FileStream fs = new FileStream(executionLocation + policyFile, FileMode.Open))
                //{
                //    _Policy = new byte[fs.Length];
                //    fs.Read(_Policy, 0, _Policy.Length);
                //}

                //增加的代码
                _Policy = Encoding.Default.GetBytes(policyConfig);

                _ReceiveBuffer = new byte[_PolicyRequestString.Length];

                //Using TcpListener which is a wrapper around a Socket
                //Allowed port is 943 for Silverlight sockets policy data
                _Listener = new TcpListener(IPAddress.Any, 943);
                _Listener.Start();
                _Listener.BeginAcceptTcpClient(new AsyncCallback(OnBeginAccept), null);
            }
            catch(Exception exp)
            {
                LogError(exp);
            }
        }

        /// <summary>
        /// Called when [begin accept].
        /// </summary>
        /// <param name="ar">The ar.</param>
        private void OnBeginAccept(IAsyncResult ar)
        {
            _Client = _Listener.EndAcceptTcpClient(ar);
            _Client.Client.BeginReceive(_ReceiveBuffer, 0, _PolicyRequestString.Length, SocketFlags.None,
                new AsyncCallback(OnReceiveComplete), null);
        }

        /// <summary>
        /// Called when [receive complete].
        /// </summary>
        /// <param name="ar">The ar.</param>
        private void OnReceiveComplete(IAsyncResult ar)
        {
            try
            {
                _ReceivedLength += _Client.Client.EndReceive(ar);
                //See if there‘s more data that we need to grab
                if(_ReceivedLength < _PolicyRequestString.Length)
                {
                    //Need to grab more data so receive remaining data
                    _Client.Client.BeginReceive(_ReceiveBuffer, _ReceivedLength,
                        _PolicyRequestString.Length - _ReceivedLength,
                        SocketFlags.None, new AsyncCallback(OnReceiveComplete), null);
                    return;
                }

                //Check that <policy-file-request/> was sent from client
                string request = System.Text.Encoding.UTF8.GetString(_ReceiveBuffer, 0, _ReceivedLength);
                if(StringComparer.InvariantCultureIgnoreCase.Compare(request, _PolicyRequestString) != 0)
                {
                    //Data received isn‘t valid so close
                    _Client.Client.Close();
                    return;
                }
                //Valid request received....send policy file
                _Client.Client.BeginSend(_Policy, 0, _Policy.Length, SocketFlags.None,
                    new AsyncCallback(OnSendComplete), _Client.Client);
            }
            catch(Exception exp)
            {
                _Client.Client.Close();
                LogError(exp);
            }
            _ReceivedLength = 0;
            //listen for the next client
            _Listener.BeginAcceptTcpClient(new AsyncCallback(OnBeginAccept), null);
        }

        /// <summary>
        /// Called when [send complete].
        /// </summary>
        /// <param name="ar">The ar.</param>
        private void OnSendComplete(IAsyncResult ar)
        {
            Socket socket = (Socket)ar.AsyncState;
            try
            {
                socket.EndSend(ar);
            }
            catch(Exception exp)
            {
                LogError(exp);
            }
            finally
            {
                socket.Close();
            }
        }

        /// <summary>
        /// Logs the error.
        /// </summary>
        /// <param name="exp">The exp.</param>
        private void LogError(Exception exp)
        {
            eventLog.WriteEntry(string.Format("Error in PolicySocketServer: {0} \r\n StackTrace: {1}", exp.Message, exp.StackTrace));
        }
    }
}

然后,在SocketServer的主程序中,加入下面的代码(只需要添加红色这一行即可

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

//额外导入的两个命名空间
using System.Net.Sockets;
using System.Net;

namespace SocketServer
{
    class Program
    {
        /// <summary>
        /// Socket Server 演示
        /// 作者:陈希章
        /// </summary>
        /// <param name="args"></param>
        static void Main(string[] args)
        {

            var policyServer = new SocketServerService.SL_SocketPortPolicyListener();

            //创建一个新的Socket,这里我们使用最常用的基于TCP的Stream Socket(流式套接字)
            var socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);

            //将该socket绑定到主机上面的某个端口
            //方法参考:http://msdn.microsoft.com/zh-cn/library/system.net.sockets.socket.bind.aspx
            socket.Bind(new IPEndPoint(IPAddress.Any, 4530));

            //启动监听,并且设置一个最大的队列长度
            //方法参考:http://msdn.microsoft.com/zh-cn/library/system.net.sockets.socket.listen(v=VS.100).aspx
            socket.Listen(4);

            //开始接受客户端连接请求
            //方法参考:http://msdn.microsoft.com/zh-cn/library/system.net.sockets.socket.beginaccept.aspx
            socket.BeginAccept(new AsyncCallback(ClientAccepted), socket);

            Console.WriteLine("Server is ready!");
            Console.Read();
        }

        public static void ClientAccepted(IAsyncResult ar)
        {

            var socket = ar.AsyncState as Socket;

            //这就是客户端的Socket实例,我们后续可以将其保存起来
            var client = socket.EndAccept(ar);

            //给客户端发送一个欢迎消息
            client.Send(Encoding.Unicode.GetBytes("Hi there, I accept you request at " + DateTime.Now.ToString()));

            //实现每隔两秒钟给服务器发一个消息
            //这里我们使用了一个定时器
            var timer = new System.Timers.Timer();
            timer.Interval = 2000D;
            timer.Enabled = true;
            timer.Elapsed += (o, a) =>
            {
                //检测客户端Socket的状态
                if(client.Connected)
                {
                    try
                    {
                        client.Send(Encoding.Unicode.GetBytes("Message from server at " + DateTime.Now.ToString()));
                    }
                    catch(SocketException ex)
                    {
                        Console.WriteLine(ex.Message);
                    }
                }
                else
                {
                    timer.Stop();
                    timer.Enabled = false;
                    Console.WriteLine("Client is disconnected, the timer is stop.");
                }
            };
            timer.Start();

            //接收客户端的消息(这个和在客户端实现的方式是一样的)
            client.BeginReceive(buffer, 0, buffer.Length, SocketFlags.None, new AsyncCallback(ReceiveMessage), client);

            //准备接受下一个客户端请求
            socket.BeginAccept(new AsyncCallback(ClientAccepted), socket);
        }

        static byte[] buffer = new byte[1024];

        public static void ReceiveMessage(IAsyncResult ar)
        {

            try
            {
                var socket = ar.AsyncState as Socket;

                //方法参考:http://msdn.microsoft.com/zh-cn/library/system.net.sockets.socket.endreceive.aspx
                var length = socket.EndReceive(ar);
                //读取出来消息内容
                var message = Encoding.Unicode.GetString(buffer, 0, length);
                //显示消息
                Console.WriteLine(message);

                //接收下一个消息(因为这是一个递归的调用,所以这样就可以一直接收消息了)
                socket.BeginReceive(buffer, 0, buffer.Length, SocketFlags.None, new AsyncCallback(ReceiveMessage), socket);
            }
            catch(Exception ex){
                Console.WriteLine(ex.Message);
            }
        }
    }
}

再次测试,我们就发现Silverlight客户端能够连接到服务器了

既然连接上了服务器,那么就让我们来将Silverlight客户端里面其他的一些功能都实现一下吧

第五步:实现Silverlight客户端的消息收发

using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Animation;
using System.Windows.Shapes;

//导入命名空间
using System.Net.Sockets;
using System.Text;

namespace SocketSilverlightClient
{
    /// <summary>
    /// 演示如何在Silverlight中使用Socket技术
    /// 作者:陈希章
    /// </summary>
    public partial class MainPage : UserControl
    {
        public MainPage()
        {
            InitializeComponent();
        }

        //创建一个套接字
        Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);

        /// <summary>
        /// 尝试连接到服务器
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void btConnect_Click(object sender, RoutedEventArgs e)
        {
            //准备一个异步参数(这是特有的)
            var args = new SocketAsyncEventArgs();
            //设置远程服务器地址,这里用DnsSafeHost,可以获取到宿主远程服务器的主机名称
            args.RemoteEndPoint = new DnsEndPoint(App.Current.Host.Source.DnsSafeHost, 4530);
            //注册Completed事件处理程序
            args.Completed += ConnectCompleted;

            //发起异步的连接请求
            socket.ConnectAsync(args);
        }

        /// <summary>
        /// 该事件在连接成功时发生
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="args"></param>
        public void ConnectCompleted(object sender, SocketAsyncEventArgs e)
        {
            if(e.SocketError > 0)//0表示成功,其他的表示有错误
            {
                //注意,因为Completed方法是在工作线程触发的,所以要对主线程进行访问,必须使用Dispatcher机制
                this.Dispatcher.BeginInvoke(() =>
                {
                    MessageBox.Show("Connect fail:" + e.SocketError.ToString());
                });
            }
            else
            {
                this.Dispatcher.BeginInvoke(() =>
                {
                    //MessageBox.Show("Connect success");

                    //将连接按钮禁用掉
                    btConnect.Content = "Connected";
                    btConnect.IsEnabled = false;

                    var buffer = new byte[1024];
                    e.SetBuffer(buffer, 0, buffer.Length);
                    e.Completed -= ConnectCompleted;
                    e.Completed += ReceiveCompleted;
                    socket.ReceiveAsync(e);
                });
            }
        }

        /// <summary>
        /// 该事件在接收消息时发生
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="args"></param>
        public void ReceiveCompleted(object sender, SocketAsyncEventArgs e)
        {
            //将消息显示在界面上
            var result = Encoding.Unicode.GetString(e.Buffer, 0, e.Count);
            this.Dispatcher.BeginInvoke(() =>
            {
                icResult.Items.Add(result);
            });
            //递归继续接收消息
            socket.ReceiveAsync(e);
        }

        private void btSend_Click(object sender, RoutedEventArgs e)
        {
            var args = new SocketAsyncEventArgs();
            //将用户输入的文本转成字节
            var buffer = Encoding.Unicode.GetBytes(txtInput.Text);
            args.SetBuffer(buffer, 0, buffer.Length);
            //设置远程服务器地址,这里用DnsSafeHost,可以获取到宿主远程服务器的主机名称
            args.RemoteEndPoint = new DnsEndPoint(App.Current.Host.Source.DnsSafeHost, 4530);

            //发送完成的话,将控件清空,激活
            args.Completed += (o, a) => {
                this.Dispatcher.BeginInvoke(() =>
                {
                    txtInput.Text = string.Empty;
                    btSend.IsEnabled = true;
                });
            };
            //禁用按钮
            btSend.IsEnabled = false;
            //发送消息
            socket.SendAsync(args);
        }
    }
}

运行起来看看吧

还不错对吧,这个例子给大家演示了如何在Silverlight中使用Socket,接下来大家可以结合自己的现实工作做一些研究和扩展吧

本文完整代码如下 http://files.cnblogs.com/chenxizhang/SocketWorkshop(with-silverlight).rar

转自:http://www.cnblogs.com/chenxizhang/archive/2011/09/10/2173101.html

时间: 2024-08-24 01:19:24

你得学会并且学得会的Socket编程基础知识(续)——Silverlight客户端的相关文章

你得学会并且学得会的Socket编程基础知识

这一篇文章,我将图文并茂地介绍Socket编程的基础知识,我相信,如果你按照步骤做完实验,一定可以对Socket编程有更好地理解. 本文源代码,可以通过这里下载 http://files.cnblogs.com/chenxizhang/SocketWorkshop.rar 第一步:创建解决方案 第二步:创建服务端程序 这里可以选择“Console Application”这个类型,比较方便调试 然后编写如下代码,实现服务器的基本功能 using System; using System.Coll

学大数据需要什么编程基础?大数据学习步骤是什么?

学大数据需要什么编程基础?大数据学习步骤是什么? 大数据是什么? 有很多朋友问过我,大数据到底是什么?一句话来- 学大数据需要什么编程基础?大数据学习步骤是什么?大数据是什么? 有很多朋友问过我,大数据到底是什么?一句话来概括 针对非软件行业的朋友 根据你平时在超市,加油站,饭店等地方的一些消费行为,通过大数据这个技术,我们可以知道你现在的年龄范围,是否婚配,是否有孩子,孩子大致是几岁,是否有固定住宅,车大致是什么价位的等信息. 针对软件行业的朋友 平时我们写的程序都是在一台机器上运行,处理能力

Java从零开始学四十五(Socket编程基础)

一.网络编程中两个主要的问题 一个是如何准确的定位网络上一台或多台主机,另一个就是找到主机后如何可靠高效的进行数据传输. 在TCP/IP协议中IP层主要负责网络主机的定位,数据传输的路由,由IP地址可以唯一地确定Internet上的一台主机. 而TCP层则提供面向应用的可靠(tcp)的或非可靠(UDP)的数据传输机制,这是网络编程的主要对象,一般不需要关心IP层是如何处理数据的. 目前较为流行的网络编程模型是客户机/服务器(C/S)结构.即通信双方一方作为服务器等待客户提出请求并予以响应.客户则

学golang之前都需要哪些前置知识?

我学golang,感觉前面基础语法部分都很快能学会,但是到了goroutine,channel等后面的部分就看不懂了,是不是我学这个之前还得学习其他什么知识啊?(我有C语言基础,对于C语言里面的指针,结构体,位运算等都有所了解) 学golang之前都需要哪些前置知识? >> golang 这个答案描述的挺清楚的:http://www.goodpm.net/postreply/golang/1010000008866706/学golang之前都需要哪些前置知识.html

没有任何编程基础可以直接学习python语言吗?学会后能够做什么?

很明确的说 python非常适合没有任何编程基础的人入门.. 目前应用最多的:全栈开发.数据分析.运维开发,今天我们就以这三个重点的岗位来做一下自学Python的规划,希望你在学之前就能有结果的来走得更远. 很多人在问,python学了之后能做些什么? 既然你没有碰过 Python ,不知编程为何物的人,我就不提编程里面的项目名了--创一个小群,供大家学习交流聊天如果有对学python方面有什么疑惑问题的,或者有什么想说的想聊的大家可以一起交流学习一起进步呀.也希望大家对学python能够持之以

Rhythmk 一步一步学 JAVA (22) JAVA 网络编程

1.获取主机信息 ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 @Test     public void GetDomainInfo() throws UnknownHostException {         String domain = "www.baidu.com";         InetAddress netAddress = InetAddress.getByName(domain);         // 获取

学习Spring必学的Java基础知识(2)----动态代理

学习Spring必学的Java基础知识(2)----动态代理 引述要学习Spring框架的技术内幕,必须事先掌握一些基本的Java知识,正所谓“登高必自卑,涉远必自迩”.以下几项Java知识和Spring框架息息相关,不可不学(我将通过一个系列分别介绍这些Java基础知识,希望对大家有所帮助.): [1] Java反射知识-->Spring IoC :http://www.iteye.com/topic/1123081 [2] Java动态代理-->Spring AOP :http://www

女孩子适合学web前端还是Java编程?

近几年随着互联网的快速发展,对于Web前端开发的人才需求越来越大,就业薪资也不断的上升,随着行业的火爆,高薪回报吸引了很多有志青年投身互联网行业.女孩子适合学Web前端还是Java? 纵观现阶段互联网Web前端开发工程师的就业人员,女孩子从事这个行业的比例不大,由于这种现象的存在,当有女孩说想要学Web前端开发,很多不一样的声音就出来了,说女生不适合做程序员,其实不然,没有什么东西是你天生做不了的,干不了,只能说明你还不够努力. 虽然少,不是照样有女孩子在做么?而且由于男女比例不协调,所以女生做

快看Sample代码,速学Swift语言(2)-基础介绍 快看Sample代码,速学Swift语言(1)-语法速览

快看Sample代码,速学Swift语言(2)-基础介绍 Swift语言是一个新的编程语言,用于iOS, macOS, watchOS, 和 tvOS的开发,不过Swift很多部分内容,我们可以从C或者Objective-C的开发经验获得一种熟悉感.Swift提供很多基础类型,如Int,String,Double,Bool等类型,它和Objective-C的相关类型对应,不过他是值类型,而Objective-C的基础类型是引用类型,另外Swift还提供了几个集合类型,如Array, Set, 和