当前位置: 首页 > news >正文

阜城网站建设公司军事新闻内容摘抄

阜城网站建设公司,军事新闻内容摘抄,网站建设使用多语言,优秀集团网站设计文章目录Java 线程基础线程简介什么是进程什么是线程进程和线程的区别创建线程ThreadRunnableCallable、Future、FutureTaskCallableFutureFutureTaskCallable Future FutureTask 示例线程基本用法线程休眠线程礼让终止线程守护线程线程通信wait/notify/notifyAlljoin管道线程… 文章目录Java 线程基础线程简介什么是进程什么是线程进程和线程的区别创建线程ThreadRunnableCallable、Future、FutureTaskCallableFutureFutureTaskCallable Future FutureTask 示例线程基本用法线程休眠线程礼让终止线程守护线程线程通信wait/notify/notifyAlljoin管道线程生命周期线程常见问题sleep、yield、join 方法有什么区别为什么 sleep 和 yield 方法是静态的Java 线程是否按照线程优先级严格执行一个线程两次调用 start()方法会怎样start 和 run 方法有什么区别可以直接调用 Thread 类的 run 方法么参考资料Java 线程基础 关键词Thread、Runnable、Callable、Future、wait、notify、notifyAll、join、sleep、yeild、线程状态、线程通信 线程简介 什么是进程 简言之进程可视为一个正在运行的程序。它是系统运行程序的基本单位因此进程是动态的。进程是具有一定独立功能的程序关于某个数据集合上的一次运行活动。进程是操作系统进行资源分配的基本单位。 什么是线程 线程是操作系统进行调度的基本单位。线程也叫轻量级进程Light Weight Process在一个进程里可以创建多个线程这些线程都拥有各自的计数器、堆栈和局部变量等属性并且能够访问共享的内存变量。 进程和线程的区别 一个程序至少有一个进程一个进程至少有一个线程。线程比进程划分更细所以执行开销更小并发性更高。进程是一个实体拥有独立的资源而同一个进程中的多个线程共享进程的资源。 创建线程 创建线程有三种方式 继承 Thread 类实现 Runnable 接口实现 Callable 接口 Thread 通过继承 Thread 类创建线程的步骤 定义 Thread 类的子类并覆写该类的 run 方法。run 方法的方法体就代表了线程要完成的任务因此把 run 方法称为执行体。创建 Thread 子类的实例即创建了线程对象。调用线程对象的 start 方法来启动该线程。 public class ThreadDemo {public static void main(String[] args) {// 实例化对象MyThread tA new MyThread(Thread 线程-A);MyThread tB new MyThread(Thread 线程-B);// 调用线程主体tA.start();tB.start();}static class MyThread extends Thread {private int ticket 5;MyThread(String name) {super(name);}Overridepublic void run() {while (ticket 0) {System.out.println(Thread.currentThread().getName() 卖出了第 ticket 张票);ticket--;}}}}Runnable 实现 Runnable 接口优于继承 Thread 类因为 Java 不支持多重继承所有的类都只允许继承一个父类但可以实现多个接口。如果继承了 Thread 类就无法继承其它类这不利于扩展。类可能只要求可执行就行继承整个 Thread 类开销过大。 通过实现 Runnable 接口创建线程的步骤 定义 Runnable 接口的实现类并覆写该接口的 run 方法。该 run 方法的方法体同样是该线程的线程执行体。创建 Runnable 实现类的实例并以此实例作为 Thread 的 target 来创建 Thread 对象该 Thread 对象才是真正的线程对象。调用线程对象的 start 方法来启动该线程。 public class RunnableDemo {public static void main(String[] args) {// 实例化对象Thread tA new Thread(new MyThread(), Runnable 线程-A);Thread tB new Thread(new MyThread(), Runnable 线程-B);// 调用线程主体tA.start();tB.start();}static class MyThread implements Runnable {private int ticket 5;Overridepublic void run() {while (ticket 0) {System.out.println(Thread.currentThread().getName() 卖出了第 ticket 张票);ticket--;}}}}Callable、Future、FutureTask 继承 Thread 类和实现 Runnable 接口这两种创建线程的方式都没有返回值。所以线程执行完后无法得到执行结果。但如果期望得到执行结果该怎么做 为了解决这个问题Java 1.5 后提供了 Callable 接口和 Future 接口通过它们可以在线程执行结束后返回执行结果。 Callable Callable 接口只声明了一个方法这个方法叫做 call() public interface CallableV {/*** Computes a result, or throws an exception if unable to do so.** return computed result* throws Exception if unable to compute a result*/V call() throws Exception; }那么怎么使用 Callable 呢一般情况下是配合 ExecutorService 来使用的在 ExecutorService 接口中声明了若干个 submit 方法的重载版本 T FutureT submit(CallableT task); T FutureT submit(Runnable task, T result); Future? submit(Runnable task);第一个 submit 方法里面的参数类型就是 Callable。 Future Future 就是对于具体的 Callable 任务的执行结果进行取消、查询是否完成、获取结果。必要时可以通过 get 方法获取执行结果该方法会阻塞直到任务返回结果。 public interface FutureV {boolean cancel(boolean mayInterruptIfRunning);boolean isCancelled();boolean isDone();V get() throws InterruptedException, ExecutionException;V get(long timeout, TimeUnit unit)throws InterruptedException, ExecutionException, TimeoutException; }FutureTask FutureTask 类实现了 RunnableFuture 接口RunnableFuture 继承了 Runnable 接口和 Future 接口。 所以FutureTask 既可以作为 Runnable 被线程执行又可以作为 Future 得到 Callable 的返回值。 public class FutureTaskV implements RunnableFutureV {// ...public FutureTask(CallableV callable) {}public FutureTask(Runnable runnable, V result) {} }public interface RunnableFutureV extends Runnable, FutureV {void run(); }事实上FutureTask 是 Future 接口的一个唯一实现类。 Callable Future FutureTask 示例 通过实现 Callable 接口创建线程的步骤 创建 Callable 接口的实现类并实现 call 方法。该 call 方法将作为线程执行体并且有返回值。创建 Callable 实现类的实例使用 FutureTask 类来包装 Callable 对象该 FutureTask 对象封装了该 Callable 对象的 call 方法的返回值。使用 FutureTask 对象作为 Thread 对象的 target 创建并启动新线程。调用 FutureTask 对象的 get 方法来获得线程执行结束后的返回值。 public class CallableDemo {public static void main(String[] args) {CallableLong callable new MyThread();FutureTaskLong future new FutureTask(callable);new Thread(future, Callable 线程).start();try {System.out.println(任务耗时 (future.get() / 1000000) 毫秒);} catch (InterruptedException | ExecutionException e) {e.printStackTrace();}}static class MyThread implements CallableLong {private int ticket 10000;Overridepublic Long call() {long begin System.nanoTime();while (ticket 0) {System.out.println(Thread.currentThread().getName() 卖出了第 ticket 张票);ticket--;}long end System.nanoTime();return (end - begin);}}}线程基本用法 线程Thread基本方法清单 方法描述run线程的执行实体。start线程的启动方法。currentThread返回对当前正在执行的线程对象的引用。setName设置线程名称。getName获取线程名称。setPriority设置线程优先级。Java 中的线程优先级的范围是 [1,10]一般来说高优先级的线程在运行时会具有优先权。可以通过 thread.setPriority(Thread.MAX_PRIORITY) 的方式设置默认优先级为 5。getPriority获取线程优先级。setDaemon设置线程为守护线程。isDaemon判断线程是否为守护线程。isAlive判断线程是否启动。interrupt中断另一个线程的运行状态。interrupted测试当前线程是否已被中断。通过此方法可以清除线程的中断状态。换句话说如果要连续调用此方法两次则第二次调用将返回 false除非当前线程在第一次调用清除其中断状态之后且在第二次调用检查其状态之前再次中断。join可以使一个线程强制运行线程强制运行期间其他线程无法运行必须等待此线程完成之后才可以继续执行。Thread.sleep静态方法。将当前正在执行的线程休眠。Thread.yield静态方法。将当前正在执行的线程暂停让其他线程执行。 线程休眠 使用 Thread.sleep 方法可以使得当前正在执行的线程进入休眠状态。 使用 Thread.sleep 需要向其传入一个整数值这个值表示线程将要休眠的毫秒数。 Thread.sleep 方法可能会抛出 InterruptedException因为异常不能跨线程传播回 main 中因此必须在本地进行处理。线程中抛出的其它异常也同样需要在本地进行处理。 public class ThreadSleepDemo {public static void main(String[] args) {new Thread(new MyThread(线程A, 500)).start();new Thread(new MyThread(线程B, 1000)).start();new Thread(new MyThread(线程C, 1500)).start();}static class MyThread implements Runnable {/** 线程名称 */private String name;/** 休眠时间 */private int time;private MyThread(String name, int time) {this.name name;this.time time;}Overridepublic void run() {try {// 休眠指定的时间Thread.sleep(this.time);} catch (InterruptedException e) {e.printStackTrace();}System.out.println(this.name 休眠 this.time 毫秒。);}}}线程礼让 Thread.yield 方法的调用声明了当前线程已经完成了生命周期中最重要的部分可以切换给其它线程来执行 。 该方法只是对线程调度器的一个建议而且也只是建议具有相同优先级的其它线程可以运行。 public class ThreadYieldDemo {public static void main(String[] args) {MyThread t new MyThread();new Thread(t, 线程A).start();new Thread(t, 线程B).start();}static class MyThread implements Runnable {Overridepublic void run() {for (int i 0; i 5; i) {try {Thread.sleep(1000);} catch (Exception e) {e.printStackTrace();}System.out.println(Thread.currentThread().getName() 运行i i);if (i 2) {System.out.print(线程礼让);Thread.yield();}}}} }终止线程 Thread 中的 stop 方法有缺陷已废弃。 使用 Thread.stop 停止线程会导致它解锁所有已锁定的监视器由于未经检查的 ThreadDeath 异常会在堆栈中传播这是自然的结果。 如果先前由这些监视器保护的任何对象处于不一致状态则损坏的对象将对其他线程可见从而可能导致任意行为。 stop() 方法会真的杀死线程不给线程喘息的机会如果线程持有 ReentrantLock 锁被 stop() 的线程并不会自动调用 ReentrantLock 的 unlock() 去释放锁那其他线程就再也没机会获得 ReentrantLock 锁这实在是太危险了。所以该方法就不建议使用了类似的方法还有 suspend() 和 resume() 方法这两个方法同样也都不建议使用了所以这里也就不多介绍了。Thread.stop 的许多用法应由仅修改某些变量以指示目标线程应停止运行的代码代替。 目标线程应定期检查此变量如果该变量指示要停止运行则应按有序方式从其运行方法返回。如果目标线程等待很长时间例如在条件变量上则应使用中断方法来中断等待。 当一个线程运行时另一个线程可以直接通过 interrupt 方法中断其运行状态。 public class ThreadInterruptDemo {public static void main(String[] args) {MyThread mt new MyThread(); // 实例化Runnable子类对象Thread t new Thread(mt, 线程); // 实例化Thread对象t.start(); // 启动线程try {Thread.sleep(2000); // 线程休眠2秒} catch (InterruptedException e) {System.out.println(3、main线程休眠被终止);}t.interrupt(); // 中断线程执行}static class MyThread implements Runnable {Overridepublic void run() {System.out.println(1、进入run()方法);try {Thread.sleep(10000); // 线程休眠10秒System.out.println(2、已经完成了休眠);} catch (InterruptedException e) {System.out.println(3、MyThread线程休眠被终止);return; // 返回调用处}System.out.println(4、run()方法正常结束);}} }如果一个线程的 run 方法执行一个无限循环并且没有执行 sleep 等会抛出 InterruptedException 的操作那么调用线程的 interrupt 方法就无法使线程提前结束。 但是调用 interrupt 方法会设置线程的中断标记此时调用 interrupted 方法会返回 true。因此可以在循环体中使用 interrupted 方法来判断线程是否处于中断状态从而提前结束线程。 安全地终止线程有两种方法 定义 volatile 标志位在 run 方法中使用标志位控制线程终止使用 interrupt 方法和 Thread.interrupted 方法配合使用来控制线程终止 【示例】使用 volatile 标志位控制线程终止 public class ThreadStopDemo2 {public static void main(String[] args) throws Exception {MyTask task new MyTask();Thread thread new Thread(task, MyTask);thread.start();TimeUnit.MILLISECONDS.sleep(50);task.cancel();}private static class MyTask implements Runnable {private volatile boolean flag true;private volatile long count 0L;Overridepublic void run() {System.out.println(Thread.currentThread().getName() 线程启动);while (flag) {System.out.println(count);}System.out.println(Thread.currentThread().getName() 线程终止);}/*** 通过 volatile 标志位来控制线程终止*/public void cancel() {flag false;}}}【示例】使用 interrupt 方法和 Thread.interrupted 方法配合使用来控制线程终止 public class ThreadStopDemo3 {public static void main(String[] args) throws Exception {MyTask task new MyTask();Thread thread new Thread(task, MyTask);thread.start();TimeUnit.MILLISECONDS.sleep(50);thread.interrupt();}private static class MyTask implements Runnable {private volatile long count 0L;Overridepublic void run() {System.out.println(Thread.currentThread().getName() 线程启动);// 通过 Thread.interrupted 和 interrupt 配合来控制线程终止while (!Thread.interrupted()) {System.out.println(count);}System.out.println(Thread.currentThread().getName() 线程终止);}} }守护线程 什么是守护线程 守护线程Daemon Thread是在后台执行并且不会阻止 JVM 终止的线程。当所有非守护线程结束时程序也就终止同时会杀死所有守护线程。与守护线程Daemon Thread相反的叫用户线程User Thread也就是非守护线程。 为什么需要守护线程 守护线程的优先级比较低用于为系统中的其它对象和线程提供服务。典型的应用就是垃圾回收器。 如何使用守护线程 可以使用 isDaemon 方法判断线程是否为守护线程。可以使用 setDaemon 方法设置线程为守护线程。 正在运行的用户线程无法设置为守护线程所以 setDaemon 必须在 thread.start 方法之前设置否则会抛出 llegalThreadStateException 异常一个守护线程创建的子线程依然是守护线程。不要认为所有的应用都可以分配给守护线程来进行服务比如读写操作或者计算逻辑。 public class ThreadDaemonDemo {public static void main(String[] args) {Thread t new Thread(new MyThread(), 线程);t.setDaemon(true); // 此线程在后台运行System.out.println(线程 t 是否是守护进程 t.isDaemon());t.start(); // 启动线程}static class MyThread implements Runnable {Overridepublic void run() {while (true) {System.out.println(Thread.currentThread().getName() 在运行。);}}} }参考阅读Java 中守护线程的总结 线程通信 当多个线程可以一起工作去解决某个问题时如果某些部分必须在其它部分之前完成那么就需要对线程进行协调。 wait/notify/notifyAll wait - wait 会自动释放当前线程占有的对象锁并请求操作系统挂起当前线程让线程从 Running 状态转入 Waiting 状态等待 notify / notifyAll 来唤醒。如果没有释放锁那么其它线程就无法进入对象的同步方法或者同步控制块中那么就无法执行 notify 或者 notifyAll 来唤醒挂起的线程造成死锁。notify - 唤醒一个正在 Waiting 状态的线程并让它拿到对象锁具体唤醒哪一个线程由 JVM 控制 。notifyAll - 唤醒所有正在 Waiting 状态的线程接下来它们需要竞争对象锁。 注意 wait、notify、notifyAll 都是 Object 类中的方法而非 Thread。wait、notify、notifyAll 只能用在 synchronized 方法或者 synchronized 代码块中使用否则会在运行时抛出 IllegalMonitorStateException。 为什么 wait、notify、notifyAll 不定义在 Thread 中为什么 wait、notify、notifyAll 要配合 synchronized 使用 首先需要了解几个基本知识点 每一个 Java 对象都有一个与之对应的 监视器monitor每一个监视器里面都有一个 对象锁 、一个 等待队列、一个 同步队列 了解了以上概念我们回过头来理解前面两个问题。 为什么这几个方法不定义在 Thread 中 由于每个对象都拥有对象锁让当前线程等待某个对象锁自然应该基于这个对象Object来操作而非使用当前线程Thread来操作。因为当前线程可能会等待多个线程的锁如果基于线程Thread来操作就非常复杂了。 为什么 wait、notify、notifyAll 要配合 synchronized 使用 如果调用某个对象的 wait 方法当前线程必须拥有这个对象的对象锁因此调用 wait 方法必须在 synchronized 方法和 synchronized 代码块中。 生产者、消费者模式是 wait、notify、notifyAll 的一个经典使用案例 public class ThreadWaitNotifyDemo02 {private static final int QUEUE_SIZE 10;private static final PriorityQueueInteger queue new PriorityQueue(QUEUE_SIZE);public static void main(String[] args) {new Producer(生产者A).start();new Producer(生产者B).start();new Consumer(消费者A).start();new Consumer(消费者B).start();}static class Consumer extends Thread {Consumer(String name) {super(name);}Overridepublic void run() {while (true) {synchronized (queue) {while (queue.size() 0) {try {System.out.println(队列空等待数据);queue.wait();} catch (InterruptedException e) {e.printStackTrace();queue.notifyAll();}}queue.poll(); // 每次移走队首元素queue.notifyAll();try {Thread.sleep(500);} catch (InterruptedException e) {e.printStackTrace();}System.out.println(Thread.currentThread().getName() 从队列取走一个元素队列当前有 queue.size() 个元素);}}}}static class Producer extends Thread {Producer(String name) {super(name);}Overridepublic void run() {while (true) {synchronized (queue) {while (queue.size() QUEUE_SIZE) {try {System.out.println(队列满等待有空余空间);queue.wait();} catch (InterruptedException e) {e.printStackTrace();queue.notifyAll();}}queue.offer(1); // 每次插入一个元素queue.notifyAll();try {Thread.sleep(500);} catch (InterruptedException e) {e.printStackTrace();}System.out.println(Thread.currentThread().getName() 向队列取中插入一个元素队列当前有 queue.size() 个元素);}}}} }join 在线程操作中可以使用 join 方法让一个线程强制运行线程强制运行期间其他线程无法运行必须等待此线程完成之后才可以继续执行。 public class ThreadJoinDemo {public static void main(String[] args) {MyThread mt new MyThread(); // 实例化Runnable子类对象Thread t new Thread(mt, mythread); // 实例化Thread对象t.start(); // 启动线程for (int i 0; i 50; i) {if (i 10) {try {t.join(); // 线程强制运行} catch (InterruptedException e) {e.printStackTrace();}}System.out.println(Main 线程运行 -- i);}}static class MyThread implements Runnable {Overridepublic void run() {for (int i 0; i 50; i) {System.out.println(Thread.currentThread().getName() 运行i i); // 取得当前线程的名字}}} }管道 管道输入/输出流和普通的文件输入/输出流或者网络输入/输出流不同之处在于它主要用于线程之间的数据传输而传输的媒介为内存。 管道输入/输出流主要包括了如下 4 种具体实现PipedOutputStream、PipedInputStream、PipedReader 和 PipedWriter前两种面向字节而后两种面向字符。 public class Piped {public static void main(String[] args) throws Exception {PipedWriter out new PipedWriter();PipedReader in new PipedReader();// 将输出流和输入流进行连接否则在使用时会抛出IOExceptionout.connect(in);Thread printThread new Thread(new Print(in), PrintThread);printThread.start();int receive 0;try {while ((receive System.in.read()) ! -1) {out.write(receive);}} finally {out.close();}}static class Print implements Runnable {private PipedReader in;Print(PipedReader in) {this.in in;}public void run() {int receive 0;try {while ((receive in.read()) ! -1) {System.out.print((char) receive);}} catch (IOException e) {e.printStackTrace();}}} }线程生命周期 java.lang.Thread.State 中定义了 6 种不同的线程状态在给定的一个时刻线程只能处于其中的一个状态。 以下是各状态的说明以及状态间的联系 新建New - 尚未调用 start 方法的线程处于此状态。此状态意味着创建的线程尚未启动。 就绪Runnable - 已经调用了 start 方法的线程处于此状态。此状态意味着线程已经在 JVM 中运行。但是在操作系统层面它可能处于运行状态也可能等待资源调度例如处理器资源资源调度完成就进入运行状态。所以该状态的可运行是指可以被运行具体有没有运行要看底层操作系统的资源调度。 阻塞Blocked - 此状态意味着线程处于被阻塞状态。表示线程在等待 synchronized 的隐式锁Monitor lock。synchronized 修饰的方法、代码块同一时刻只允许一个线程执行其他线程只能等待即处于阻塞状态。当占用 synchronized 隐式锁的线程释放锁并且等待的线程获得 synchronized 隐式锁时就又会从 BLOCKED 转换到 RUNNABLE 状态。 等待Waiting - 此状态意味着线程无限期等待直到被其他线程显式地唤醒。 阻塞和等待的区别在于阻塞是被动的它是在等待获取 synchronized 的隐式锁。而等待是主动的通过调用 Object.wait 等方法进入。 进入方法退出方法没有设置 Timeout 参数的 Object.wait 方法Object.notify / Object.notifyAll没有设置 Timeout 参数的 Thread.join 方法被调用的线程执行完毕LockSupport.park 方法Java 并发包中的锁都是基于它实现的LockSupport.unpark 定时等待Timed waiting - 此状态意味着无需等待其它线程显式地唤醒在一定时间之后会被系统自动唤醒。 进入方法退出方法Thread.sleep 方法时间结束获得 synchronized 隐式锁的线程调用设置了 Timeout 参数的 Object.wait 方法时间结束 / Object.notify / Object.notifyAll设置了 Timeout 参数的 Thread.join 方法时间结束 / 被调用的线程执行完毕LockSupport.parkNanos 方法LockSupport.unparkLockSupport.parkUntil 方法LockSupport.unpark 终止(Terminated) - 线程执行完 run 方法或者因异常退出了 run 方法。此状态意味着线程结束了生命周期。 线程常见问题 sleep、yield、join 方法有什么区别 yield 方法 yield 方法会 让线程从 Running 状态转入 Runnable 状态。当调用了 yield 方法后只有与当前线程相同或更高优先级的Runnable 状态线程才会获得执行的机会。 sleep 方法 sleep 方法会 让线程从 Running 状态转入 Waiting 状态。sleep 方法需要指定等待的时间超过等待时间后JVM 会将线程从 Waiting 状态转入 Runnable 状态。当调用了 sleep 方法后无论什么优先级的线程都可以得到执行机会。sleep 方法不会释放“锁标志”也就是说如果有 synchronized 同步块其他线程仍然不能访问共享数据。 join join 方法会 让线程从 Running 状态转入 Waiting 状态。当调用了 join 方法后当前线程必须等待调用 join 方法的线程结束后才能继续执行。 为什么 sleep 和 yield 方法是静态的 Thread 类的 sleep 和 yield 方法将处理 Running 状态的线程。 所以在其他处于非 Running 状态的线程上执行这两个方法是没有意义的。这就是为什么这些方法是静态的。它们可以在当前正在执行的线程中工作并避免程序员错误的认为可以在其他非运行线程调用这些方法。 Java 线程是否按照线程优先级严格执行 即使设置了线程的优先级也无法保证高优先级的线程一定先执行。 原因在于线程优先级依赖于操作系统的支持然而不同的操作系统支持的线程优先级并不相同不能很好的和 Java 中线程优先级一一对应。 一个线程两次调用 start()方法会怎样 Java 的线程是不允许启动两次的第二次调用必然会抛出 IllegalThreadStateException这是一种运行时异常多次调用 start 被认为是编程错误。 start 和 run 方法有什么区别 run 方法是线程的执行体。start 方法会启动线程然后 JVM 会让这个线程去执行 run 方法。 可以直接调用 Thread 类的 run 方法么 可以。但是如果直接调用 Thread 的 run 方法它的行为就会和普通的方法一样。为了在新的线程中执行我们的代码必须使用 Thread 的 start 方法。 参考资料 进程和线程关系及区别sleep()wait()yield()和 join()方法的区别Java 并发编程线程间协作的两种方式wait、notify、notifyAll 和 ConditionJava 并发编程Callable、Future 和 FutureTaskJava 中守护线程的总结Java 并发
http://www.dnsts.com.cn/news/24934.html

相关文章:

  • 网站权限怎么设置做花馍网站
  • 北京网站设计优选刻珠海网站制作外包
  • 商丘做网站哪个好app开发的价格清单
  • 排名好的郑州网站建设做视频网站视频存放问题
  • 网络站点推广的方法有哪些怎样给企业做网站
  • 怎么做多语言网站找资料的免费网站
  • 过时的网站昆明装修公司
  • 服装网站建设策划书预期投入商城网站建设是 什么软件
  • 玩具网站开发背景建盏公司官方网站
  • 郑州巩义网站建设简单网站建设合同
  • 做效果图的方便的网站优秀响应式网站
  • 深圳网站制作公司哪家好天津北京网站建设公司
  • 政企网站建设网站布局方法分类
  • 静态单页网站wordpress网站的信息管理建设的必要性
  • 宁波其它区低价企业网站搭建哪家好河南省交通工程造价信息网
  • 网站的经典推广方法东莞网站建设企慕
  • 进一步强化网站建设企业网站推广方法有哪些
  • 外贸建站公司网页设计教程视频dw
  • wordpress做下载型网站深圳大型网站建设
  • 个人网站开发计划书山西省网站
  • 织梦模板网站好优化吗网站建设可自学吗
  • 云建站网址现代简约室内设计案例分析
  • 网站开发的安全问题中国企业信息公示网登录
  • 找人做购物网站河北省建设信息中心网站
  • 深圳企业网站制作哪个wordpress 增加注册页面
  • 做网站客户要求多很烦济南 网站设计公司
  • wix做网站步骤个人主题网站做的步骤
  • 电子平台网站加盟网站制作费用
  • 南城做网站中天钢铁 网站建设
  • 西安云众网站建设厦门网站建设培训费用